summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--chrome_frame/CFInstall.js222
-rw-r--r--chrome_frame/CFInstance.js1656
-rw-r--r--chrome_frame/DEPS6
-rw-r--r--chrome_frame/bho.cc247
-rw-r--r--chrome_frame/bho.h102
-rw-r--r--chrome_frame/bho.rgs23
-rw-r--r--chrome_frame/chrome_active_document.bmpbin0 -> 246 bytes
-rw-r--r--chrome_frame/chrome_active_document.cc648
-rw-r--r--chrome_frame/chrome_active_document.h258
-rw-r--r--chrome_frame/chrome_active_document.rgs67
-rw-r--r--chrome_frame/chrome_frame.gyp725
-rw-r--r--chrome_frame/chrome_frame_activex.cc505
-rw-r--r--chrome_frame/chrome_frame_activex.h140
-rw-r--r--chrome_frame/chrome_frame_activex.rgs79
-rw-r--r--chrome_frame/chrome_frame_activex_base.h848
-rw-r--r--chrome_frame/chrome_frame_automation.cc975
-rw-r--r--chrome_frame/chrome_frame_automation.h356
-rw-r--r--chrome_frame/chrome_frame_delegate.cc67
-rw-r--r--chrome_frame/chrome_frame_delegate.h99
-rw-r--r--chrome_frame/chrome_frame_histograms.cc81
-rw-r--r--chrome_frame/chrome_frame_histograms.h54
-rw-r--r--chrome_frame/chrome_frame_npapi.cc1462
-rw-r--r--chrome_frame/chrome_frame_npapi.h333
-rw-r--r--chrome_frame/chrome_frame_npapi.rgs13
-rw-r--r--chrome_frame/chrome_frame_npapi_entrypoints.cc53
-rw-r--r--chrome_frame/chrome_frame_npapi_unittest.cc551
-rw-r--r--chrome_frame/chrome_frame_plugin.h214
-rw-r--r--chrome_frame/chrome_frame_unittest_main.cc48
-rw-r--r--chrome_frame/chrome_launcher.cc125
-rw-r--r--chrome_frame/chrome_launcher.h45
-rw-r--r--chrome_frame/chrome_launcher_main.cc79
-rw-r--r--chrome_frame/chrome_launcher_unittest.cc49
-rw-r--r--chrome_frame/chrome_protocol.cc85
-rw-r--r--chrome_frame/chrome_protocol.h65
-rw-r--r--chrome_frame/chrome_protocol.rgs39
-rw-r--r--chrome_frame/chrome_tab.cc308
-rw-r--r--chrome_frame/chrome_tab.def15
-rw-r--r--chrome_frame/chrome_tab.idl153
-rw-r--r--chrome_frame/chrome_tab.rgs12
-rw-r--r--chrome_frame/chrome_tab_version.rc.version45
-rw-r--r--chrome_frame/com_message_event.cc157
-rw-r--r--chrome_frame/com_message_event.h75
-rw-r--r--chrome_frame/com_type_info_holder.cc123
-rw-r--r--chrome_frame/com_type_info_holder.h191
-rw-r--r--chrome_frame/combine_libs.py114
-rw-r--r--chrome_frame/common/extra_defines.vsprops12
-rw-r--r--chrome_frame/crash_report.cc144
-rw-r--r--chrome_frame/crash_report.h13
-rw-r--r--chrome_frame/extra_system_apis.h37
-rw-r--r--chrome_frame/ff_30_privilege_check.cc99
-rw-r--r--chrome_frame/ff_privilege_check.h14
-rw-r--r--chrome_frame/find_dialog.cc107
-rw-r--r--chrome_frame/find_dialog.h54
-rw-r--r--chrome_frame/frame.html20
-rw-r--r--chrome_frame/frame_w_controls.html29
-rw-r--r--chrome_frame/function_stub.h241
-rw-r--r--chrome_frame/host.html47
-rw-r--r--chrome_frame/host_w_controls.html97
-rw-r--r--chrome_frame/html_utils.cc281
-rw-r--r--chrome_frame/html_utils.h120
-rw-r--r--chrome_frame/icu_stubs.cc169
-rw-r--r--chrome_frame/ie8_types.h27
-rw-r--r--chrome_frame/iids.cc10
-rw-r--r--chrome_frame/in_place_menu.h231
-rw-r--r--chrome_frame/installer_util/test.txt1
-rw-r--r--chrome_frame/np_browser_functions.cc512
-rw-r--r--chrome_frame/np_browser_functions.h246
-rw-r--r--chrome_frame/np_event_listener.cc367
-rw-r--r--chrome_frame/np_event_listener.h187
-rw-r--r--chrome_frame/np_proxy_service.cc306
-rw-r--r--chrome_frame/np_proxy_service.h137
-rw-r--r--chrome_frame/npapi_url_request.cc157
-rw-r--r--chrome_frame/npapi_url_request.h45
-rw-r--r--chrome_frame/ns_associate_iid_win.h15
-rw-r--r--chrome_frame/ns_isupports_impl.h68
-rw-r--r--chrome_frame/ole_document_impl.h253
-rw-r--r--chrome_frame/plugin_url_request.cc65
-rw-r--r--chrome_frame/plugin_url_request.h114
-rw-r--r--chrome_frame/precompiled.cc9
-rw-r--r--chrome_frame/precompiled.h18
-rw-r--r--chrome_frame/protocol_sink_wrap.cc615
-rw-r--r--chrome_frame/protocol_sink_wrap.h221
-rw-r--r--chrome_frame/rename_me_to_supplement.gypi24
-rw-r--r--chrome_frame/resource.h19
-rw-r--r--chrome_frame/resources/chrome_frame_resources.grd55
-rw-r--r--chrome_frame/resources/chrome_frame_strings.grd40
-rw-r--r--chrome_frame/resources/structured_resources.rc24
-rw-r--r--chrome_frame/resources/tlb_resource.rc68
-rw-r--r--chrome_frame/scoped_ns_ptr_win.h149
-rw-r--r--chrome_frame/script_security_manager.h785
-rw-r--r--chrome_frame/support.gyp17
-rw-r--r--chrome_frame/sync_msg_reply_dispatcher.cc61
-rw-r--r--chrome_frame/sync_msg_reply_dispatcher.h96
-rw-r--r--chrome_frame/test/ChromeTab_UnitTests.vcproj273
-rw-r--r--chrome_frame/test/chrome_frame_automation_mock.cc47
-rw-r--r--chrome_frame/test/chrome_frame_automation_mock.h208
-rw-r--r--chrome_frame/test/chrome_frame_test_utils.cc416
-rw-r--r--chrome_frame/test/chrome_frame_test_utils.h61
-rw-r--r--chrome_frame/test/chrome_frame_unittests.cc1510
-rw-r--r--chrome_frame/test/chrome_frame_unittests.h164
-rw-r--r--chrome_frame/test/chrometab_unittests.vsprops14
-rw-r--r--chrome_frame/test/com_message_event_unittest.cc325
-rw-r--r--chrome_frame/test/data/CFInstance_basic_frame.html8
-rw-r--r--chrome_frame/test/data/CFInstance_basic_host.html55
-rw-r--r--chrome_frame/test/data/CFInstance_default_ctor_host.html45
-rw-r--r--chrome_frame/test/data/CFInstance_delay_host.html47
-rw-r--r--chrome_frame/test/data/CFInstance_fallback_host.html44
-rw-r--r--chrome_frame/test/data/CFInstance_iframe_onload_host.html41
-rw-r--r--chrome_frame/test/data/CFInstance_iframe_post_host.html50
-rw-r--r--chrome_frame/test/data/CFInstance_no_src_host.html43
-rw-r--r--chrome_frame/test/data/CFInstance_post_frame.html26
-rw-r--r--chrome_frame/test/data/CFInstance_post_host.html47
-rw-r--r--chrome_frame/test/data/CFInstance_rpc_frame.html27
-rw-r--r--chrome_frame/test/data/CFInstance_rpc_host.html51
-rw-r--r--chrome_frame/test/data/CFInstance_rpc_internal_frame.html28
-rw-r--r--chrome_frame/test/data/CFInstance_rpc_internal_host.html45
-rw-r--r--chrome_frame/test/data/CFInstance_singleton_frame.html20
-rw-r--r--chrome_frame/test/data/CFInstance_singleton_host.html44
-rw-r--r--chrome_frame/test/data/CFInstance_zero_size_host.html41
-rw-r--r--chrome_frame/test/data/back_to_ie.html21
-rw-r--r--chrome_frame/test/data/cf_protocol.html20
-rw-r--r--chrome_frame/test/data/chrome_frame_mime_filter_test.html32
-rw-r--r--chrome_frame/test/data/chrome_frame_resize.html138
-rw-r--r--chrome_frame/test/data/chrome_frame_resize_hosted.html48
-rw-r--r--chrome_frame/test/data/chrome_frame_tester_helpers.js142
-rw-r--r--chrome_frame/test/data/event_listener.html42
-rw-r--r--chrome_frame/test/data/iframe_basic_host.html13
-rw-r--r--chrome_frame/test/data/in_head.html62
-rw-r--r--chrome_frame/test/data/initialize_hidden.html120
-rw-r--r--chrome_frame/test/data/meta_tag.html21
-rw-r--r--chrome_frame/test/data/navigate_out.html20
-rw-r--r--chrome_frame/test/data/navigateurl_absolute_host.html64
-rw-r--r--chrome_frame/test/data/navigateurl_basic_frame.html12
-rw-r--r--chrome_frame/test/data/navigateurl_relative_host.html60
-rw-r--r--chrome_frame/test/data/persistent_cookie_test_page.html36
-rw-r--r--chrome_frame/test/data/postmessage_basic_frame.html27
-rw-r--r--chrome_frame/test/data/postmessage_basic_host.html69
-rw-r--r--chrome_frame/test/data/privileged_apis_frame.html33
-rw-r--r--chrome_frame/test/data/privileged_apis_host.html85
-rw-r--r--chrome_frame/test/data/simple_object_focus.html95
-rw-r--r--chrome_frame/test/data/simple_object_focus_cf.html10
-rw-r--r--chrome_frame/test/data/src_property_frame1.html13
-rw-r--r--chrome_frame/test/data/src_property_frame2.html8
-rw-r--r--chrome_frame/test/data/src_property_host.html65
-rw-r--r--chrome_frame/test/data/version.html29
-rw-r--r--chrome_frame/test/function_stub_unittest.cc66
-rw-r--r--chrome_frame/test/helper_gmock.h597
-rw-r--r--chrome_frame/test/html_util_test_data/basic_test.html11
-rw-r--r--chrome_frame/test/html_util_test_data/degenerate_cases_test.html7
-rw-r--r--chrome_frame/test/html_util_test_data/multiple_tags.html17
-rw-r--r--chrome_frame/test/html_util_test_data/quotes_test.html10
-rw-r--r--chrome_frame/test/html_util_unittests.cc215
-rw-r--r--chrome_frame/test/http_server.cc56
-rw-r--r--chrome_frame/test/http_server.h32
-rw-r--r--chrome_frame/test/icu_stubs_unittests.cc73
-rw-r--r--chrome_frame/test/net/dialog_watchdog.cc146
-rw-r--r--chrome_frame/test/net/dialog_watchdog.h64
-rw-r--r--chrome_frame/test/net/fake_external_tab.cc391
-rw-r--r--chrome_frame/test/net/fake_external_tab.h106
-rw-r--r--chrome_frame/test/net/process_singleton_subclass.cc111
-rw-r--r--chrome_frame/test/net/process_singleton_subclass.h36
-rw-r--r--chrome_frame/test/net/test_automation_provider.cc89
-rw-r--r--chrome_frame/test/net/test_automation_provider.h52
-rw-r--r--chrome_frame/test/net/test_automation_resource_message_filter.cc60
-rw-r--r--chrome_frame/test/net/test_automation_resource_message_filter.h50
-rw-r--r--chrome_frame/test/perf/chrome_frame_perftest.cc1137
-rw-r--r--chrome_frame/test/perf/chrome_frame_perftest.h21
-rw-r--r--chrome_frame/test/perf/chrometab_perftests.vcproj247
-rw-r--r--chrome_frame/test/perf/chrometab_perftests.vsprops14
-rw-r--r--chrome_frame/test/perf/run_all.cc17
-rw-r--r--chrome_frame/test/perf/silverlight.cc165
-rw-r--r--chrome_frame/test/run_all_unittests.cc49
-rw-r--r--chrome_frame/test/test_server.cc211
-rw-r--r--chrome_frame/test/test_server.h296
-rw-r--r--chrome_frame/test/test_server_test.cc194
-rw-r--r--chrome_frame/test/util_unittests.cc120
-rw-r--r--chrome_frame/test_utils.cc95
-rw-r--r--chrome_frame/test_utils.h39
-rw-r--r--chrome_frame/tools/test/page_cycler/cf_cycler.py99
-rw-r--r--chrome_frame/tools/test/page_cycler/urllist500
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/First Run1
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/avcodec-52.dllbin0 -> 686592 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/avformat-52.dllbin0 -> 95744 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/avutil-50.dllbin0 -> 55808 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/chrome.dllbin0 -> 16211456 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/chrome.exebin0 -> 438784 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/chrome_dll.pdbbin0 -> 130362368 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/chrome_exe.pdbbin0 -> 4557824 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/crash_service.exebin0 -> 272896 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/gears.dllbin0 -> 3174400 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/icudt42.dllbin0 -> 10899456 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ar.dllbin0 -> 94208 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/bg.dllbin0 -> 108032 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/bn.dllbin0 -> 99328 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ca.dllbin0 -> 106496 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/cs.dllbin0 -> 100864 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/da.dllbin0 -> 99840 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/de.dllbin0 -> 106496 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/el.dllbin0 -> 113664 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/en-GB.dllbin0 -> 94208 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/en-US.dllbin0 -> 94208 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/es-419.dllbin0 -> 106496 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/es.dllbin0 -> 106496 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/et.dllbin0 -> 96768 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/fi.dllbin0 -> 98816 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/fil.dllbin0 -> 110080 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/fr.dllbin0 -> 109056 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/gu.dllbin0 -> 96768 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/he.dllbin0 -> 83968 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/hi.dllbin0 -> 100352 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/hr.dllbin0 -> 100352 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/hu.dllbin0 -> 105472 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/id.dllbin0 -> 97280 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/it.dllbin0 -> 103936 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ja.dllbin0 -> 70144 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/kn.dllbin0 -> 103936 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ko.dllbin0 -> 69120 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/lt.dllbin0 -> 102912 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/lv.dllbin0 -> 99328 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ml.dllbin0 -> 115712 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/mr.dllbin0 -> 96768 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/nb.dllbin0 -> 98304 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/nl.dllbin0 -> 102912 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/or.dllbin0 -> 108032 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/pl.dllbin0 -> 102912 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/pt-BR.dllbin0 -> 101888 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/pt-PT.dllbin0 -> 104960 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ro.dllbin0 -> 106496 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ru.dllbin0 -> 103936 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/sk.dllbin0 -> 103936 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/sl.dllbin0 -> 99840 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/sr.dllbin0 -> 101888 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/sv.dllbin0 -> 97792 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/ta.dllbin0 -> 108544 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/te.dllbin0 -> 103424 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/th.dllbin0 -> 93184 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/tr.dllbin0 -> 99328 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/uk.dllbin0 -> 102400 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/vi.dllbin0 -> 97280 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/zh-CN.dllbin0 -> 59392 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/locales/zh-TW.dllbin0 -> 59904 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/mini_installer.pdbbin0 -> 125952 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/BottomUpProfileDataGridTree.js252
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Breakpoint.js96
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/BreakpointsSidebarPane.js167
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/CallStackSidebarPane.js166
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Callback.js56
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ChangesView.js80
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Color.js661
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ConsoleView.js970
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/CookieItemsView.js266
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMAgent.js650
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorage.js72
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageDataGrid.js165
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageItemsView.js112
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/DataGrid.js1041
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Database.js113
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseQueryView.js195
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseTableView.js89
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Drawer.js202
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsPanel.js1045
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsTreeOutline.js742
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/FontView.js104
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ImageView.js74
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScript.js1133
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScriptAccess.js78
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/KeyboardShortcut.js108
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/MetricsSidebarPane.js215
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Object.js82
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectPropertiesSection.js258
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectProxy.js44
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Panel.js273
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/PanelEnablerView.js97
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Placard.js106
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Popup.js168
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileDataGridTree.js398
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileView.js539
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfilesPanel.js534
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSection.js145
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSidebarPane.js64
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Resource.js622
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceCategory.js68
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceView.js263
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourcesPanel.js1462
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScopeChainSidebarPane.js146
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/Script.js50
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptView.js103
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptsPanel.js918
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarPane.js125
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarTreeElement.js201
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceFrame.js906
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceView.js311
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/StatusBarButton.js118
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/StoragePanel.js685
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/StylesSidebarPane.js1373
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/SummaryBar.js364
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/TextPrompt.js319
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/TimelineAgent.js54
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/TopDownProfileDataGridTree.js111
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/View.js74
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/WatchExpressionsSidebarPane.js274
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/base.js1015
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/codemap.js258
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/consarray.js93
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/csvparser.js98
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/debugger_agent.js1490
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.css99
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.html142
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.js392
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_callback.js57
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_host_stub.js335
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/heap_profiler_panel.js680
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/back.pngbin0 -> 4205 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/checker.pngbin0 -> 3471 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/clearConsoleButtonGlyph.pngbin0 -> 396 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/closeButtons.pngbin0 -> 4355 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/consoleButtonGlyph.pngbin0 -> 183 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/cookie.pngbin0 -> 2246 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/database.pngbin0 -> 2329 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/databaseTable.pngbin0 -> 4325 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerContinue.pngbin0 -> 4190 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerPause.pngbin0 -> 4081 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepInto.pngbin0 -> 4282 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOut.pngbin0 -> 4271 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOver.pngbin0 -> 4366 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDown.pngbin0 -> 3919 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownBlack.pngbin0 -> 3802 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownWhite.pngbin0 -> 3820 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRight.pngbin0 -> 3898 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightBlack.pngbin0 -> 3807 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDown.pngbin0 -> 3953 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownBlack.pngbin0 -> 3816 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownWhite.pngbin0 -> 3838 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightWhite.pngbin0 -> 3818 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/dockButtonGlyph.pngbin0 -> 164 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/elementsIcon.pngbin0 -> 6639 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableOutlineButtonGlyph.pngbin0 -> 363 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableSolidButtonGlyph.pngbin0 -> 302 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorIcon.pngbin0 -> 4337 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorMediumIcon.pngbin0 -> 4059 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/excludeButtonGlyph.pngbin0 -> 212 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/focusButtonGlyph.pngbin0 -> 285 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/forward.pngbin0 -> 4202 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeader.pngbin0 -> 3720 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderPressed.pngbin0 -> 3721 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelected.pngbin0 -> 3738 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelectedPressed.pngbin0 -> 3739 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/goArrow.pngbin0 -> 3591 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutLeft.pngbin0 -> 3790 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutRight.pngbin0 -> 3789 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/largerResourcesButtonGlyph.pngbin0 -> 192 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/localStorage.pngbin0 -> 1081 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/nodeSearchButtonGlyph.pngbin0 -> 283 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrow.pngbin0 -> 3457 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrowActive.pngbin0 -> 3457 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneGrowHandleLine.pngbin0 -> 3443 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/pauseOnExceptionButtonGlyph.pngbin0 -> 331 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/percentButtonGlyph.pngbin0 -> 357 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileGroupIcon.pngbin0 -> 5126 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileIcon.pngbin0 -> 4953 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileSmallIcon.pngbin0 -> 579 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesIcon.pngbin0 -> 4158 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesSilhouette.pngbin0 -> 48600 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/radioDot.pngbin0 -> 235 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordButtonGlyph.pngbin0 -> 213 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordToggledButtonGlyph.pngbin0 -> 510 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/reloadButtonGlyph.pngbin0 -> 267 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceCSSIcon.pngbin0 -> 1066 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIcon.pngbin0 -> 4959 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIconSmall.pngbin0 -> 787 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceJSIcon.pngbin0 -> 879 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIcon.pngbin0 -> 4321 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIconSmall.pngbin0 -> 731 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesIcon.pngbin0 -> 6431 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSilhouette.pngbin0 -> 42925 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSizeGraphIcon.pngbin0 -> 5606 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesTimeGraphIcon.pngbin0 -> 5743 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsIcon.pngbin0 -> 7428 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsSilhouette.pngbin0 -> 49028 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBlue.pngbin0 -> 3968 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBrightBlue.pngbin0 -> 3966 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallGray.pngbin0 -> 3936 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallWhite.pngbin0 -> 3844 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segment.pngbin0 -> 4349 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentEnd.pngbin0 -> 4070 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHover.pngbin0 -> 4310 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHoverEnd.pngbin0 -> 4074 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelected.pngbin0 -> 4302 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelectedEnd.pngbin0 -> 4070 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/sessionStorage.pngbin0 -> 1097 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDimple.pngbin0 -> 216 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDividerBackground.pngbin0 -> 149 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBackground.pngbin0 -> 4024 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBottomBackground.pngbin0 -> 4021 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarButtons.pngbin0 -> 4175 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButton.pngbin0 -> 4293 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButtonSelected.pngbin0 -> 4291 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerHorizontal.pngbin0 -> 4026 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerVertical.pngbin0 -> 4036 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/storageIcon.pngbin0 -> 7148 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillBlue.pngbin0 -> 3450 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGray.pngbin0 -> 3392 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGreen.pngbin0 -> 3452 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillOrange.pngbin0 -> 3452 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillPurple.pngbin0 -> 3453 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillRed.pngbin0 -> 3460 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillYellow.pngbin0 -> 3444 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillBlue.pngbin0 -> 3346 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGray.pngbin0 -> 3297 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGreen.pngbin0 -> 3350 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillOrange.pngbin0 -> 3352 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillPurple.pngbin0 -> 3353 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillRed.pngbin0 -> 3343 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillYellow.pngbin0 -> 3336 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloon.pngbin0 -> 3689 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloonBottom.pngbin0 -> 3139 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIcon.pngbin0 -> 1212 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIconPressed.pngbin0 -> 1224 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/toolbarItemSelected.pngbin0 -> 4197 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleBlack.pngbin0 -> 3570 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleWhite.pngbin0 -> 3531 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleBlack.pngbin0 -> 3561 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleWhite.pngbin0 -> 3535 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleBlack.pngbin0 -> 3584 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleWhite.pngbin0 -> 3558 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/undockButtonGlyph.pngbin0 -> 179 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputIcon.pngbin0 -> 777 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputPreviousIcon.pngbin0 -> 765 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputResultIcon.pngbin0 -> 259 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningIcon.pngbin0 -> 4244 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningMediumIcon.pngbin0 -> 3833 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningsErrors.pngbin0 -> 5192 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/inject.js50
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.css3302
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.html112
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.js1553
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller.js506
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller_impl.js286
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/logreader.js320
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile.js621
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile_view.js224
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/profiler_processor.js449
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/splaytree.js322
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/tests.js628
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/treeoutline.js849
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/resources/inspector/utilities.js905
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.exebin0 -> 3584 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.pdbbin0 -> 125952 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.dllbin0 -> 783872 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.pdbbin0 -> 10284032 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/setup.pdbbin0 -> 6188032 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/syncapi.dllbin0 -> 2271744 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/themes/default.dllbin0 -> 508928 bytes
-rw-r--r--chrome_frame/tools/test/reference_build/chrome/wow_helper.exebin0 -> 67072 bytes
-rw-r--r--chrome_frame/unittest_precompile.cc5
-rw-r--r--chrome_frame/unittest_precompile.h18
-rw-r--r--chrome_frame/urlmon_upload_data_stream.cc99
-rw-r--r--chrome_frame/urlmon_upload_data_stream.h89
-rw-r--r--chrome_frame/urlmon_upload_data_stream_unittest.cc162
-rw-r--r--chrome_frame/urlmon_url_request.cc751
-rw-r--r--chrome_frame/urlmon_url_request.h231
-rw-r--r--chrome_frame/utils.cc643
-rw-r--r--chrome_frame/utils.h227
-rw-r--r--chrome_frame/vectored_handler-impl.h106
-rw-r--r--chrome_frame/vectored_handler.h87
-rw-r--r--chrome_frame/vtable_patch_manager.cc82
-rw-r--r--chrome_frame/vtable_patch_manager.h64
467 files changed, 65754 insertions, 0 deletions
diff --git a/chrome_frame/CFInstall.js b/chrome_frame/CFInstall.js
new file mode 100644
index 0000000..915f958
--- /dev/null
+++ b/chrome_frame/CFInstall.js
@@ -0,0 +1,222 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview CFInstall.js provides a set of utilities for managing
+ * the Chrome Frame detection and installation process.
+ * @author slightlyoff@google.com (Alex Russell)
+ */
+
+(function(scope) {
+ // bail if we'd be over-writing an existing CFInstall object
+ if (scope['CFInstall']) {
+ return;
+ }
+
+ /**
+ * returns an item based on DOM ID. Optionally a document may be provided to
+ * specify the scope to search in. If a node is passed, it's returned as-is.
+ * @param {string|Node} id The ID of the node to be located or a node
+ * @param {Node} doc Optional A document to search for id.
+ * @return {Node}
+ */
+ var byId = function(id, doc) {
+ return (typeof id == 'string') ? (doc || document).getElementById(id) : id;
+ };
+
+ /////////////////////////////////////////////////////////////////////////////
+ // Plugin Detection
+ /////////////////////////////////////////////////////////////////////////////
+
+ var cachedAvailable;
+
+ /**
+ * Checks to find out if ChromeFrame is available as a plugin
+ * @return {Boolean}
+ */
+ var isAvailable = function() {
+ if (typeof cachedAvailable != 'undefined') {
+ return cachedAvailable;
+ }
+
+ cachedAvailable = false;
+
+ // Look for CF in the User Agent before trying more expensive checks
+ var ua = navigator.userAgent.toLowerCase();
+ if (ua.indexOf("chromeframe") >= 0 || ua.indexOf("x-clock") >= 0) {
+ cachedAvailable = true;
+ return cachedAvailable;
+ }
+
+ if (typeof window['ActiveXObject'] != 'undefined') {
+ try {
+ var obj = new ActiveXObject('ChromeTab.ChromeFrame');
+ if (obj) {
+ cachedAvailable = true;
+ }
+ } catch(e) {
+ // squelch
+ }
+ }
+ return cachedAvailable;
+ };
+
+
+ /** @type {boolean} */
+ var cfStyleTagInjected = false;
+
+ /**
+ * Creates a style sheet in the document which provides default styling for
+ * ChromeFrame instances. Successive calls should have no additive effect.
+ */
+ var injectCFStyleTag = function() {
+ if (cfStyleTagInjected) {
+ // Once and only once
+ return;
+ }
+ try {
+ var rule = '.chromeFrameInstallDefaultStyle {' +
+ 'width: 500px;' +
+ 'height: 400px;' +
+ 'padding: 0;' +
+ 'border: 1px solid #0028c4;' +
+ 'margin: 0;' +
+ '}';
+ var ss = document.createElement('style');
+ ss.setAttribute('type', 'text/css');
+ if (ss.styleSheet) {
+ ss.styleSheet.cssText = rule;
+ } else {
+ ss.appendChild(document.createTextNode(rule));
+ }
+ var h = document.getElementsByTagName('head')[0];
+ var firstChild = h.firstChild;
+ h.insertBefore(ss, firstChild);
+ cfStyleTagInjected = true;
+ } catch (e) {
+ // squelch
+ }
+ };
+
+
+ /**
+ * Plucks properties from the passed arguments and sets them on the passed
+ * DOM node
+ * @param {Node} node The node to set properties on
+ * @param {Object} args A map of user-specified properties to set
+ */
+ var setProperties = function(node, args) {
+ injectCFStyleTag();
+
+ var srcNode = byId(args['node']);
+
+ node.id = args['id'] || (srcNode ? srcNode['id'] || getUid(srcNode) : '');
+
+ // TODO(slightlyoff): Opera compat? need to test there
+ var cssText = args['cssText'] || '';
+ node.style.cssText = ' ' + cssText;
+
+ var classText = args['className'] || '';
+ node.className = 'chromeFrameInstallDefaultStyle ' + classText;
+
+ // default if the browser doesn't so we don't show sad-tab
+ var src = args['src'] || 'about:blank';
+
+ node.src = src;
+
+ if (srcNode) {
+ srcNode.parentNode.replaceChild(node, srcNode);
+ }
+ };
+
+ /**
+ * Creates an iframe.
+ * @param {Object} args A bag of configuration properties, including values
+ * like 'node', 'cssText', 'className', 'id', 'src', etc.
+ * @return {Node}
+ */
+ var makeIframe = function(args) {
+ var el = document.createElement('iframe');
+ setProperties(el, args);
+ return el;
+ };
+
+ var CFInstall = {};
+ /**
+ * Checks to see if Chrome Frame is available, if not, prompts the user to
+ * install. Once installation is begun, a background timer starts,
+ * checkinging for a successful install every 2 seconds. Upon detection of
+ * successful installation, the current page is reloaded, or if a
+ * 'destination' parameter is passed, the page navigates there instead.
+ * @param {Object} args A bag of configuration properties. Respected
+ * properties are: 'mode', 'url', 'destination', 'node', 'onmissing',
+ * 'preventPrompt', 'oninstall', 'preventInstallDetection', 'cssText', and
+ * 'className'.
+ * @public
+ */
+ CFInstall.check = function(args) {
+ args = args || {};
+
+ // We currently only support CF in IE
+ // TODO(slightlyoff): Update this should we support other browsers!
+ var ieRe = /MSIE (\S+)/;
+ if (!ieRe.test(navigator.userAgent)) {
+ return;
+ }
+
+
+ // TODO(slightlyoff): Update this URL when a mini-installer page is
+ // available.
+ var installUrl = '//www.google.com/chromeframe';
+ if (!isAvailable()) {
+ if (args.onmissing) {
+ args.onmissing();
+ }
+
+ args.src = args.url || installUrl;
+ var mode = args.mode || 'inline';
+ var preventPrompt = args.preventPrompt || false;
+
+ if (!preventPrompt) {
+ if (mode == 'inline') {
+ var ifr = makeIframe(args);
+ // TODO(slightlyoff): handle placement more elegantly!
+ if (!ifr.parentNode) {
+ var firstChild = document.body.firstChild;
+ document.body.insertBefore(ifr, firstChild);
+ }
+ } else {
+ window.open(args.src);
+ }
+ }
+
+ if (args.preventInstallDetection) {
+ return;
+ }
+
+ // Begin polling for install success.
+ var installTimer = setInterval(function() {
+ // every 2 seconds, look to see if CF is available, if so, proceed on
+ // to our destination
+ if (isAvailable()) {
+ if (args.oninstall) {
+ args.oninstall();
+ }
+
+ clearInterval(installTimer);
+ // TODO(slightlyoff): add a way to prevent navigation or make it
+ // contingent on oninstall?
+ window.location = args.destination || window.location;
+ }
+ }, 2000);
+ }
+ };
+
+ CFInstall.isAvailable = isAvailable;
+
+ // expose CFInstall to the external scope. We've already checked to make
+ // sure we're not going to blow existing objects away.
+ scope.CFInstall = CFInstall;
+
+})(this['ChromeFrameInstallScope'] || this);
diff --git a/chrome_frame/CFInstance.js b/chrome_frame/CFInstance.js
new file mode 100644
index 0000000..90a28f5
--- /dev/null
+++ b/chrome_frame/CFInstance.js
@@ -0,0 +1,1656 @@
+// Copyright (c) 2009 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.
+
+// Parts Copyright 2005-2009, the Dojo Foundation. Used under the terms of the
+// "New" BSD License:
+//
+// http://download.dojotoolkit.org/release-1.3.2/dojo-release-1.3.2/dojo/LICENSE
+//
+
+/**
+ * @fileoverview CFInstance.js provides a set of utilities for managing
+ * ChromeFrame plugins, including creation, RPC services, and a singleton to
+ * use for communicating from ChromeFrame hosted content to an external
+ * CFInstance wrapper. CFInstance.js is stand-alone, designed to be served from
+ * a CDN, and built to not create side-effects for other hosted content.
+ * @author slightlyoff@google.com (Alex Russell)
+ */
+
+(function(scope) {
+ // TODO:
+ // * figure out if there's any way to install w/o a browser restart, and if
+ // so, where and how
+ // * slim down Deferred and RPC scripts
+ // * determine what debugging APIs should be exposed and how they should be
+ // surfaced. What about content authoring in Chrome instances? Stubbing
+ // the other side of RPC's?
+
+ // bail if we'd be over-writing an existing CFInstance object
+ if (scope['CFInstance']) {
+ return;
+ }
+
+ /////////////////////////////////////////////////////////////////////////////
+ // Utiliity and Cross-Browser Functions
+ /////////////////////////////////////////////////////////////////////////////
+
+ // a monotonically incrementing counter
+ var _counter = 0;
+
+ var undefStr = 'undefined';
+
+ //
+ // Browser detection: ua.isIE, ua.isSafari, ua.isOpera, etc.
+ //
+
+ /**
+ * An object for User Agent detection
+ * @type {!Object}
+ * @protected
+ */
+ var ua = {};
+ var n = navigator;
+ var dua = String(n.userAgent);
+ var dav = String(n.appVersion);
+ var tv = parseFloat(dav);
+ var duaParse = function(s){
+ var c = 0;
+ try {
+ return parseFloat(
+ dua.split(s)[1].replace(/\./g, function() {
+ c++;
+ return (c > 1) ? '' : '.';
+ } )
+ );
+ } catch(e) {
+ // squelch to intentionally return undefined
+ }
+ };
+ /** @type {number} */
+ ua.isOpera = dua.indexOf('Opera') >= 0 ? tv: undefined;
+ /** @type {number} */
+ ua.isWebKit = duaParse('WebKit/');
+ /** @type {number} */
+ ua.isChrome = duaParse('Chrome/');
+ /** @type {number} */
+ ua.isKhtml = dav.indexOf('KHTML') >= 0 ? tv : undefined;
+
+ var index = Math.max(dav.indexOf('WebKit'), dav.indexOf('Safari'), 0);
+
+ if (index && !ua.isChrome) {
+ /** @type {number} */
+ ua.isSafari = parseFloat(dav.split('Version/')[1]);
+ if(!ua.isSafari || parseFloat(dav.substr(index + 7)) <= 419.3){
+ ua.isSafari = 2;
+ }
+ }
+
+ if (dua.indexOf('Gecko') >= 0 && !ua.isKhtml) {
+ /** @type {number} */
+ ua.isGecko = duaParse(' rv:');
+ }
+
+ if (ua.isGecko) {
+ /** @type {number} */
+ ua.isFF = parseFloat(dua.split('Firefox/')[1]) || undefined;
+ }
+
+ if (document.all && !ua.isOpera) {
+ /** @type {number} */
+ ua.isIE = parseFloat(dav.split('MSIE ')[1]) || undefined;
+ }
+
+
+ /**
+ * Log out varargs to a browser-provided console object (if available). Else
+ * a no-op.
+ * @param {*} var_args Optional Things to log.
+ * @protected
+ **/
+ var log = function() {
+ if (window['console']) {
+ try {
+ if (ua.isSafari || ua.isChrome) {
+ throw Error();
+ }
+ console.log.apply(console, arguments);
+ } catch(e) {
+ try {
+ console.log(toArray(arguments).join(' '));
+ } catch(e2) {
+ // squelch
+ }
+ }
+ }
+ };
+
+ //
+ // Language utility methods
+ //
+
+ /**
+ * Determine if the passed item is a String
+ * @param {*} item Item to test.
+ * @protected
+ **/
+ var isString = function(item) {
+ return typeof item == 'string';
+ };
+
+ /**
+ * Determine if the passed item is a Function object
+ * @param {*} item Item to test.
+ * @protected
+ **/
+ var isFunction = function(item) {
+ return (
+ item && (
+ typeof item == 'function' || item instanceof Function
+ )
+ );
+ };
+
+ /**
+ * Determine if the passed item is an array.
+ * @param {*} item Item to test.
+ * @protected
+ **/
+ var isArray = function(item){
+ return (
+ item && (
+ item instanceof Array || (
+ typeof item == 'object' &&
+ typeof item.length != undefStr
+ )
+ )
+ );
+ };
+
+ /**
+ * A toArray version which takes advantage of builtins
+ * @param {*} obj The array-like object to convert to a real array.
+ * @param {number} opt_offset An index to being copying from in the source.
+ * @param {Array} opt_startWith An array to extend with elements of obj in
+ * lieu of creating a new array to return.
+ * @private
+ **/
+ var _efficientToArray = function(obj, opt_offset, opt_startWith){
+ return (opt_startWith || []).concat(
+ Array.prototype.slice.call(obj, opt_offset || 0 )
+ );
+ };
+
+ /**
+ * A version of toArray that iterates in lieu of using array generics.
+ * @param {*} obj The array-like object to convert to a real array.
+ * @param {number} opt_offset An index to being copying from in the source.
+ * @param {Array} opt_startWith An array to extend with elements of obj in
+ * @private
+ **/
+ var _slowToArray = function(obj, opt_offset, opt_startWith){
+ var arr = opt_startWith || [];
+ for(var x = opt_offset || 0; x < obj.length; x++){
+ arr.push(obj[x]);
+ }
+ return arr;
+ };
+
+ /**
+ * Converts an array-like object (e.g., an "arguments" object) to a real
+ * Array.
+ * @param {*} obj The array-like object to convert to a real array.
+ * @param {number} opt_offset An index to being copying from in the source.
+ * @param {Array} opt_startWith An array to extend with elements of obj in
+ * @protected
+ */
+ var toArray = ua.isIE ?
+ function(obj){
+ return (
+ obj.item ? _slowToArray : _efficientToArray
+ ).apply(this, arguments);
+ } :
+ _efficientToArray;
+
+ var _getParts = function(arr, obj, cb){
+ return [
+ isString(arr) ? arr.split('') : arr,
+ obj || window,
+ isString(cb) ? new Function('item', 'index', 'array', cb) : cb
+ ];
+ };
+
+ /**
+ * like JS1.6 Array.forEach()
+ * @param {Array} arr the array to iterate
+ * @param {function(Object, number, Array)} callback the method to invoke for
+ * each item in the array
+ * @param {function?} thisObject Optional a scope to use with callback
+ * @return {array} the original arr
+ * @protected
+ */
+ var forEach = function(arr, callback, thisObject) {
+ if(!arr || !arr.length){
+ return arr;
+ }
+ var parts = _getParts(arr, thisObject, callback);
+ // parts has a structure of:
+ // [
+ // array,
+ // scope,
+ // function
+ // ]
+ arr = parts[0];
+ for (var i = 0, l = arr.length; i < l; ++i) {
+ parts[2].call( parts[1], arr[i], i, arr );
+ }
+ return arr;
+ };
+
+ /**
+ * returns a new function bound to scope with a variable number of positional
+ * params pre-filled
+ * @private
+ */
+ var _hitchArgs = function(scope, method /*,...*/) {
+ var pre = toArray(arguments, 2);
+ var named = isString(method);
+ return function() {
+ var args = toArray(arguments);
+ var f = named ? (scope || window)[method] : method;
+ return f && f.apply(scope || this, pre.concat(args));
+ }
+ };
+
+ /**
+ * Like goog.bind(). Hitches the method (named or provided as a function
+ * object) to scope, optionally partially applying positional arguments.
+ * @param {Object} scope the object to hitch the method to
+ * @param {string|function} method the method to be bound
+ * @return {function} The bound method
+ * @protected
+ */
+ var hitch = function(scope, method){
+ if (arguments.length > 2) {
+ return _hitchArgs.apply(window, arguments); // Function
+ }
+
+ if (!method) {
+ method = scope;
+ scope = null;
+ }
+
+ if (isString(method)) {
+ scope = scope || window;
+ if (!scope[method]) {
+ throw(
+ ['scope["', method, '"] is null (scope="', scope, '")'].join('')
+ );
+ }
+ return function() {
+ return scope[method].apply(scope, arguments || []);
+ };
+ }
+
+ return !scope ?
+ method :
+ function() {
+ return method.apply(scope, arguments || []);
+ };
+ };
+
+ /**
+ * A version of addEventListener that works on IE too. *sigh*.
+ * @param {!Object} obj The object to attach to
+ * @param {!String} type Name of the event to attach to
+ * @param {!Function} handler The function to connect
+ * @protected
+ */
+ var listen = function(obj, type, handler) {
+ if (obj['attachEvent']) {
+ obj.attachEvent('on' + type, handler);
+ } else {
+ obj.addEventListener(type, handler, false);
+ }
+ };
+
+ /**
+ * Adds "listen" and "_dispatch" methods to the passed object, taking
+ * advantage of native event hanlding if it's available.
+ * @param {Object} instance The object to install the event system on
+ * @protected
+ */
+ var installEvtSys = function(instance) {
+ var eventsMap = {};
+
+ var isNative = (
+ (typeof instance.addEventListener != undefStr) &&
+ ((instance['tagName'] || '').toLowerCase() != 'iframe')
+ );
+
+ instance.listen = function(type, func) {
+ var t = eventsMap[type];
+ if (!t) {
+ t = eventsMap[type] = [];
+ if (isNative) {
+ listen(instance, type, hitch(instance, "_dispatch", type));
+ }
+ }
+ t.push(func);
+ return t;
+ };
+
+ instance._dispatch = function(type, evt) {
+ var stopped = false;
+ var stopper = function() {
+ stopped = true;
+ };
+
+ forEach(eventsMap[type], function(f) {
+ if (!stopped) {
+ f(evt, stopper);
+ }
+ });
+ };
+
+ return instance;
+ };
+
+ /**
+ * Deserialize the passed JSON string
+ * @param {!String} json A string to be deserialized
+ * @return {Object}
+ * @protected
+ */
+ var fromJson = window['JSON'] ? function(json) {
+ return JSON.parse(json);
+ } :
+ function(json) {
+ return eval('(' + (json || undefStr) + ')');
+ };
+
+ /**
+ * String escaping for use in JSON serialization
+ * @param {string} str The string to escape
+ * @return {string}
+ * @private
+ */
+ var _escapeString = function(str) {
+ return ('"' + str.replace(/(["\\])/g, '\\$1') + '"').
+ replace(/[\f]/g, '\\f').
+ replace(/[\b]/g, '\\b').
+ replace(/[\n]/g, '\\n').
+ replace(/[\t]/g, '\\t').
+ replace(/[\r]/g, '\\r').
+ replace(/[\x0B]/g, '\\u000b'); // '\v' is not supported in JScript;
+ };
+
+ /**
+ * JSON serialization for arbitrary objects. Circular references or strong
+ * typing information are not handled.
+ * @param {Object} it Any valid JavaScript object or type
+ * @return {string} the serialized representation of the passed object
+ * @protected
+ */
+ var toJson = window['JSON'] ? function(it) {
+ return JSON.stringify(it);
+ } :
+ function(it) {
+
+ if (it === undefined) {
+ return undefStr;
+ }
+
+ var objtype = typeof it;
+ if (objtype == 'number' || objtype == 'boolean') {
+ return it + '';
+ }
+
+ if (it === null) {
+ return 'null';
+ }
+
+ if (isString(it)) {
+ return _escapeString(it);
+ }
+
+ // recurse
+ var recurse = arguments.callee;
+
+ if(it.nodeType && it.cloneNode){ // isNode
+ // we can't seriailize DOM nodes as regular objects because they have
+ // cycles DOM nodes could be serialized with something like outerHTML,
+ // but that can be provided by users in the form of .json or .__json__
+ // function.
+ throw new Error('Cannot serialize DOM nodes');
+ }
+
+ // array
+ if (isArray(it)) {
+ var res = [];
+ forEach(it, function(obj) {
+ var val = recurse(obj);
+ if (typeof val != 'string') {
+ val = undefStr;
+ }
+ res.push(val);
+ });
+ return '[' + res.join(',') + ']';
+ }
+
+ if (objtype == 'function') {
+ return null;
+ }
+
+ // generic object code path
+ var output = [];
+ for (var key in it) {
+ var keyStr, val;
+ if (typeof key == 'number') {
+ keyStr = '"' + key + '"';
+ } else if(typeof key == 'string') {
+ keyStr = _escapeString(key);
+ } else {
+ // skip non-string or number keys
+ continue;
+ }
+ val = recurse(it[key]);
+ if (typeof val != 'string') {
+ // skip non-serializable values
+ continue;
+ }
+ // TODO(slightlyoff): use += on Moz since it's faster there
+ output.push(keyStr + ':' + val);
+ }
+ return '{' + output.join(',') + '}'; // String
+ };
+
+ // code to register with the earliest safe onload-style handler
+
+ var _loadedListenerList = [];
+ var _loadedFired = false;
+
+ /**
+ * a default handler for document onload. When called (the first time),
+ * iterates over the list of registered listeners, calling them in turn.
+ * @private
+ */
+ var documentLoaded = function() {
+ if (!_loadedFired) {
+ _loadedFired = true;
+ forEach(_loadedListenerList, 'item();');
+ }
+ };
+
+ if (document.addEventListener) {
+ // NOTE:
+ // due to a threading issue in Firefox 2.0, we can't enable
+ // DOMContentLoaded on that platform. For more information, see:
+ // http://trac.dojotoolkit.org/ticket/1704
+ if (ua.isWebKit > 525 || ua.isOpera || ua.isFF >= 3) {
+ listen(document, 'DOMContentLoaded', documentLoaded);
+ }
+ // mainly for Opera 8.5, won't be fired if DOMContentLoaded fired already.
+ // also used for FF < 3.0 due to nasty DOM race condition
+ listen(window, 'load', documentLoaded);
+ } else {
+ // crazy hack for IE that relies on the "deferred" behavior of script
+ // tags
+ document.write(
+ '<scr' + 'ipt defer src="//:" '
+ + 'onreadystatechange="if(this.readyState==\'complete\')'
+ + '{ CFInstance._documentLoaded();}">'
+ + '</scr' + 'ipt>'
+ );
+ }
+
+ // TODO(slightlyoff): known KHTML init issues are ignored for now
+
+ //
+ // DOM utility methods
+ //
+
+ /**
+ * returns an item based on DOM ID. Optionally a doucment may be provided to
+ * specify the scope to search in. If a node is passed, it's returned as-is.
+ * @param {string|Node} id The ID of the node to be located or a node
+ * @param {Node} doc Optional A document to search for id.
+ * @return {Node}
+ * @protected
+ */
+ var byId = (ua.isIE || ua.isOpera) ?
+ function(id, doc) {
+ if (isString(id)) {
+ doc = doc || document;
+ var te = doc.getElementById(id);
+ // attributes.id.value is better than just id in case the
+ // user has a name=id inside a form
+ if (te && te.attributes.id.value == id) {
+ return te;
+ } else {
+ var elements = doc.all[id];
+ if (!elements || !elements.length) {
+ return elements;
+ }
+ // if more than 1, choose first with the correct id
+ var i=0;
+ while (te = elements[i++]) {
+ if (te.attributes.id.value == id) {
+ return te;
+ }
+ }
+ }
+ } else {
+ return id; // DomNode
+ }
+ } :
+ function(id, doc) {
+ return isString(id) ? (doc || document).getElementById(id) : id;
+ };
+
+
+ /**
+ * returns a unique DOM id which can be used to locate the node via byId().
+ * If the node already has an ID, it's used. If not, one is generated. Like
+ * IE's uniqueID property.
+ * @param {Node} node The element to create or fetch a unique ID for
+ * @return {String}
+ * @protected
+ */
+ var getUid = function(node) {
+ var u = 'cfUnique' + (_counter++);
+ return (!node) ? u : ( node.id || node.uniqueID || (node.id = u) );
+ };
+
+ //
+ // the Deferred class, borrowed from Twisted Python and Dojo
+ //
+
+ /**
+ * A class that models a single response (past or future) to a question.
+ * Multiple callbacks and error handlers may be added. If the response was
+ * added in the past, adding callbacks has the effect of calling them
+ * immediately. In this way, Deferreds simplify thinking about synchronous
+ * vs. asynchronous programming in languages which don't have continuations
+ * or generators which might otherwise provide syntax for deferring
+ * operations.
+ * @param {function} canceller Optional A function to be called when the
+ * Deferred is canceled.
+ * @param {number} timeout Optional How long to wait (in ms) before errback
+ * is called with a timeout error. If no timeout is passed, the default
+ * is 1hr. Passing -1 will disable any timeout.
+ * @constructor
+ * @public
+ */
+ Deferred = function(/*Function?*/ canceller, timeout){
+ // example:
+ // var deferred = new Deferred();
+ // setTimeout(function(){ deferred.callback({success: true}); }, 1000);
+ // return deferred;
+ this.chain = [];
+ this.id = _counter++;
+ this.fired = -1;
+ this.paused = 0;
+ this.results = [ null, null ];
+ this.canceller = canceller;
+ // FIXME(slightlyoff): is it really smart to be creating this many timers?
+ if (typeof timeout == 'number') {
+ if (timeout <= 0) {
+ timeout = 216000; // give it an hour
+ }
+ }
+ this._timer = setTimeout(
+ hitch(this, 'errback', new Error('timeout')),
+ (timeout || 1000)
+ );
+ this.silentlyCancelled = false;
+ };
+
+ /**
+ * Cancels a Deferred that has not yet received a value, or is waiting on
+ * another Deferred as its value. If a canceller is defined, the canceller
+ * is called. If the canceller did not return an error, or there was no
+ * canceller, then the errback chain is started.
+ * @public
+ */
+ Deferred.prototype.cancel = function() {
+ var err;
+ if (this.fired == -1) {
+ if (this.canceller) {
+ err = this.canceller(this);
+ } else {
+ this.silentlyCancelled = true;
+ }
+ if (this.fired == -1) {
+ if ( !(err instanceof Error) ) {
+ var res = err;
+ var msg = 'Deferred Cancelled';
+ if (err && err.toString) {
+ msg += ': ' + err.toString();
+ }
+ err = new Error(msg);
+ err.dType = 'cancel';
+ err.cancelResult = res;
+ }
+ this.errback(err);
+ }
+ } else if (
+ (this.fired == 0) &&
+ (this.results[0] instanceof Deferred)
+ ) {
+ this.results[0].cancel();
+ }
+ };
+
+
+ /**
+ * internal function for providing a result. If res is an instance of Error,
+ * we treat it like such and start the error chain.
+ * @param {Object|Error} res the result
+ * @private
+ */
+ Deferred.prototype._resback = function(res) {
+ if (this._timer) {
+ clearTimeout(this._timer);
+ }
+ this.fired = res instanceof Error ? 1 : 0;
+ this.results[this.fired] = res;
+ this._fire();
+ };
+
+ /**
+ * determine if the deferred has already been resolved
+ * @return {boolean}
+ * @private
+ */
+ Deferred.prototype._check = function() {
+ if (this.fired != -1) {
+ if (!this.silentlyCancelled) {
+ return 0;
+ }
+ this.silentlyCancelled = 0;
+ return 1;
+ }
+ return 0;
+ };
+
+ /**
+ * Begin the callback sequence with a non-error value.
+ * @param {Object|Error} res the result
+ * @public
+ */
+ Deferred.prototype.callback = function(res) {
+ this._check();
+ this._resback(res);
+ };
+
+ /**
+ * Begin the callback sequence with an error result.
+ * @param {Error|string} res the result. If not an Error, it's treated as the
+ * message for a new Error.
+ * @public
+ */
+ Deferred.prototype.errback = function(res) {
+ this._check();
+ if ( !(res instanceof Error) ) {
+ res = new Error(res);
+ }
+ this._resback(res);
+ };
+
+ /**
+ * Add a single function as the handler for both callback and errback,
+ * allowing you to specify a scope (unlike addCallbacks).
+ * @param {function|Object} cb A function. If cbfn is passed, the value of cb
+ * is treated as a scope
+ * @param {function|string} cbfn Optional A function or name of a function in
+ * the scope cb.
+ * @return {Deferred} this
+ * @public
+ */
+ Deferred.prototype.addBoth = function(cb, cbfn) {
+ var enclosed = hitch.apply(window, arguments);
+ return this.addCallbacks(enclosed, enclosed);
+ };
+
+ /**
+ * Add a single callback to the end of the callback sequence. Add a function
+ * as the handler for successful resolution of the Deferred. May be called
+ * multiple times to register many handlers. Note that return values are
+ * chained if provided, so it's best for callback handlers not to return
+ * anything.
+ * @param {function|Object} cb A function. If cbfn is passed, the value of cb
+ * is treated as a scope
+ * @param {function|string} cbfn Optional A function or name of a function in
+ * the scope cb.
+ * @return {Deferred} this
+ * @public
+ */
+ Deferred.prototype.addCallback = function(cb, cbfn /*...*/) {
+ return this.addCallbacks(hitch.apply(window, arguments));
+ };
+
+
+ /**
+ * Add a function as the handler for errors in the Deferred. May be called
+ * multiple times to add multiple error handlers.
+ * @param {function|Object} cb A function. If cbfn is passed, the value of cb
+ * is treated as a scope
+ * @param {function|string} cbfn Optional A function or name of a function in
+ * the scope cb.
+ * @return {Deferred} this
+ * @public
+ */
+ Deferred.prototype.addErrback = function(cb, cbfn) {
+ return this.addCallbacks(null, hitch.apply(window, arguments));
+ };
+
+ /**
+ * Add a functions as handlers for callback and errback in a single shot.
+ * @param {function} callback A function
+ * @param {function} errback A function
+ * @return {Deferred} this
+ * @public
+ */
+ Deferred.prototype.addCallbacks = function(callback, errback) {
+ this.chain.push([callback, errback]);
+ if (this.fired >= 0) {
+ this._fire();
+ }
+ return this;
+ };
+
+ /**
+ * when this Deferred is satisfied, pass it on to def, allowing it to run.
+ * @param {Deferred} def A deferred to add to the end of this Deferred in a chain
+ * @return {Deferred} this
+ * @public
+ */
+ Deferred.prototype.chain = function(def) {
+ this.addCallbacks(def.callback, def.errback);
+ return this;
+ };
+
+ /**
+ * Used internally to exhaust the callback sequence when a result is
+ * available.
+ * @private
+ */
+ Deferred.prototype._fire = function() {
+ var chain = this.chain;
+ var fired = this.fired;
+ var res = this.results[fired];
+ var cb = null;
+ while ((chain.length > 0) && (this.paused == 0)) {
+ var f = chain.shift()[fired];
+ if (!f) {
+ continue;
+ }
+ var func = hitch(this, function() {
+ var ret = f(res);
+ //If no response, then use previous response.
+ if (typeof ret != undefStr) {
+ res = ret;
+ }
+ fired = res instanceof Error ? 1 : 0;
+ if (res instanceof Deferred) {
+ cb = function(res) {
+ this._resback(res);
+ // inlined from _pause()
+ this.paused--;
+ if ( (this.paused == 0) && (this.fired >= 0)) {
+ this._fire();
+ }
+ }
+ // inlined from _unpause
+ this.paused++;
+ }
+ });
+
+ try {
+ func.call(this);
+ } catch(err) {
+ fired = 1;
+ res = err;
+ }
+ }
+
+ this.fired = fired;
+ this.results[fired] = res;
+ if (cb && this.paused ) {
+ // this is for "tail recursion" in case the dependent
+ // deferred is already fired
+ res.addBoth(cb);
+ }
+ };
+
+ /////////////////////////////////////////////////////////////////////////////
+ // Plugin Initialization Class and Helper Functions
+ /////////////////////////////////////////////////////////////////////////////
+
+ var returnFalse = function() {
+ return false;
+ };
+
+ var cachedHasVideo;
+ var cachedHasAudio;
+
+ var contentTests = {
+ canvas: function() {
+ return !!(
+ ua.isChrome || ua.isSafari >= 3 || ua.isFF >= 3 || ua.isOpera >= 9.2
+ );
+ },
+
+ svg: function() {
+ return !!(ua.isChrome || ua.isSafari || ua.isFF || ua.isOpera);
+ },
+
+ postMessage: function() {
+ return (
+ !!window['postMessage'] ||
+ ua.isChrome ||
+ ua.isIE >= 8 ||
+ ua.isSafari >= 3 ||
+ ua.isFF >= 3 ||
+ ua.isOpera >= 9.2
+ );
+ },
+
+ // the spec isn't settled and nothing currently supports it
+ websocket: returnFalse,
+
+ 'css-anim': function() {
+ // pretty much limited to WebKit's special transition and animation
+ // properties. Need to figure out a better way to triangulate this as
+ // FF3.x adds more of these properties in parallel.
+ return ua.isWebKit > 500;
+ },
+
+ // "working" video/audio tag?
+ video: function() {
+ if (typeof cachedHasVideo != undefStr) {
+ return cachedHasVideo;
+ }
+
+ // We haven't figured it out yet, so probe the <video> tag and cache the
+ // result.
+ var video = document.createElement('video');
+ return cachedHasVideo = (typeof video['play'] != undefStr);
+ },
+
+ audio: function() {
+ if (typeof cachedHasAudio != undefStr) {
+ return cachedHasAudio;
+ }
+
+ var audio = document.createElement('audio');
+ return cachedHasAudio = (typeof audio['play'] != undefStr);
+ },
+
+ 'video-theora': function() {
+ return contentTests.video() && (ua.isChrome || ua.isFF > 3);
+ },
+
+ 'video-h264': function() {
+ return contentTests.video() && (ua.isChrome || ua.isSafari >= 4);
+ },
+
+ 'audio-vorbis': function() {
+ return contentTests.audio() && (ua.isChrome || ua.isFF > 3);
+ },
+
+ 'audio-mp3': function() {
+ return contentTests.audio() && (ua.isChrome || ua.isSafari >= 4);
+ },
+
+ // can we implement RPC over available primitives?
+ rpc: function() {
+ // on IE we need the src to be on the same domain or we need postMessage
+ // to work. Since we can't count on the src being same-domain, we look
+ // for things that have postMessage. We may re-visit this later and add
+ // same-domain checking and cross-window-call-as-postMessage-replacement
+ // code.
+
+ // use "!!" to avoid null-is-an-object weirdness
+ return !!window['postMessage'];
+ },
+
+ sql: function() {
+ // HTML 5 databases
+ return !!window['openDatabase'];
+ },
+
+ storage: function(){
+ // DOM storage
+
+ // IE8, Safari, etc. support "localStorage", FF supported "globalStorage"
+ return !!window['globalStorage'] || !!window['localStorage'];
+ }
+ };
+
+ // isIE, isFF, isWebKit, etc.
+ forEach([
+ 'isOpera', 'isWebKit', 'isChrome', 'isKhtml', 'isSafari',
+ 'isGecko', 'isFF', 'isIE'
+ ],
+ function(name) {
+ contentTests[name] = function() {
+ return !!ua[name];
+ };
+ }
+ );
+
+ /**
+ * Checks the list of requirements to determine if the current host browser
+ * meets them natively. Primarialy relies on the contentTests array.
+ * @param {Array} reqs A list of tests, either names of test functions in
+ * contentTests or functions to execute.
+ * @return {boolean}
+ * @private
+ */
+ var testRequirements = function(reqs) {
+ // never use CF on Chrome or Safari
+ if (ua.isChrome || ua.isSafari) {
+ return true;
+ }
+
+ var allMatch = true;
+ if (!reqs) {
+ return false;
+ }
+ forEach(reqs, function(i) {
+ var matches = false;
+ if (isFunction(i)) {
+ // support custom test functions
+ matches = i();
+ } else {
+ // else it's a lookup by name
+ matches = (!!contentTests[i] && contentTests[i]());
+ }
+ allMatch = allMatch && matches;
+ });
+ return allMatch;
+ };
+
+ var cachedAvailable;
+
+ /**
+ * Checks to find out if ChromeFrame is available as a plugin
+ * @return {Boolean}
+ * @private
+ */
+ var isCfAvailable = function() {
+ if (typeof cachedAvailable != undefStr) {
+ return cachedAvailable;
+ }
+
+ cachedAvailable = false;
+ var p = n.plugins;
+ if (typeof window['ActiveXObject'] != undefStr) {
+ try {
+ var i = new ActiveXObject('ChromeTab.ChromeFrame');
+ if (i) {
+ cachedAvailable = true;
+ }
+ } catch(e) {
+ log('ChromeFrame not available, error:', e.message);
+ // squelch
+ }
+ } else {
+ for (var x = 0; x < p.length; x++) {
+ if (p[x].name.indexOf('Google Chrome Frame') == 0) {
+ cachedAvailable = true;
+ break;
+ }
+ }
+ }
+ return cachedAvailable;
+ };
+
+ /**
+ * Creates a <param> element with the specified name and value. If a parent
+ * is provided, the <param> element is appended to it.
+ * @param {string} name The name of the param
+ * @param {string} value The value
+ * @param {Node} parent Optional parent element
+ * @return {Boolean}
+ * @private
+ */
+ var param = function(name, value, parent) {
+ var p = document.createElement('param');
+ p.setAttribute('name', name);
+ p.setAttribute('value', value);
+ if (parent) {
+ parent.appendChild(p);
+ }
+ return p;
+ };
+
+ /** @type {boolean} */
+ var cfStyleTagInjected = false;
+
+ /**
+ * Creates a style sheet in the document which provides default styling for
+ * ChromeFrame instances. Successive calls should have no additive effect.
+ * @private
+ */
+ var injectCFStyleTag = function() {
+ if (cfStyleTagInjected) {
+ // once and only once
+ return;
+ }
+ try {
+ var rule = ['.chromeFrameDefaultStyle {',
+ 'width: 400px;',
+ 'height: 300px;',
+ 'padding: 0;',
+ 'margin: 0;',
+ '}'].join('');
+ var ss = document.createElement('style');
+ ss.setAttribute('type', 'text/css');
+ if (ss.styleSheet) {
+ ss.styleSheet.cssText = rule;
+ } else {
+ ss.appendChild(document.createTextNode(rule));
+ }
+ var h = document.getElementsByTagName('head')[0];
+ if (h.firstChild) {
+ h.insertBefore(ss, h.firstChild);
+ } else {
+ h.appendChild(ss);
+ }
+ cfStyleTagInjected = true;
+ } catch (e) {
+ // squelch
+
+ // FIXME(slightlyoff): log? retry?
+ }
+ };
+
+ /**
+ * Plucks properties from the passed arguments and sets them on the passed
+ * DOM node
+ * @param {Node} node The node to set properties on
+ * @param {Object} args A map of user-specified properties to set
+ * @private
+ */
+ var setProperties = function(node, args) {
+ injectCFStyleTag();
+
+ var srcNode = byId(args['node']);
+
+ node.id = args['id'] || (srcNode ? srcNode['id'] || getUid(srcNode) : '');
+
+ // TODO(slightlyoff): Opera compat? need to test there
+ var cssText = args['cssText'] || '';
+ node.style.cssText = ' ' + cssText;
+
+ var classText = args['className'] || '';
+ node.className = 'chromeFrameDefaultStyle ' + classText;
+
+ // default if the browser doesn't so we don't show sad-tab
+ var src = args['src'] || 'about:blank';
+
+ if (ua.isIE || ua.isOpera) {
+ node.src = src;
+ } else {
+ // crazyness regarding when things are set in NPAPI
+ node.setAttribute('src', src);
+ }
+
+ if (srcNode) {
+ srcNode.parentNode.replaceChild(node, srcNode);
+ }
+ };
+
+ /**
+ * Creates a plugin instance, taking named parameters from the passed args.
+ * @param {Object} args A bag of configuration properties, including values
+ * like 'node', 'cssText', 'className', 'id', 'src', etc.
+ * @return {Node}
+ * @private
+ */
+ var makeCFPlugin = function(args) {
+ var el; // the element
+ if (!ua.isIE) {
+ el = document.createElement('object');
+ el.setAttribute("type", "application/chromeframe");
+ } else {
+ var dummy = document.createElement('span');
+ dummy.innerHTML = [
+ '<object codeBase="//www.google.com"',
+ "type='application/chromeframe'",
+ 'classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A"></object>'
+ ].join(' ');
+ el = dummy.firstChild;
+ }
+ setProperties(el, args);
+ return el;
+ };
+
+ /**
+ * Creates an iframe in lieu of a ChromeFrame plugin, taking named parameters
+ * from the passed args.
+ * @param {Object} args A bag of configuration properties, including values
+ * like 'node', 'cssText', 'className', 'id', 'src', etc.
+ * @return {Node}
+ * @private
+ */
+ var makeCFIframe = function(args) {
+ var el = document.createElement('iframe');
+ setProperties(el, args);
+ // FIXME(slightlyoff):
+ // This is where we'll need to slot in "don't fire load events for
+ // fallback URL" logic.
+ listen(el, 'load', hitch(el, '_dispatch', 'load'));
+ return el;
+ };
+
+
+ var msgPrefix = 'CFInstance.rpc:';
+
+ /**
+ * A class that provides the ability for widget-mode hosted content to more
+ * easily call hosting-page exposed APIs (and vice versa). It builds upon the
+ * message-passing nature of ChromeFrame to route messages to the other
+ * side's RPC host and coordinate events such as 'RPC readyness', buffering
+ * calls until both sides indicate they are ready to participate.
+ * @constructor
+ * @public
+ */
+ var RPC = function(instance) {
+ this.initDeferred = new Deferred();
+
+ this.instance = instance;
+
+ instance.listen('message', hitch(this, '_handleMessage'));
+
+ this._localExposed = {};
+ this._doWithAckCallbacks = {};
+
+ this._open = false;
+ this._msgBacklog = [];
+
+ this._initialized = false;
+ this._exposeMsgBacklog = [];
+
+ this._exposed = false;
+ this._callRemoteMsgBacklog = [];
+
+ this._inFlight = {};
+
+ var sendLoadMsg = hitch(this, function(evt) {
+ this.doWithAck('load').addCallback(this, function() {
+ this._open = true;
+ this._postMessageBacklog();
+ });
+ });
+
+ if (instance['tagName']) {
+ instance.listen('load', sendLoadMsg);
+ } else {
+ sendLoadMsg();
+ }
+ };
+
+ RPC.prototype._postMessageBacklog = function() {
+ if (this._open) {
+ forEach(this._msgBacklog, this._postMessage, this);
+ this._msgBacklog = [];
+ }
+ };
+
+ RPC.prototype._postMessage = function(msg, force) {
+ if (!force && !this._open) {
+ this._msgBacklog.push(msg);
+ } else {
+ // FIXME(slightlyoff): need to check domains list here!
+ this.instance.postMessage(msgPrefix + msg, '*');
+ }
+ };
+
+ // currently no-ops. We may need them in the future
+ // RPC.prototype._doWithAck_load = function() { };
+ // RPC.prototype._doWithAck_init = function() { };
+
+ RPC.prototype._doWithAck = function(what) {
+ var f = this['_doWithAck_' + what];
+ if (f) {
+ f.call(this);
+ }
+
+ this._postMessage('doWithAckCallback:' + what, what == 'load');
+ };
+
+ RPC.prototype.doWithAck = function(what) {
+ var d = new Deferred();
+ this._doWithAckCallbacks[what] = d;
+ this._postMessage('doWithAck:' + what, what == 'load');
+ return d;
+ };
+
+ RPC.prototype._handleMessage = function(evt, stopper) {
+ var d = String(evt.data);
+
+ if (d.indexOf(msgPrefix) != 0) {
+ // not for us, allow the event dispatch to continue...
+ return;
+ }
+
+ // ...else we're the end of the line for this event
+ stopper();
+
+ // see if we know what type of message it is
+ d = d.substr(msgPrefix.length);
+
+ var cIndex = d.indexOf(':');
+
+ var type = d.substr(0, cIndex);
+
+ if (type == 'doWithAck') {
+ this._doWithAck(d.substr(cIndex + 1));
+ return;
+ }
+
+ var msgBody = d.substr(cIndex + 1);
+
+ if (type == 'doWithAckCallback') {
+ this._doWithAckCallbacks[msgBody].callback(1);
+ return;
+ }
+
+ if (type == 'init') {
+ return;
+ }
+
+ // All the other stuff we can do uses a JSON payload.
+ var obj = fromJson(msgBody);
+
+ if (type == 'callRemote') {
+
+ if (obj.method && obj.params && obj.id) {
+
+ var ret = {
+ success: 0,
+ returnId: obj.id
+ };
+
+ try {
+ // Undefined isn't valid JSON, so use null as default value.
+ ret.value = this._localExposed[ obj.method ](evt, obj) || null;
+ ret.success = 1;
+ } catch(e) {
+ ret.error = e.message;
+ }
+
+ this._postMessage('callReturn:' + toJson(ret));
+ }
+ }
+
+ if (type == 'callReturn') {
+ // see if we're waiting on an outstanding RPC call, which
+ // would be identified by returnId.
+ var rid = obj['returnId'];
+ if (!rid) {
+ // throw an error?
+ return;
+ }
+ var callWrap = this._inFlight[rid];
+ if (!callWrap) {
+ return;
+ }
+
+ if (obj.success) {
+ callWrap.d.callback(obj['value'] || 1);
+ } else {
+ callWrap.d.errback(new Error(obj['error'] || 'unspecified RPC error'));
+ }
+ delete this._inFlight[rid];
+ }
+
+ };
+
+ /**
+ * Makes a method visible to be called
+ * @param {string} name The name to expose the method at.
+ * @param {Function|string} method The function (or name of the function) to
+ * expose. If a name is provided, it's looked up from the passed scope.
+ * If no scope is provided, the global scope is queried for a function
+ * with that name.
+ * @param {Object} scope Optional A scope to bind the passed method to. If
+ * the method parameter is specified by a string, the method is both
+ * located on the passed scope and bound to it.
+ * @param {Array} domains Optional A list of domains in
+ * 'http://example.com:8080' format which may call the given method.
+ * Currently un-implemented.
+ * @public
+ */
+ RPC.prototype.expose = function(name, method, scope, domains) {
+ scope = scope || window;
+ method = isString(method) ? scope[method] : method;
+
+ // local call proxy that the other side will hit when calling our method
+ this._localExposed[name] = function(evt, obj) {
+ return method.apply(scope, obj.params);
+ };
+
+ if (!this._initialized) {
+ this._exposeMsgBacklog.push(arguments);
+ return;
+ }
+
+ var a = [name, method, scope, domains];
+ this._sendExpose.apply(this, a);
+ };
+
+ RPC.prototype._sendExpose = function(name) {
+ // now tell the other side that we're advertising this method
+ this._postMessage('expose:' + toJson({ name: name }));
+ };
+
+
+ /**
+ * Calls a remote method asynchronously and returns a Deferred object
+ * representing the response.
+ * @param {string} method Name of the method to call. Should be the same name
+ * which the other side has expose()'d.
+ * @param {Array} params Optional A list of arguments to pass to the called
+ * method. All elements in the list must be cleanly serializable to
+ * JSON.
+ * @param {CFInstance.Deferred} deferred Optional A Deferred to use for
+ * reporting the response of the call. If no deferred is passed, a new
+ * Deferred is created and returned.
+ * @return {CFInstance.Deferred}
+ * @public
+ */
+ RPC.prototype.callRemote = function(method, params, timeout, deferred) {
+ var d = deferred || new Deferred(null, timeout || -1);
+
+ if (!this._exposed) {
+ var args = toArray(arguments);
+ args.length = 3;
+ args.push(d);
+ this._callRemoteMsgBacklog.push(args);
+ return d;
+ }
+
+
+ if (!method) {
+ d.errback('no method provided!');
+ return d;
+ }
+
+ var id = msgPrefix + (_counter++);
+
+ // JSON-ify the whole bundle
+ var callWrapper = {
+ method: String(method),
+ params: params || [],
+ id: id
+ };
+ var callJson = toJson(callWrapper);
+ callWrapper.d = d;
+ this._inFlight[id] = callWrapper;
+ this._postMessage('callRemote:' + callJson);
+ return d;
+ };
+
+
+ /**
+ * Tells the other side of the connection that we're ready to start receiving
+ * calls. Returns a Deferred that is called back when both sides have
+ * initialized and any backlogged messages have been sent. RPC users should
+ * generally work to make sure that they call expose() on all of the methods
+ * they'd like to make available to the other side *before* calling init()
+ * @return {CFInstance.Deferred}
+ * @public
+ */
+ RPC.prototype.init = function() {
+ var d = this.initDeferred;
+ this.doWithAck('init').addCallback(this, function() {
+ // once the init floodgates are open, send our backlogs one at a time,
+ // with a little lag in the middle to prevent ordering problems
+
+ this._initialized = true;
+ while (this._exposeMsgBacklog.length) {
+ this.expose.apply(this, this._exposeMsgBacklog.shift());
+ }
+
+ setTimeout(hitch(this, function(){
+
+ this._exposed = true;
+ while (this._callRemoteMsgBacklog.length) {
+ this.callRemote.apply(this, this._callRemoteMsgBacklog.shift());
+ }
+
+ d.callback(1);
+
+ }), 30);
+
+ });
+ return d;
+ };
+
+ // CFInstance design notes:
+ //
+ // The CFInstance constructor is only ever used in host environments. In
+ // content pages (things hosted by a ChromeFrame instance), CFInstance
+ // acts as a singleton which provides services like RPC for communicating
+ // with it's mirror-image object in the hosting environment. We want the
+ // same methods and properties to be available on *instances* of
+ // CFInstance objects in the host env as on the singleton in the hosted
+ // content, despite divergent implementation.
+ //
+ // Further complicating things, CFInstance may specialize behavior
+ // internally based on whether or not it is communicationg with a fallback
+ // iframe or a 'real' ChromeFrame instance.
+
+ var CFInstance; // forward declaration
+ var h = window['externalHost'];
+ var inIframe = (window.parent != window);
+
+ if (inIframe) {
+ h = window.parent;
+ }
+
+ var normalizeTarget = function(targetOrigin) {
+ var l = window.location;
+ if (!targetOrigin) {
+ if (l.protocol != 'file:') {
+ targetOrigin = l.protocol + '//' + l.host + "/";
+ } else {
+ // TODO(slightlyoff):
+ // is this secure enough? Is there another way to get messages
+ // flowing reliably across file-hosted documents?
+ targetOrigin = '*';
+ }
+ }
+ return targetOrigin;
+ };
+
+ var postMessageToDest = function(dest, msg, targetOrigin) {
+ return dest.postMessage(msg, normalizeTarget(targetOrigin));
+ };
+
+ if (h) {
+ //
+ // We're loaded inside a ChromeFrame widget (or something that should look
+ // like we were).
+ //
+
+ CFInstance = {};
+
+ installEvtSys(CFInstance);
+
+ // FIXME(slightlyoff):
+ // passing a target origin to externalHost's postMessage seems b0rked
+ // right now, so pass null instead. Will re-enable hitch()'d variant
+ // once that's fixed.
+
+ // CFInstance.postMessage = hitch(null, postMessageToDest, h);
+
+ CFInstance.postMessage = function(msg, targetOrigin) {
+ return h.postMessage(msg,
+ (inIframe ? normalizeTarget(targetOrigin) : null) );
+ };
+
+ // Attach to the externalHost's onmessage to proxy it in to CFInstance's
+ // onmessage.
+ var dispatchMsg = function(evt) {
+ try {
+ CFInstance._dispatch('message', evt);
+ } catch(e) {
+ log(e);
+ // squelch
+ }
+ };
+ if (inIframe) {
+ listen(window, 'message', dispatchMsg);
+ } else {
+ h.onmessage = dispatchMsg;
+ }
+
+ CFInstance.rpc = new RPC(CFInstance);
+
+ _loadedListenerList.push(function(evt) {
+ CFInstance._dispatch('load', evt);
+ });
+
+ } else {
+ //
+ // We're the host document.
+ //
+
+ var installProperties = function(instance, args) {
+ var s = instance.supportedEvents = ['load', 'message'];
+ instance._msgPrefix = 'CFMessage:';
+
+ installEvtSys(instance);
+
+ instance.log = log;
+
+ // set up an RPC instance
+ instance.rpc = new RPC(instance);
+
+ forEach(s, function(evt) {
+ var l = args['on' + evt];
+ if (l) {
+ instance.listen(evt, l);
+ }
+ });
+
+ var contentWindow = instance.contentWindow;
+
+ // if it doesn't have a postMessage, route to/from the iframe's built-in
+ if (typeof instance['postMessage'] == undefStr && !!contentWindow) {
+
+ instance.postMessage = hitch(null, postMessageToDest, contentWindow);
+
+ listen(window, 'message', function(evt) {
+ if (evt.source == contentWindow) {
+ instance._dispatch('message', evt);
+ }
+ });
+ }
+
+ return instance;
+ };
+
+ /**
+ * A class whose instances correspond to ChromeFrame instances. Passing an
+ * arguments object to CFInstance helps parameterize the instance.
+ * @constructor
+ * @public
+ */
+ CFInstance = function(args) {
+ args = args || {};
+ var instance;
+ var success = false;
+
+ // If we've been passed a CFInstance object as our source node, just
+ // re-use it.
+ if (args['node']) {
+ var n = byId(args['node']);
+ // Look for CF-specific properties.
+ if (n && n.tagName == 'OBJECT' && n.success && n.rpc) {
+ // Navigate, set styles, etc.
+ setProperties(n, args);
+ return n;
+ }
+ }
+
+ var force = !!args['forcePlugin'];
+
+ if (!force && testRequirements(args['requirements'])) {
+ instance = makeCFIframe(args);
+ success = true;
+ } else if (isCfAvailable()) {
+ instance = makeCFPlugin(args);
+ success = true;
+ } else {
+ // else create an iframe but load the failure content and
+ // not the 'normal' content
+
+ // grab the fallback URL, and if none, use the 'click here
+ // to install ChromeFrame' URL. Note that we only support
+ // polling for install success if we're using the default
+ // URL
+
+ var fallback = '//www.google.com/chromeframe';
+
+ args.src = args['fallback'] || fallback;
+ instance = makeCFIframe(args);
+
+ if (args.src == fallback) {
+ // begin polling for install success.
+
+ // TODO(slightlyoff): need to prevent firing of onload hooks!
+ // TODO(slightlyoff): implement polling
+ // TODO(slightlyoff): replacement callback?
+ // TODO(slightlyoff): add flag to disable this behavior
+ }
+ }
+ instance.success = success;
+
+ installProperties(instance, args);
+
+ return instance;
+ };
+
+ // compatibility shims for development-time. These mirror the methods that
+ // are created on the CFInstance singleton if we detect that we're running
+ // inside of CF.
+ if (!CFInstance['postMessage']) {
+ CFInstance.postMessage = function() {
+ var args = toArray(arguments);
+ args.unshift('CFInstance.postMessage:');
+ log.apply(null, args);
+ };
+ CFInstance.listen = function() {
+ // this space intentionally left blank
+ };
+ }
+ }
+
+ // expose some properties
+ CFInstance.ua = ua;
+ CFInstance._documentLoaded = documentLoaded;
+ CFInstance.contentTests = contentTests;
+ CFInstance.isAvailable = function(requirements) {
+ var hasCf = isCfAvailable();
+ return requirements ? (hasCf || testRequirements(requirements)) : hasCf;
+
+ };
+ CFInstance.Deferred = Deferred;
+ CFInstance.toJson = toJson;
+ CFInstance.fromJson = fromJson;
+ CFInstance.log = log;
+
+ // expose CFInstance to the external scope. We've already checked to make
+ // sure we're not going to blow existing objects away.
+ scope.CFInstance = CFInstance;
+
+})( this['ChromeFrameScope'] || this );
+
+// vim: shiftwidth=2:et:ai:tabstop=2
diff --git a/chrome_frame/DEPS b/chrome_frame/DEPS
new file mode 100644
index 0000000..545ddc6
--- /dev/null
+++ b/chrome_frame/DEPS
@@ -0,0 +1,6 @@
+deps = {
+ # TODO(slightlyoff): need to add to Chromium third_party/ !!
+ # Chrome Frame needs these gecko SDKs and internals.
+ "src/third_party/xulrunner-sdk":
+ "svn://chrome-svn/chrome/trunk/deps/third_party/xulrunner-sdk",
+}
diff --git a/chrome_frame/bho.cc b/chrome_frame/bho.cc
new file mode 100644
index 0000000..e8c0374
--- /dev/null
+++ b/chrome_frame/bho.cc
@@ -0,0 +1,247 @@
+// Copyright (c) 2009 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 "chrome_frame/bho.h"
+
+#include <shlguid.h>
+#include <shobjidl.h>
+
+#include "base/logging.h"
+#include "base/registry.h"
+#include "base/scoped_bstr_win.h"
+#include "base/scoped_comptr_win.h"
+#include "base/scoped_variant_win.h"
+#include "base/string_util.h"
+#include "chrome_tab.h" // NOLINT
+#include "chrome_frame/protocol_sink_wrap.h"
+#include "chrome_frame/utils.h"
+#include "chrome_frame/vtable_patch_manager.h"
+
+const wchar_t kUrlMonDllName[] = L"urlmon.dll";
+const wchar_t kPatchProtocols[] = L"PatchProtocols";
+static const int kIBrowserServiceOnHttpEquivIndex = 30;
+
+PatchHelper g_patch_helper;
+
+BEGIN_VTABLE_PATCHES(IBrowserService)
+ VTABLE_PATCH_ENTRY(kIBrowserServiceOnHttpEquivIndex, Bho::OnHttpEquiv)
+END_VTABLE_PATCHES()
+
+_ATL_FUNC_INFO Bho::kBeforeNavigate2Info = {
+ CC_STDCALL, VT_EMPTY, 7, {
+ VT_DISPATCH,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_BOOL | VT_BYREF
+ }
+};
+
+Bho::Bho() {
+}
+
+STDMETHODIMP Bho::SetSite(IUnknown* site) {
+ HRESULT hr = S_OK;
+ if (site) {
+ ScopedComPtr<IWebBrowser2> web_browser2;
+ web_browser2.QueryFrom(site);
+ if (web_browser2) {
+ hr = DispEventAdvise(web_browser2, &DIID_DWebBrowserEvents2);
+ DCHECK(SUCCEEDED(hr)) << "DispEventAdvise failed. Error: " << hr;
+ }
+
+ if (g_patch_helper.state() == PatchHelper::PATCH_IBROWSER) {
+ ScopedComPtr<IBrowserService> browser_service;
+ hr = DoQueryService(SID_SShellBrowser, site, browser_service.Receive());
+ DCHECK(browser_service) << "DoQueryService - SID_SShellBrowser failed."
+ << " Site: " << site << " Error: " << hr;
+ if (browser_service) {
+ g_patch_helper.PatchBrowserService(browser_service);
+ DCHECK(SUCCEEDED(hr)) << "vtable_patch::PatchInterfaceMethods failed."
+ << " Site: " << site << " Error: " << hr;
+ }
+ }
+ }
+
+ return IObjectWithSiteImpl<Bho>::SetSite(site);
+}
+
+STDMETHODIMP Bho::BeforeNavigate2(IDispatch* dispatch, VARIANT* url,
+ VARIANT* flags, VARIANT* target_frame_name, VARIANT* post_data,
+ VARIANT* headers, VARIANT_BOOL* cancel) {
+ ScopedComPtr<IWebBrowser2> web_browser2;
+ if (dispatch)
+ web_browser2.QueryFrom(dispatch);
+
+ if (!web_browser2) {
+ NOTREACHED() << "Can't find WebBrowser2 with given dispatch";
+ return S_OK; // Return success, we operate on best effort basis.
+ }
+
+ DLOG(INFO) << "BeforeNavigate2: " << url->bstrVal;
+
+ if (g_patch_helper.state() == PatchHelper::PATCH_IBROWSER) {
+ VARIANT_BOOL is_top_level = VARIANT_FALSE;
+ web_browser2->get_TopLevelContainer(&is_top_level);
+
+ std::wstring current_url;
+ bool is_chrome_protocol = false;
+ if (is_top_level && IsValidUrlScheme(url->bstrVal)) {
+ current_url.assign(url->bstrVal, SysStringLen(url->bstrVal));
+ is_chrome_protocol = StartsWith(current_url, kChromeProtocolPrefix,
+ false);
+
+ if (!is_chrome_protocol && IsOptInUrl(current_url.c_str())) {
+ DLOG(INFO) << "Canceling navigation and switching to cf";
+ // Cancel original navigation
+ *cancel = VARIANT_TRUE;
+
+ // Issue new request with 'cf:'
+ current_url.insert(0, kChromeProtocolPrefix);
+ ScopedVariant new_url(current_url.c_str());
+ HRESULT hr = web_browser2->Navigate2(new_url.AsInput(), flags,
+ target_frame_name, post_data,
+ headers);
+ DCHECK(SUCCEEDED(hr)) << "web_browser2->Navigate2 failed. Error: " << hr
+ << std::endl << "Url: " << current_url
+ << std::endl << "flags: " << flags
+ << std::endl << "post data: " << post_data
+ << std::endl << "headers: " << headers;
+ }
+ }
+ }
+ return S_OK;
+}
+
+HRESULT Bho::FinalConstruct() {
+ return S_OK;
+}
+
+void Bho::FinalRelease() {
+}
+
+HRESULT STDMETHODCALLTYPE Bho::OnHttpEquiv(
+ IBrowserService_OnHttpEquiv_Fn original_httpequiv,
+ IBrowserService* browser, IShellView* shell_view, BOOL done,
+ VARIANT* in_arg, VARIANT* out_arg) {
+ if (!done && in_arg && (VT_BSTR == V_VT(in_arg))) {
+ if (StrStrI(V_BSTR(in_arg), kChromeContentPrefix)) {
+ // OnHttpEquiv is invoked for meta tags within sub frames as well.
+ // We want to switch renderers only for the top level frame. Since
+ // the same |browser| and |shell_view| are passed in to OnHttpEquiv
+ // even for sub iframes, we determine if this is the top one by
+ // checking if there are any sub frames created or not.
+ ScopedComPtr<IWebBrowser2> web_browser2;
+ DoQueryService(SID_SWebBrowserApp, browser, web_browser2.Receive());
+ if (web_browser2 && !HasSubFrames(web_browser2))
+ SwitchRenderer(web_browser2, browser, shell_view, V_BSTR(in_arg));
+ }
+ }
+
+ return original_httpequiv(browser, shell_view, done, in_arg, out_arg);
+}
+
+bool Bho::HasSubFrames(IWebBrowser2* web_browser2) {
+ bool has_sub_frames = false;
+ ScopedComPtr<IDispatch> doc_dispatch;
+ if (web_browser2) {
+ HRESULT hr = web_browser2->get_Document(doc_dispatch.Receive());
+ DCHECK(SUCCEEDED(hr) && doc_dispatch) <<
+ "web_browser2->get_Document failed. Error: " << hr;
+ ScopedComPtr<IOleContainer> container;
+ if (SUCCEEDED(hr) && doc_dispatch) {
+ container.QueryFrom(doc_dispatch);
+ ScopedComPtr<IEnumUnknown> enumerator;
+ if (container) {
+ container->EnumObjects(OLECONTF_EMBEDDINGS, enumerator.Receive());
+ ScopedComPtr<IUnknown> unk;
+ ULONG items_retrieved = 0;
+ if (enumerator)
+ enumerator->Next(1, unk.Receive(), &items_retrieved);
+ has_sub_frames = (items_retrieved != 0);
+ }
+ }
+ }
+
+ return has_sub_frames;
+}
+
+HRESULT Bho::SwitchRenderer(IWebBrowser2* web_browser2,
+ IBrowserService* browser, IShellView* shell_view,
+ const wchar_t* meta_tag) {
+ DCHECK(web_browser2 && browser && shell_view && meta_tag);
+
+ // Get access to the mshtml instance and the moniker
+ ScopedComPtr<IOleObject> mshtml_ole_object;
+ HRESULT hr = shell_view->GetItemObject(SVGIO_BACKGROUND, IID_IOleObject,
+ reinterpret_cast<void**>(mshtml_ole_object.Receive()));
+ if (!mshtml_ole_object) {
+ NOTREACHED() << "shell_view->GetItemObject failed. Error: " << hr;
+ return hr;
+ }
+
+ std::wstring url;
+ ScopedComPtr<IMoniker> moniker;
+ hr = mshtml_ole_object->GetMoniker(OLEGETMONIKER_ONLYIFTHERE,
+ OLEWHICHMK_OBJFULL, moniker.Receive());
+ DCHECK(moniker) << "mshtml_ole_object->GetMoniker failed. Error: " << hr;
+
+ if (moniker)
+ hr = GetUrlFromMoniker(moniker, NULL, &url);
+
+ DCHECK(!url.empty()) << "GetUrlFromMoniker failed. Error: " << hr;
+ DCHECK(!StartsWith(url, kChromeProtocolPrefix, false));
+
+ if (!url.empty()) {
+ url.insert(0, kChromeProtocolPrefix);
+ // Navigate to new url
+ VARIANT empty = ScopedVariant::kEmptyVariant;
+ VARIANT flags = { VT_I4 };
+ V_I4(&flags) = 0;
+ ScopedVariant url_var(url.c_str());
+ hr = web_browser2->Navigate2(url_var.AsInput(), &flags, &empty, &empty,
+ &empty);
+ DCHECK(SUCCEEDED(hr)) << "web_browser2->Navigate2 failed. Error: " << hr
+ << std::endl << "Url: " << url;
+ }
+
+ return S_OK;
+}
+
+void PatchHelper::InitializeAndPatchProtocolsIfNeeded() {
+ if (state_ != UNKNOWN)
+ return;
+
+ bool patch_protocol = GetConfigBool(true, kPatchProtocols);
+ if (patch_protocol) {
+ ProtocolSinkWrap::PatchProtocolHandler(kUrlMonDllName, CLSID_HttpProtocol);
+ ProtocolSinkWrap::PatchProtocolHandler(kUrlMonDllName, CLSID_HttpSProtocol);
+ state_ = PATCH_PROTOCOL;
+ } else {
+ state_ = PATCH_IBROWSER;
+ }
+}
+
+void PatchHelper::PatchBrowserService(IBrowserService* browser_service) {
+ DCHECK(state_ == PATCH_IBROWSER);
+ state_ = PATCH_IBROWSER_OK;
+ vtable_patch::PatchInterfaceMethods(browser_service,
+ IBrowserService_PatchInfo);
+}
+
+extern vtable_patch::MethodPatchInfo IInternetProtocol_PatchInfo[];
+extern vtable_patch::MethodPatchInfo IInternetProtocolEx_PatchInfo[];
+void PatchHelper::UnpatchIfNeeded() {
+ if (state_ == PATCH_PROTOCOL) {
+ vtable_patch::UnpatchInterfaceMethods(IInternetProtocol_PatchInfo);
+ vtable_patch::UnpatchInterfaceMethods(IInternetProtocolEx_PatchInfo);
+ } else if (state_ == PATCH_IBROWSER_OK) {
+ vtable_patch::UnpatchInterfaceMethods(IBrowserService_PatchInfo);
+ }
+
+ state_ = UNKNOWN;
+}
+
diff --git a/chrome_frame/bho.h b/chrome_frame/bho.h
new file mode 100644
index 0000000..ea9fe43
--- /dev/null
+++ b/chrome_frame/bho.h
@@ -0,0 +1,102 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_BHO_H_
+#define CHROME_FRAME_BHO_H_
+
+#include <string>
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <exdisp.h>
+#include <exdispid.h>
+#include <mshtml.h>
+#include <shdeprecated.h>
+
+#include "chrome_tab.h" // NOLINT
+#include "chrome_frame/resource.h"
+#include "grit/chrome_frame_resources.h"
+
+class PatchHelper {
+ public:
+ enum State { UNKNOWN, PATCH_IBROWSER, PATCH_IBROWSER_OK, PATCH_PROTOCOL };
+ PatchHelper() : state_(UNKNOWN) {
+ }
+
+ State state() const {
+ return state_;
+ }
+
+ void InitializeAndPatchProtocolsIfNeeded();
+ void PatchBrowserService(IBrowserService* p);
+ void UnpatchIfNeeded();
+ protected:
+ State state_;
+};
+
+// Single global variable
+extern PatchHelper g_patch_helper;
+
+class ATL_NO_VTABLE Bho
+ : public CComObjectRootEx<CComSingleThreadModel>,
+ public CComCoClass<Bho, &CLSID_ChromeFrameBHO>,
+ public IObjectWithSiteImpl<Bho>,
+ public IDispEventSimpleImpl<0, Bho, &DIID_DWebBrowserEvents2> {
+ public:
+ typedef HRESULT (STDMETHODCALLTYPE* IBrowserService_OnHttpEquiv_Fn)(
+ IBrowserService* browser, IShellView* shell_view, BOOL done,
+ VARIANT* in_arg, VARIANT* out_arg);
+
+DECLARE_REGISTRY_RESOURCEID(IDR_BHO)
+DECLARE_NOT_AGGREGATABLE(Bho)
+DECLARE_PROTECT_FINAL_CONSTRUCT()
+
+BEGIN_COM_MAP(Bho)
+ COM_INTERFACE_ENTRY(IObjectWithSite)
+END_COM_MAP()
+
+BEGIN_SINK_MAP(Bho)
+ SINK_ENTRY_INFO(0, DIID_DWebBrowserEvents2, DISPID_BEFORENAVIGATE2,
+ BeforeNavigate2, &kBeforeNavigate2Info)
+END_SINK_MAP()
+
+ // Lifetime management methods
+ Bho();
+
+ HRESULT FinalConstruct();
+ void FinalRelease();
+
+ // IObjectWithSite
+ STDMETHODIMP SetSite(IUnknown* site);
+ STDMETHOD(BeforeNavigate2)(IDispatch* dispatch, VARIANT* url, VARIANT* flags,
+ VARIANT* target_frame_name, VARIANT* post_data, VARIANT* headers,
+ VARIANT_BOOL* cancel);
+
+ // mshtml sends an IOleCommandTarget::Exec of OLECMDID_HTTPEQUIV
+ // (and OLECMDID_HTTPEQUIV_DONE) as soon as it parses a meta tag.
+ // It also sends contents of the meta tag as an argument. IEFrame
+ // handles this in IBrowserService::OnHttpEquiv. So this allows
+ // us to sniff the META tag by simply patching it. The renderer
+ // switching can be achieved by cancelling original navigation
+ // and issuing a new one using IWebBrowser2->Navigate2.
+ static HRESULT STDMETHODCALLTYPE OnHttpEquiv(
+ IBrowserService_OnHttpEquiv_Fn original_httpequiv,
+ IBrowserService* browser, IShellView* shell_view, BOOL done,
+ VARIANT* in_arg, VARIANT* out_arg);
+
+ protected:
+ bool PatchProtocolHandler(const CLSID& handler_clsid);
+ static bool HasSubFrames(IWebBrowser2* web_browser2);
+ static HRESULT SwitchRenderer(IWebBrowser2* web_browser2,
+ IBrowserService* browser, IShellView* shell_view,
+ const wchar_t* meta_tag);
+
+ static _ATL_FUNC_INFO kBeforeNavigate2Info;
+};
+
+// Add Bho to class library table so IE can CoCreate it.
+OBJECT_ENTRY_AUTO(CLSID_ChromeFrameBHO, Bho);
+
+#endif // CHROME_FRAME_BHO_H_
+
diff --git a/chrome_frame/bho.rgs b/chrome_frame/bho.rgs
new file mode 100644
index 0000000..824fd7d
--- /dev/null
+++ b/chrome_frame/bho.rgs
@@ -0,0 +1,23 @@
+HKLM {
+ NoRemove SOFTWARE {
+ NoRemove Classes {
+ ChromeFrame.Bho.1 = s 'Bho Class' {
+ CLSID = s '{ECB3C477-1A0A-44bd-BB57-78F9EFE34FA7}'
+ }
+ ChromeFrame.Bho = s 'ChromeFrame BHO' {
+ CLSID = s '{ECB3C477-1A0A-44bd-BB57-78F9EFE34FA7}'
+ CurVer = s 'ChromeFrame.Bho.1'
+ }
+ NoRemove CLSID {
+ ForceRemove {ECB3C477-1A0A-44bd-BB57-78F9EFE34FA7} = s 'ChromeFrame BHO' {
+ ProgID = s 'ChromeFrame.Bho.1'
+ VersionIndependentProgID = s 'ChromeFrame.Bho'
+ InprocServer32 = s '%MODULE%' {
+ val ThreadingModel = s 'Apartment'
+ }
+ 'TypeLib' = s '{6F2664E1-FF6E-488A-BCD1-F4CA6001DFCC}'
+ }
+ }
+ }
+ }
+}
diff --git a/chrome_frame/chrome_active_document.bmp b/chrome_frame/chrome_active_document.bmp
new file mode 100644
index 0000000..1229764
--- /dev/null
+++ b/chrome_frame/chrome_active_document.bmp
Binary files differ
diff --git a/chrome_frame/chrome_active_document.cc b/chrome_frame/chrome_active_document.cc
new file mode 100644
index 0000000..bbe38b7
--- /dev/null
+++ b/chrome_frame/chrome_active_document.cc
@@ -0,0 +1,648 @@
+// Copyright (c) 2009 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.
+
+// Implementation of ChromeActiveDocument
+#include "chrome_frame/chrome_active_document.h"
+
+#include <hlink.h>
+#include <htiface.h>
+#include <mshtmcid.h>
+#include <shdeprecated.h>
+#include <shlguid.h>
+#include <shobjidl.h>
+#include <tlogstg.h>
+#include <urlmon.h>
+#include <wininet.h>
+
+#include "base/command_line.h"
+#include "base/file_util.h"
+#include "base/logging.h"
+#include "base/path_service.h"
+#include "base/process_util.h"
+#include "base/registry.h"
+#include "base/scoped_variant_win.h"
+#include "base/string_tokenizer.h"
+#include "base/string_util.h"
+#include "base/thread.h"
+#include "base/thread_local.h"
+
+#include "grit/generated_resources.h"
+#include "chrome/browser/tab_contents/tab_contents.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/common/navigation_types.h"
+#include "chrome/test/automation/browser_proxy.h"
+#include "chrome/test/automation/tab_proxy.h"
+#include "chrome_frame/utils.h"
+
+const wchar_t kChromeAttachExternalTabPrefix[] = L"attach_external_tab";
+
+static const wchar_t kUseChromeNetworking[] = L"UseChromeNetworking";
+static const wchar_t kHandleTopLevelRequests[] =
+ L"HandleTopLevelRequests";
+
+base::ThreadLocalPointer<ChromeActiveDocument> g_active_doc_cache;
+
+bool g_first_launch_by_process_ = true;
+
+ChromeActiveDocument::ChromeActiveDocument()
+ : first_navigation_(true),
+ is_automation_client_reused_(false) {
+}
+
+HRESULT ChromeActiveDocument::FinalConstruct() {
+ // If we have a cached ChromeActiveDocument instance in TLS, then grab
+ // ownership of the cached document's automation client. This is an
+ // optimization to get Chrome active documents to load faster.
+ ChromeActiveDocument* cached_document = g_active_doc_cache.Get();
+ if (cached_document) {
+ DCHECK(automation_client_.get() == NULL);
+ automation_client_.reset(cached_document->automation_client_.release());
+ DLOG(INFO) << "Reusing automation client instance from "
+ << cached_document;
+ DCHECK(automation_client_.get() != NULL);
+ automation_client_->Reinitialize(this);
+ is_automation_client_reused_ = true;
+ } else {
+ // The FinalConstruct implementation in the ChromeFrameActivexBase class
+ // i.e. Base creates an instance of the ChromeFrameAutomationClient class
+ // and initializes it, which would spawn a new Chrome process, etc.
+ // We don't want to be doing this if we have a cached document, whose
+ // automation client instance can be reused.
+ HRESULT hr = Base::FinalConstruct();
+ if (FAILED(hr))
+ return hr;
+ }
+
+ bool chrome_network = GetConfigBool(false, kUseChromeNetworking);
+ bool top_level_requests = GetConfigBool(true, kHandleTopLevelRequests);
+ automation_client_->set_use_chrome_network(chrome_network);
+ automation_client_->set_handle_top_level_requests(top_level_requests);
+
+ find_dialog_.Init(automation_client_.get());
+
+ enabled_commands_map_[OLECMDID_PRINT] = true;
+ enabled_commands_map_[OLECMDID_FIND] = true;
+ enabled_commands_map_[OLECMDID_CUT] = true;
+ enabled_commands_map_[OLECMDID_COPY] = true;
+ enabled_commands_map_[OLECMDID_PASTE] = true;
+ enabled_commands_map_[OLECMDID_SELECTALL] = true;
+ return S_OK;
+}
+
+ChromeActiveDocument::~ChromeActiveDocument() {
+ DLOG(INFO) << __FUNCTION__;
+ if (find_dialog_.IsWindow()) {
+ find_dialog_.DestroyWindow();
+ }
+}
+
+// Override DoVerb
+STDMETHODIMP ChromeActiveDocument::DoVerb(LONG verb,
+ LPMSG msg,
+ IOleClientSite* active_site,
+ LONG index,
+ HWND parent_window,
+ LPCRECT pos) {
+ // IE will try and in-place activate us in some cases. This happens when
+ // the user opens a new IE window with a URL that has us as the DocObject.
+ // Here we refuse to be activated in-place and we will force IE to UIActivate
+ // us.
+ if (OLEIVERB_INPLACEACTIVATE == verb) {
+ return E_NOTIMPL;
+ }
+ // Check if we should activate as a docobject or not
+ // (client supports IOleDocumentSite)
+ if (doc_site_) {
+ switch (verb) {
+ case OLEIVERB_SHOW:
+ case OLEIVERB_OPEN:
+ case OLEIVERB_UIACTIVATE:
+ if (!m_bUIActive) {
+ return doc_site_->ActivateMe(NULL);
+ }
+ break;
+ }
+ }
+ return IOleObjectImpl<ChromeActiveDocument>::DoVerb(verb,
+ msg,
+ active_site,
+ index,
+ parent_window,
+ pos);
+}
+
+STDMETHODIMP ChromeActiveDocument::InPlaceDeactivate(void) {
+ // Release the pointers we have no need for now.
+ doc_site_.Release();
+ in_place_frame_.Release();
+ return IOleInPlaceObjectWindowlessImpl<ChromeActiveDocument>::
+ InPlaceDeactivate();
+}
+
+// Override IOleInPlaceActiveObjectImpl::OnDocWindowActivate
+STDMETHODIMP ChromeActiveDocument::OnDocWindowActivate(BOOL activate) {
+ DLOG(INFO) << __FUNCTION__;
+ return S_OK;
+}
+
+STDMETHODIMP ChromeActiveDocument::TranslateAccelerator(MSG* msg) {
+ DLOG(INFO) << __FUNCTION__;
+ if (msg == NULL)
+ return E_POINTER;
+
+ if (msg->message == WM_KEYDOWN && msg->wParam == VK_TAB) {
+ HWND focus = ::GetFocus();
+ if (focus != m_hWnd && !::IsChild(m_hWnd, focus)) {
+ // The call to SetFocus triggers a WM_SETFOCUS that makes the base class
+ // set focus to the correct element in Chrome.
+ ::SetFocus(m_hWnd);
+ return S_OK;
+ }
+ }
+
+ return S_FALSE;
+}
+// Override IPersistStorageImpl::IsDirty
+STDMETHODIMP ChromeActiveDocument::IsDirty() {
+ DLOG(INFO) << __FUNCTION__;
+ return S_FALSE;
+}
+
+bool ChromeActiveDocument::is_frame_busting_enabled() {
+ return false;
+}
+
+STDMETHODIMP ChromeActiveDocument::Load(BOOL fully_avalable,
+ IMoniker* moniker_name,
+ LPBC bind_context,
+ DWORD mode) {
+ if (NULL == moniker_name) {
+ return E_INVALIDARG;
+ }
+ CComHeapPtr<WCHAR> display_name;
+ moniker_name->GetDisplayName(bind_context, NULL, &display_name);
+ std::wstring url = display_name;
+
+ bool is_chrome_protocol = StartsWith(url, kChromeProtocolPrefix, false);
+ bool is_new_navigation = true;
+
+ if (is_chrome_protocol) {
+ url.erase(0, lstrlen(kChromeProtocolPrefix));
+ is_new_navigation =
+ !StartsWith(url, kChromeAttachExternalTabPrefix, false);
+ }
+
+ if (!IsValidUrlScheme(url)) {
+ DLOG(WARNING) << __FUNCTION__ << " Disallowing navigation to url: "
+ << url;
+ return E_INVALIDARG;
+ }
+
+ if (!is_new_navigation) {
+ WStringTokenizer tokenizer(url, L"&");
+ // Skip over kChromeAttachExternalTabPrefix
+ tokenizer.GetNext();
+
+ intptr_t external_tab_cookie = 0;
+
+ if (tokenizer.GetNext())
+ StringToInt(tokenizer.token(),
+ reinterpret_cast<int*>(&external_tab_cookie));
+
+ if (external_tab_cookie == 0) {
+ NOTREACHED() << "invalid url for attach tab: " << url;
+ return E_FAIL;
+ }
+
+ automation_client_->AttachExternalTab(external_tab_cookie);
+ }
+
+ // Initiate navigation before launching chrome so that the url will be
+ // cached and sent with launch settings.
+ if (is_new_navigation) {
+ url_.Reset(::SysAllocString(url.c_str()));
+ if (url_.Length()) {
+ std::string utf8_url;
+ WideToUTF8(url_, url_.Length(), &utf8_url);
+ if (!automation_client_->InitiateNavigation(utf8_url)) {
+ DLOG(ERROR) << "Invalid URL: " << url;
+ Error(L"Invalid URL");
+ url_.Reset();
+ return E_INVALIDARG;
+ }
+
+ DLOG(INFO) << "Url is " << url_;
+ }
+ }
+
+ if (!is_automation_client_reused_ &&
+ !InitializeAutomation(GetHostProcessName(false), L"", IsIEInPrivate())) {
+ return E_FAIL;
+ }
+
+ if (!is_chrome_protocol) {
+ CComObject<UrlmonUrlRequest>* new_request = NULL;
+ CComObject<UrlmonUrlRequest>::CreateInstance(&new_request);
+ new_request->AddRef();
+
+ if (SUCCEEDED(new_request->ConnectToExistingMoniker(moniker_name,
+ bind_context,
+ url))) {
+ base_url_request_.swap(&new_request);
+ DCHECK(new_request == NULL);
+ } else {
+ new_request->Release();
+ }
+ }
+
+ UMA_HISTOGRAM_CUSTOM_COUNTS("ChromeFrame.FullTabLaunchType",
+ is_chrome_protocol, 0, 1, 2);
+ return S_OK;
+}
+
+STDMETHODIMP ChromeActiveDocument::Save(IMoniker* moniker_name,
+ LPBC bind_context,
+ BOOL remember) {
+ return E_NOTIMPL;
+}
+
+STDMETHODIMP ChromeActiveDocument::SaveCompleted(IMoniker* moniker_name,
+ LPBC bind_context) {
+ return E_NOTIMPL;
+}
+
+STDMETHODIMP ChromeActiveDocument::GetCurMoniker(IMoniker** moniker_name) {
+ return E_NOTIMPL;
+}
+
+STDMETHODIMP ChromeActiveDocument::GetClassID(CLSID* class_id) {
+ if (NULL == class_id) {
+ return E_POINTER;
+ }
+ *class_id = GetObjectCLSID();
+ return S_OK;
+}
+
+STDMETHODIMP ChromeActiveDocument::QueryStatus(const GUID* cmd_group_guid,
+ ULONG number_of_commands,
+ OLECMD commands[],
+ OLECMDTEXT* command_text) {
+ DLOG(INFO) << __FUNCTION__;
+ for (ULONG command_index = 0; command_index < number_of_commands;
+ command_index++) {
+ DLOG(INFO) << "Command id = " << commands[command_index].cmdID;
+ if (enabled_commands_map_.find(commands[command_index].cmdID) !=
+ enabled_commands_map_.end()) {
+ commands[command_index].cmdf = OLECMDF_ENABLED;
+ }
+ }
+ return S_OK;
+}
+
+STDMETHODIMP ChromeActiveDocument::Exec(const GUID* cmd_group_guid,
+ DWORD command_id,
+ DWORD cmd_exec_opt,
+ VARIANT* in_args,
+ VARIANT* out_args) {
+ DLOG(INFO) << __FUNCTION__ << " Cmd id =" << command_id;
+ // Bail out if we have been uninitialized.
+ if (automation_client_.get() && automation_client_->tab()) {
+ return ProcessExecCommand(cmd_group_guid, command_id, cmd_exec_opt,
+ in_args, out_args);
+ }
+ return S_FALSE;
+}
+
+STDMETHODIMP ChromeActiveDocument::GetUrlForEvents(BSTR* url) {
+ if (NULL == url) {
+ return E_POINTER;
+ }
+ *url = ::SysAllocString(url_);
+ return S_OK;
+}
+
+HRESULT ChromeActiveDocument::IOleObject_SetClientSite(
+ IOleClientSite* client_site) {
+ if (client_site == NULL) {
+ ChromeActiveDocument* cached_document = g_active_doc_cache.Get();
+ if (cached_document) {
+ DCHECK(this == cached_document);
+ g_active_doc_cache.Set(NULL);
+ cached_document->Release();
+ }
+ }
+ return Base::IOleObject_SetClientSite(client_site);
+}
+
+
+HRESULT ChromeActiveDocument::ActiveXDocActivate(LONG verb) {
+ HRESULT hr = S_OK;
+ m_bNegotiatedWnd = TRUE;
+ if (!m_bInPlaceActive) {
+ hr = m_spInPlaceSite->CanInPlaceActivate();
+ if (FAILED(hr)) {
+ return hr;
+ }
+ m_spInPlaceSite->OnInPlaceActivate();
+ }
+ m_bInPlaceActive = TRUE;
+ // get location in the parent window,
+ // as well as some information about the parent
+ ScopedComPtr<IOleInPlaceUIWindow> in_place_ui_window;
+ frame_info_.cb = sizeof(OLEINPLACEFRAMEINFO);
+ HWND parent_window = NULL;
+ if (m_spInPlaceSite->GetWindow(&parent_window) == S_OK) {
+ in_place_frame_.Release();
+ RECT position_rect = {0};
+ RECT clip_rect = {0};
+ m_spInPlaceSite->GetWindowContext(in_place_frame_.Receive(),
+ in_place_ui_window.Receive(),
+ &position_rect,
+ &clip_rect,
+ &frame_info_);
+ if (!m_bWndLess) {
+ if (IsWindow()) {
+ ::ShowWindow(m_hWnd, SW_SHOW);
+ SetFocus();
+ } else {
+ m_hWnd = Create(parent_window, position_rect);
+ }
+ }
+ SetObjectRects(&position_rect, &clip_rect);
+ }
+
+ ScopedComPtr<IOleInPlaceActiveObject> in_place_active_object(this);
+
+ // Gone active by now, take care of UIACTIVATE
+ if (DoesVerbUIActivate(verb)) {
+ if (!m_bUIActive) {
+ m_bUIActive = TRUE;
+ hr = m_spInPlaceSite->OnUIActivate();
+ if (FAILED(hr)) {
+ return hr;
+ }
+ // set ourselves up in the host
+ if (in_place_active_object) {
+ if (in_place_frame_) {
+ in_place_frame_->SetActiveObject(in_place_active_object, NULL);
+ }
+ if (in_place_ui_window) {
+ in_place_ui_window->SetActiveObject(in_place_active_object, NULL);
+ }
+ }
+ }
+ }
+ m_spClientSite->ShowObject();
+ return S_OK;
+}
+
+void ChromeActiveDocument::OnNavigationStateChanged(int tab_handle, int flags,
+ const IPC::NavigationInfo& nav_info) {
+ // TODO(joshia): handle INVALIDATE_TAB,INVALIDATE_LOAD etc.
+ DLOG(INFO) << __FUNCTION__ << std::endl << " Flags: " << flags
+ << "Url: " << nav_info.url <<
+ ", Title: " << nav_info.title <<
+ ", Type: " << nav_info.navigation_type << ", Relative Offset: " <<
+ nav_info.relative_offset << ", Index: " << nav_info.navigation_index;;
+
+ UpdateNavigationState(nav_info);
+}
+
+void ChromeActiveDocument::OnUpdateTargetUrl(int tab_handle,
+ const std::wstring& new_target_url) {
+ if (in_place_frame_) {
+ in_place_frame_->SetStatusText(new_target_url.c_str());
+ }
+}
+
+bool IsFindAccelerator(const MSG& msg) {
+ // TODO(robertshield): This may not stand up to localization. Fix if this
+ // is the case.
+ return msg.message == WM_KEYDOWN && msg.wParam == 'F' &&
+ win_util::IsCtrlPressed() &&
+ !(win_util::IsAltPressed() || win_util::IsShiftPressed());
+}
+
+void ChromeActiveDocument::OnAcceleratorPressed(int tab_handle,
+ const MSG& accel_message) {
+ bool handled_accel = false;
+ if (in_place_frame_ != NULL) {
+ handled_accel = (S_OK == in_place_frame_->TranslateAcceleratorW(
+ const_cast<MSG*>(&accel_message), 0));
+ }
+
+ if (!handled_accel) {
+ if (IsFindAccelerator(accel_message)) {
+ // Handle the showing of the find dialog explicitly.
+ OnFindInPage();
+ } else if (AllowFrameToTranslateAccelerator(accel_message) != S_OK) {
+ DLOG(INFO) << "IE DID NOT handle accel key " << accel_message.wParam;
+ TabProxy* tab = GetTabProxy();
+ if (tab) {
+ tab->ProcessUnhandledAccelerator(accel_message);
+ }
+ }
+ } else {
+ DLOG(INFO) << "IE handled accel key " << accel_message.wParam;
+ }
+}
+
+void ChromeActiveDocument::OnTabbedOut(int tab_handle, bool reverse) {
+ DLOG(INFO) << __FUNCTION__;
+ if (in_place_frame_) {
+ MSG msg = { NULL, WM_KEYDOWN, VK_TAB };
+ in_place_frame_->TranslateAcceleratorW(&msg, 0);
+ }
+}
+
+void ChromeActiveDocument::OnDidNavigate(int tab_handle,
+ const IPC::NavigationInfo& nav_info) {
+ DLOG(INFO) << __FUNCTION__ << std::endl << "Url: " << nav_info.url <<
+ ", Title: " << nav_info.title <<
+ ", Type: " << nav_info.navigation_type << ", Relative Offset: " <<
+ nav_info.relative_offset << ", Index: " << nav_info.navigation_index;
+
+ // This could be NULL if the active document instance is being destroyed.
+ if (!m_spInPlaceSite) {
+ DLOG(INFO) << __FUNCTION__ << "m_spInPlaceSite is NULL. Returning";
+ return;
+ }
+
+ UpdateNavigationState(nav_info);
+}
+
+void ChromeActiveDocument::UpdateNavigationState(
+ const IPC::NavigationInfo& new_navigation_info) {
+ bool is_title_changed = (navigation_info_.title != new_navigation_info.title);
+ bool is_url_changed = (navigation_info_.url.is_valid() &&
+ (navigation_info_.url != new_navigation_info.url));
+ bool is_ssl_state_changed =
+ (navigation_info_.security_style != new_navigation_info.security_style) ||
+ (navigation_info_.has_mixed_content !=
+ new_navigation_info.has_mixed_content);
+
+ navigation_info_ = new_navigation_info;
+
+ if (is_title_changed) {
+ ScopedVariant title(navigation_info_.title.c_str());
+ IEExec(NULL, OLECMDID_SETTITLE, OLECMDEXECOPT_DONTPROMPTUSER,
+ title.AsInput(), NULL);
+ }
+
+ if (is_ssl_state_changed) {
+ int lock_status = SECURELOCK_SET_UNSECURE;
+ switch (navigation_info_.security_style) {
+ case SECURITY_STYLE_AUTHENTICATION_BROKEN:
+ lock_status = SECURELOCK_SET_SECUREUNKNOWNBIT;
+ break;
+ case SECURITY_STYLE_AUTHENTICATED:
+ lock_status = navigation_info_.has_mixed_content ?
+ SECURELOCK_SET_MIXED : SECURELOCK_SET_SECUREUNKNOWNBIT;
+ break;
+ default:
+ break;
+ }
+
+ ScopedVariant secure_lock_status(lock_status);
+ IEExec(&CGID_ShellDocView, INTERNAL_CMDID_SET_SSL_LOCK,
+ OLECMDEXECOPT_DODEFAULT, secure_lock_status.AsInput(), NULL);
+ }
+
+ if (navigation_info_.url.is_valid() &&
+ (is_url_changed || url_.Length() == 0)) {
+ url_.Allocate(UTF8ToWide(navigation_info_.url.spec()).c_str());
+ // Now call the FireNavigateCompleteEvent which makes IE update the text
+ // in the address-bar. We call the FireBeforeNavigateComplete2Event and
+ // FireDocumentComplete event just for completeness sake. If some BHO
+ // chooses to cancel the navigation in the OnBeforeNavigate2 handler
+ // we will ignore the cancellation request.
+
+ // Todo(joshia): investigate if there's a better way to set URL in the
+ // address bar
+ ScopedComPtr<IWebBrowserEventsService> web_browser_events_svc;
+ DoQueryService(__uuidof(web_browser_events_svc), m_spClientSite,
+ web_browser_events_svc.Receive());
+ if (web_browser_events_svc) {
+ // TODO(joshia): maybe we should call FireBeforeNavigate2Event in
+ // ChromeActiveDocument::Load and abort if cancelled.
+ VARIANT_BOOL should_cancel = VARIANT_FALSE;
+ web_browser_events_svc->FireBeforeNavigate2Event(&should_cancel);
+ web_browser_events_svc->FireNavigateComplete2Event();
+ if (VARIANT_TRUE != should_cancel) {
+ web_browser_events_svc->FireDocumentCompleteEvent();
+ }
+ }
+ }
+}
+
+void ChromeActiveDocument::OnFindInPage() {
+ TabProxy* tab = GetTabProxy();
+ if (tab) {
+ if (!find_dialog_.IsWindow()) {
+ find_dialog_.Create(m_hWnd);
+ }
+
+ find_dialog_.ShowWindow(SW_SHOW);
+ }
+}
+
+void ChromeActiveDocument::OnViewSource() {
+ DCHECK(navigation_info_.url.is_valid());
+ std::string url_to_open = "view-source:";
+ url_to_open += navigation_info_.url.spec();
+ OnOpenURL(0, GURL(url_to_open), NEW_WINDOW);
+}
+
+void ChromeActiveDocument::OnOpenURL(int tab_handle, const GURL& url_to_open,
+ int open_disposition) {
+ // If the disposition indicates that we should be opening the URL in the
+ // current tab, then we can reuse the ChromeFrameAutomationClient instance
+ // maintained by the current ChromeActiveDocument instance. We cache this
+ // instance so that it can be used by the new ChromeActiveDocument instance
+ // which may be instantiated for handling the new URL.
+ if (open_disposition == CURRENT_TAB) {
+ // Grab a reference to ensure that the document remains valid.
+ AddRef();
+ g_active_doc_cache.Set(this);
+ }
+
+ Base::OnOpenURL(tab_handle, url_to_open, open_disposition);
+}
+
+void ChromeActiveDocument::OnLoad(int tab_handle, const GURL& url) {
+ if (ready_state_ < READYSTATE_COMPLETE) {
+ ready_state_ = READYSTATE_COMPLETE;
+ FireOnChanged(DISPID_READYSTATE);
+ }
+}
+
+bool ChromeActiveDocument::PreProcessContextMenu(HMENU menu) {
+ ScopedComPtr<IBrowserService> browser_service;
+ ScopedComPtr<ITravelLog> travel_log;
+
+ DoQueryService(SID_SShellBrowser, m_spClientSite, browser_service.Receive());
+ if (!browser_service)
+ return true;
+
+ browser_service->GetTravelLog(travel_log.Receive());
+ if (!travel_log)
+ return true;
+
+ if (SUCCEEDED(travel_log->GetTravelEntry(browser_service, TLOG_BACK, NULL))) {
+ EnableMenuItem(menu, IDS_CONTENT_CONTEXT_BACK, MF_BYCOMMAND | MF_ENABLED);
+ } else {
+ EnableMenuItem(menu, IDS_CONTENT_CONTEXT_BACK, MF_BYCOMMAND | MFS_DISABLED);
+ }
+
+
+ if (SUCCEEDED(travel_log->GetTravelEntry(browser_service, TLOG_FORE, NULL))) {
+ EnableMenuItem(menu, IDS_CONTENT_CONTEXT_FORWARD,
+ MF_BYCOMMAND | MF_ENABLED);
+ } else {
+ EnableMenuItem(menu, IDS_CONTENT_CONTEXT_FORWARD,
+ MF_BYCOMMAND | MFS_DISABLED);
+ }
+
+ // Call base class (adds 'About' item)
+ return Base::PreProcessContextMenu(menu);
+}
+
+bool ChromeActiveDocument::HandleContextMenuCommand(UINT cmd) {
+ ScopedComPtr<IWebBrowser2> web_browser2;
+ DoQueryService(SID_SWebBrowserApp, m_spClientSite, web_browser2.Receive());
+
+ switch (cmd) {
+ case IDS_CONTENT_CONTEXT_BACK:
+ web_browser2->GoBack();
+ break;
+
+ case IDS_CONTENT_CONTEXT_FORWARD:
+ web_browser2->GoForward();
+ break;
+
+ case IDS_CONTENT_CONTEXT_RELOAD:
+ web_browser2->Refresh();
+ break;
+
+ default:
+ return Base::HandleContextMenuCommand(cmd);
+ }
+
+ return true;
+}
+
+HRESULT ChromeActiveDocument::IEExec(const GUID* cmd_group_guid,
+ DWORD command_id, DWORD cmd_exec_opt,
+ VARIANT* in_args, VARIANT* out_args) {
+ HRESULT hr = E_FAIL;
+ ScopedComPtr<IOleCommandTarget> frame_cmd_target;
+ if (m_spInPlaceSite)
+ hr = frame_cmd_target.QueryFrom(m_spInPlaceSite);
+
+ if (frame_cmd_target)
+ hr = frame_cmd_target->Exec(cmd_group_guid, command_id, cmd_exec_opt,
+ in_args, out_args);
+
+ return hr;
+}
diff --git a/chrome_frame/chrome_active_document.h b/chrome_frame/chrome_active_document.h
new file mode 100644
index 0000000..b488bc5
--- /dev/null
+++ b/chrome_frame/chrome_active_document.h
@@ -0,0 +1,258 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_ACTIVE_DOCUMENT_H_
+#define CHROME_FRAME_CHROME_ACTIVE_DOCUMENT_H_
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <atlctl.h>
+#include <map>
+
+#include "base/scoped_ptr.h"
+#include "base/scoped_comptr_win.h"
+#include "base/thread.h"
+
+#include "chrome_frame/chrome_frame_activex_base.h"
+#include "chrome_frame/com_type_info_holder.h"
+#include "chrome_frame/find_dialog.h"
+#include "chrome_frame/in_place_menu.h"
+#include "chrome_frame/ole_document_impl.h"
+#include "chrome_frame/resource.h"
+#include "chrome_frame/extra_system_apis.h"
+
+class Thread;
+class TabProxy;
+class ChromeActiveDocument;
+
+// A call to IOleCommandTarget::Exec on the webbrowser with this command id
+// and a command group of CGID_EXPLORER causes IE to finalize the current
+// travel log entry and move to a new location (pruning any forward entries
+// if needed)
+#define INTERNAL_CMDID_FINALIZE_TRAVEL_LOG (38)
+
+// To set the lock icon status call IOleCommandTarget::Exec on site with
+// this command id and a command group of CGID_EXPLORER The in arg is one of
+// the values: SECURELOCK_SET_UNSECURE, SECURELOCK_SET_MIXED,
+// SECURELOCK_SET_SECURE128BIT etc declared in shdeprecated.h
+#define INTERNAL_CMDID_SET_SSL_LOCK (37)
+
+// A call to IOleCommandTarget::Exec on the webbrowser with this command id
+// and a command group of CGID_EXPLORER causes IE to replace the URL in the
+// current travel log entry
+#define UNDOC_CMDID_REPLACE_CURRENT_TRAVEL_LOG_ENTRY_URL (40)
+
+#define UNDOC_IE_CONTEXTMENU_ADDFAV (2261)
+#define UNDOC_IE_CONTEXTMENU_VIEWSOURCE (2139)
+#define UNDOC_IE_CONTEXTMENU_REFRESH (6042)
+
+// This macro should be defined in the public section of the class.
+#define BEGIN_EXEC_COMMAND_MAP(theClass) \
+ public: \
+ HRESULT ProcessExecCommand(const GUID* cmd_group_guid, DWORD command_id, \
+ DWORD cmd_exec_opt, VARIANT* in_args, \
+ VARIANT* out_args) { \
+ HRESULT hr = OLECMDERR_E_NOTSUPPORTED; \
+ switch (command_id) {
+
+
+#define EXEC_COMMAND_HANDLER(id, handler) \
+ case id: { \
+ hr = S_OK; \
+ handler(cmd_group_guid, command_id, cmd_exec_opt, in_args, out_args) \
+ break; \
+ }
+
+#define EXEC_COMMAND_HANDLER_NO_ARGS(id, handler) \
+ case id: { \
+ hr = S_OK; \
+ handler(); \
+ break; \
+ }
+
+#define EXEC_COMMAND_HANDLER_GENERIC(id, code) \
+ case id: { \
+ hr = S_OK; \
+ code; \
+ break; \
+ }
+
+#define END_EXEC_COMMAND_MAP() \
+ default: \
+ break; \
+ } \
+ return hr; \
+}
+
+// ChromeActiveDocument: Implementation of the Active Document object that is
+// responsible for rendering pages in Chrome. This object delegates to
+// Chrome.exe (via the Chrome IPC-based automation mechanism) for the actual
+// rendering
+class ATL_NO_VTABLE ChromeActiveDocument
+ : public ChromeFrameActivexBase<ChromeActiveDocument,
+ CLSID_ChromeActiveDocument>,
+ public IOleDocumentImpl<ChromeActiveDocument>,
+ public IOleDocumentViewImpl<ChromeActiveDocument>,
+ public IPersistMoniker,
+ public IOleCommandTarget,
+ public InPlaceMenu<ChromeActiveDocument>,
+ public IWebBrowserEventsUrlService {
+ public:
+ typedef ChromeFrameActivexBase<ChromeActiveDocument,
+ CLSID_ChromeActiveDocument> Base;
+ ChromeActiveDocument();
+ ~ChromeActiveDocument();
+
+ DECLARE_REGISTRY_RESOURCEID(IDR_CHROMEACTIVEDOCUMENT)
+
+BEGIN_COM_MAP(ChromeActiveDocument)
+ COM_INTERFACE_ENTRY(IOleDocument)
+ COM_INTERFACE_ENTRY(IOleDocumentView)
+ COM_INTERFACE_ENTRY(IPersistMoniker)
+ COM_INTERFACE_ENTRY(IOleCommandTarget)
+ COM_INTERFACE_ENTRY(IWebBrowserEventsUrlService)
+ COM_INTERFACE_ENTRY_CHAIN(Base)
+END_COM_MAP()
+
+BEGIN_MSG_MAP(ChromeActiveDocument)
+ CHAIN_MSG_MAP(Base)
+END_MSG_MAP()
+
+ HRESULT FinalConstruct();
+
+#define FORWARD_TAB_COMMAND(id, command) \
+ EXEC_COMMAND_HANDLER_GENERIC(id, GetTabProxy() ? GetTabProxy()->command() : 1)
+
+BEGIN_EXEC_COMMAND_MAP(ChromeActiveDocument)
+ EXEC_COMMAND_HANDLER_GENERIC(OLECMDID_PRINT, automation_client_->PrintTab())
+ EXEC_COMMAND_HANDLER_NO_ARGS(OLECMDID_FIND, OnFindInPage)
+ EXEC_COMMAND_HANDLER_NO_ARGS(UNDOC_IE_CONTEXTMENU_VIEWSOURCE, OnViewSource)
+ FORWARD_TAB_COMMAND(OLECMDID_SELECTALL, SelectAll)
+ FORWARD_TAB_COMMAND(OLECMDID_CUT, Cut)
+ FORWARD_TAB_COMMAND(OLECMDID_COPY, Copy)
+ FORWARD_TAB_COMMAND(OLECMDID_PASTE, Paste)
+ FORWARD_TAB_COMMAND(OLECMDID_REFRESH, ReloadAsync)
+ FORWARD_TAB_COMMAND(OLECMDID_STOP, StopAsync)
+END_EXEC_COMMAND_MAP()
+
+ // IPCs from automation server.
+ virtual void OnNavigationStateChanged(int tab_handle, int flags,
+ const IPC::NavigationInfo& nav_info);
+ virtual void OnUpdateTargetUrl(int tab_handle,
+ const std::wstring& new_target_url);
+ virtual void OnAcceleratorPressed(int tab_handle, const MSG& accel_message);
+ virtual void OnTabbedOut(int tab_handle, bool reverse);
+ virtual void OnDidNavigate(int tab_handle,
+ const IPC::NavigationInfo& nav_info);
+
+ void OnFindInPage();
+
+ // Override DoVerb
+ STDMETHOD(DoVerb)(LONG verb,
+ LPMSG msg,
+ IOleClientSite* active_site,
+ LONG index,
+ HWND parent_window,
+ LPCRECT pos);
+ STDMETHOD(InPlaceDeactivate)(void);
+
+ // Override IOleInPlaceActiveObjectImpl::OnDocWindowActivate
+ STDMETHOD(OnDocWindowActivate)(BOOL activate);
+ STDMETHOD(TranslateAccelerator)(MSG* msg);
+
+ // IPersistMoniker methods
+ STDMETHOD(GetClassID)(CLSID* class_id);
+ STDMETHOD(IsDirty)();
+ STDMETHOD(GetCurMoniker)(IMoniker** moniker_name);
+ STDMETHOD(Load)(BOOL fully_avalable,
+ IMoniker* moniker_name,
+ LPBC bind_context,
+ DWORD mode);
+ STDMETHOD(Save)(IMoniker* moniker_name,
+ LPBC bind_context,
+ BOOL remember);
+ STDMETHOD(SaveCompleted)(IMoniker* moniker_name,
+ LPBC bind_context);
+
+ // IOleCommandTarget methods
+ STDMETHOD(QueryStatus)(const GUID* cmd_group_guid,
+ ULONG number_of_commands,
+ OLECMD commands[],
+ OLECMDTEXT* command_text);
+ STDMETHOD(Exec)(const GUID* cmd_group_guid, DWORD command_id,
+ DWORD cmd_exec_opt,
+ VARIANT* in_args,
+ VARIANT* out_args);
+
+ // IWebBrowserEventsUrlService methods
+ STDMETHOD(GetUrlForEvents)(BSTR* url);
+
+ // ChromeFrameActivexBase overrides
+ HRESULT IOleObject_SetClientSite(IOleClientSite* client_site);
+
+ HRESULT ActiveXDocActivate(LONG verb);
+
+ // Callbacks from ChromeFramePlugin<T>
+ bool PreProcessContextMenu(HMENU menu);
+ bool HandleContextMenuCommand(UINT cmd);
+
+ // Should connections initiated by this class try to block
+ // responses served with the X-Frame-Options header?
+ bool is_frame_busting_enabled();
+
+ protected:
+ // ChromeFrameActivexBase overrides
+ virtual void OnOpenURL(int tab_handle, const GURL& url_to_open,
+ int open_disposition);
+
+ virtual void OnLoad(int tab_handle, const GURL& url);
+
+ // A helper method that updates our internal navigation state
+ // as well as IE's navigation state (viz Title and current URL).
+ // The navigation_flags is a TabContents::InvalidateTypes enum
+ void UpdateNavigationState(const IPC::NavigationInfo& nav_info);
+
+ TabProxy* GetTabProxy() const {
+ if (automation_client_.get())
+ return automation_client_->tab();
+ return NULL;
+ }
+
+ // Exec command handlers
+ void OnViewSource();
+
+ // Call exec on our site's command target
+ HRESULT IEExec(const GUID* cmd_group_guid, DWORD command_id,
+ DWORD cmd_exec_opt, VARIANT* in_args, VARIANT* out_args);
+
+ protected:
+ typedef std::map<int, bool> EnabledCommandsMap;
+
+ IPC::NavigationInfo navigation_info_;
+ bool is_doc_object_;
+
+ // This indicates whether this is the first navigation in this
+ // active document. It is initalize to true and it is set to false
+ // after we get a navigation notification from Chrome
+ bool first_navigation_;
+
+ // Our find dialog
+ CFFindDialog find_dialog_;
+
+ // Contains the list of enabled commands ids.
+ EnabledCommandsMap enabled_commands_map_;
+
+ // Set to true if the automation_client_ member is initialized from
+ // an existing ChromeActiveDocument instance which is going away and
+ // a new ChromeActiveDocument instance is taking its place.
+ bool is_automation_client_reused_;
+
+ public:
+ ScopedComPtr<IOleInPlaceFrame> in_place_frame_;
+ OLEINPLACEFRAMEINFO frame_info_;
+};
+
+OBJECT_ENTRY_AUTO(__uuidof(ChromeActiveDocument), ChromeActiveDocument)
+
+#endif // CHROME_FRAME_CHROME_ACTIVE_DOCUMENT_H_
diff --git a/chrome_frame/chrome_active_document.rgs b/chrome_frame/chrome_active_document.rgs
new file mode 100644
index 0000000..2fae04b
--- /dev/null
+++ b/chrome_frame/chrome_active_document.rgs
@@ -0,0 +1,67 @@
+HKLM {
+ NoRemove Software {
+ NoRemove Classes {
+ ChromeTab.ChromeActiveDocument.1 = s 'ChromeActiveDocument Class' {
+ CLSID = s '{3E1D0E7F-F5E3-44CC-AA6A-C0A637619AB8}'
+ 'DocObject' = s '0'
+ val EditFlags = d '65536'
+ }
+ ChromeTab.ChromeActiveDocument = s 'ChromeActiveDocument Class' {
+ CLSID = s '{3E1D0E7F-F5E3-44CC-AA6A-C0A637619AB8}'
+ CurVer = s 'ChromeTab.ChromeActiveDocument.1'
+ }
+ NoRemove CLSID {
+ ForceRemove {3E1D0E7F-F5E3-44CC-AA6A-C0A637619AB8} = s 'ChromeActiveDocument Class' {
+ ProgID = s 'ChromeTab.ChromeActiveDocument.1'
+ VersionIndependentProgID = s 'ChromeTab.ChromeActiveDocument'
+ ForceRemove 'Programmable'
+ InprocServer32 = s '%MODULE%' {
+ val ThreadingModel = s 'Apartment'
+ }
+ val AppID = s '%APPID%'
+ ForceRemove 'Control'
+ ForceRemove 'ToolboxBitmap32' = s '%MODULE%, 104'
+ 'DocObject' = s '0'
+ 'MiscStatus' = s '0' {
+ '1' = s '%OLEMISC%'
+ }
+ 'TypeLib' = s '{6F2664E1-FF6E-488A-BCD1-F4CA6001DFCC}'
+ 'Version' = s '1.0'
+ }
+ }
+ }
+ }
+}
+
+HKLM {
+ NoRemove Software {
+ NoRemove Classes {
+ NoRemove MIME {
+ NoRemove Database {
+ NoRemove s 'Content Type' {
+ 'application/chromepage' {
+ val CLSID = s '{3E1D0E7F-F5E3-44CC-AA6A-C0A637619AB8}'
+ }
+ }
+ }
+ }
+ }
+ }
+}
+
+HKCU {
+ NoRemove Software {
+ NoRemove Microsoft {
+ NoRemove Windows {
+ NoRemove CurrentVersion {
+ NoRemove 'Internet Settings' {
+ NoRemove 'Secure Mime Handlers' {
+ val ChromeTab.ChromeActiveDocument.1 = d '1'
+ val ChromeTab.ChromeActiveDocument = d '1'
+ }
+ }
+ }
+ }
+ }
+ }
+}
diff --git a/chrome_frame/chrome_frame.gyp b/chrome_frame/chrome_frame.gyp
new file mode 100644
index 0000000..8782076
--- /dev/null
+++ b/chrome_frame/chrome_frame.gyp
@@ -0,0 +1,725 @@
+# Copyright (c) 2009 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.
+
+{
+ 'variables': {
+ 'chromium_code': 1,
+ 'xul_sdk_dir': '../third_party/xulrunner-sdk/<(OS)',
+
+ # Keep the archive builder happy.
+ 'chrome_personalization%': 1,
+ 'use_syncapi_stub%': 0,
+
+ # Deps info.
+ 'xul_include_directories': [
+ # TODO(slightlyoff): pare these down. This makes it too easy to
+ # regress to using unfrozen FF interfaces.
+ '<(xul_sdk_dir)/include',
+ '<(xul_sdk_dir)/include/accessibility',
+ '<(xul_sdk_dir)/include/alerts',
+ '<(xul_sdk_dir)/include/appcomps',
+ '<(xul_sdk_dir)/include/appshell',
+ '<(xul_sdk_dir)/include/autocomplete',
+ '<(xul_sdk_dir)/include/autoconfig',
+ '<(xul_sdk_dir)/include/ax_common',
+ '<(xul_sdk_dir)/include/browser',
+ '<(xul_sdk_dir)/include/cairo',
+ '<(xul_sdk_dir)/include/caps',
+ '<(xul_sdk_dir)/include/chardet',
+ '<(xul_sdk_dir)/include/chrome',
+ '<(xul_sdk_dir)/include/commandhandler',
+ '<(xul_sdk_dir)/include/composer',
+ '<(xul_sdk_dir)/include/content',
+ '<(xul_sdk_dir)/include/contentprefs',
+ '<(xul_sdk_dir)/include/cookie',
+ '<(xul_sdk_dir)/include/crashreporter',
+ '<(xul_sdk_dir)/include/docshell',
+ '<(xul_sdk_dir)/include/dom',
+ '<(xul_sdk_dir)/include/downloads',
+ '<(xul_sdk_dir)/include/editor',
+ '<(xul_sdk_dir)/include/embed_base',
+ '<(xul_sdk_dir)/include/embedcomponents',
+ '<(xul_sdk_dir)/include/expat',
+ '<(xul_sdk_dir)/include/extensions',
+ '<(xul_sdk_dir)/include/exthandler',
+ '<(xul_sdk_dir)/include/exthelper',
+ '<(xul_sdk_dir)/include/fastfind',
+ '<(xul_sdk_dir)/include/feeds',
+ '<(xul_sdk_dir)/include/find',
+ '<(xul_sdk_dir)/include/gfx',
+ '<(xul_sdk_dir)/include/htmlparser',
+ '<(xul_sdk_dir)/include/imgicon',
+ '<(xul_sdk_dir)/include/imglib2',
+ '<(xul_sdk_dir)/include/inspector',
+ '<(xul_sdk_dir)/include/intl',
+ '<(xul_sdk_dir)/include/jar',
+ '<(xul_sdk_dir)/include/java',
+ '<(xul_sdk_dir)/include/jpeg',
+ '<(xul_sdk_dir)/include/js',
+ '<(xul_sdk_dir)/include/jsdebug',
+ '<(xul_sdk_dir)/include/jsurl',
+ '<(xul_sdk_dir)/include/layout',
+ '<(xul_sdk_dir)/include/lcms',
+ '<(xul_sdk_dir)/include/libbz2',
+ '<(xul_sdk_dir)/include/libmar',
+ '<(xul_sdk_dir)/include/libpixman',
+ '<(xul_sdk_dir)/include/libreg',
+ '<(xul_sdk_dir)/include/liveconnect',
+ '<(xul_sdk_dir)/include/locale',
+ '<(xul_sdk_dir)/include/loginmgr',
+ '<(xul_sdk_dir)/include/lwbrk',
+ '<(xul_sdk_dir)/include/mimetype',
+ '<(xul_sdk_dir)/include/morkreader',
+ '<(xul_sdk_dir)/include/necko',
+ '<(xul_sdk_dir)/include/nkcache',
+ '<(xul_sdk_dir)/include/nspr',
+ '<(xul_sdk_dir)/include/nss',
+ '<(xul_sdk_dir)/include/oji',
+ '<(xul_sdk_dir)/include/parentalcontrols',
+ '<(xul_sdk_dir)/include/pipboot',
+ '<(xul_sdk_dir)/include/pipnss',
+ '<(xul_sdk_dir)/include/pippki',
+ '<(xul_sdk_dir)/include/places',
+ '<(xul_sdk_dir)/include/plugin',
+ '<(xul_sdk_dir)/include/png',
+ '<(xul_sdk_dir)/include/pref',
+ '<(xul_sdk_dir)/include/prefetch',
+ '<(xul_sdk_dir)/include/profdirserviceprovider',
+ '<(xul_sdk_dir)/include/profile',
+ '<(xul_sdk_dir)/include/rdf',
+ '<(xul_sdk_dir)/include/rdfutil',
+ '<(xul_sdk_dir)/include/satchel',
+ '<(xul_sdk_dir)/include/shistory',
+ '<(xul_sdk_dir)/include/simple',
+ '<(xul_sdk_dir)/include/spellchecker',
+ '<(xul_sdk_dir)/include/sqlite3',
+ '<(xul_sdk_dir)/include/storage',
+ '<(xul_sdk_dir)/include/string',
+ '<(xul_sdk_dir)/include/thebes',
+ '<(xul_sdk_dir)/include/toolkitcomps',
+ '<(xul_sdk_dir)/include/txmgr',
+ '<(xul_sdk_dir)/include/txtsvc',
+ '<(xul_sdk_dir)/include/uconv',
+ '<(xul_sdk_dir)/include/ucvcn',
+ '<(xul_sdk_dir)/include/ucvibm',
+ '<(xul_sdk_dir)/include/ucvja',
+ '<(xul_sdk_dir)/include/ucvko',
+ '<(xul_sdk_dir)/include/ucvlatin',
+ '<(xul_sdk_dir)/include/ucvmath',
+ '<(xul_sdk_dir)/include/ucvtw',
+ '<(xul_sdk_dir)/include/ucvtw2',
+ '<(xul_sdk_dir)/include/unicharutil',
+ '<(xul_sdk_dir)/include/update',
+ '<(xul_sdk_dir)/include/uriloader',
+ '<(xul_sdk_dir)/include/urlformatter',
+ '<(xul_sdk_dir)/include/util',
+ '<(xul_sdk_dir)/include/view',
+ '<(xul_sdk_dir)/include/webbrowserpersist',
+ '<(xul_sdk_dir)/include/webbrwsr',
+ '<(xul_sdk_dir)/include/webshell',
+ '<(xul_sdk_dir)/include/widget',
+ '<(xul_sdk_dir)/include/windowwatcher',
+ '<(xul_sdk_dir)/include/xml',
+ '<(xul_sdk_dir)/include/xml-rpc',
+ '<(xul_sdk_dir)/include/xpcom',
+ '<(xul_sdk_dir)/include/xpconnect',
+ '<(xul_sdk_dir)/include/xpinstall',
+ '<(xul_sdk_dir)/include/xulapp',
+ '<(xul_sdk_dir)/include/xuldoc',
+ '<(xul_sdk_dir)/include/xulrunner',
+ '<(xul_sdk_dir)/include/xultmpl',
+ '<(xul_sdk_dir)/include/zipwriter',
+ '<(xul_sdk_dir)/include/zlib',
+ '<(xul_sdk_dir)/sdk/include',
+ ],
+ },
+ 'includes': [
+ '../build/common.gypi',
+ ],
+ 'target_defaults': {
+ 'dependencies': [
+ '../chrome/chrome.gyp:chrome_resources',
+ '../chrome/chrome.gyp:chrome_strings',
+ '../chrome/chrome.gyp:theme_resources',
+ '../skia/skia.gyp:skia',
+ '../third_party/npapi/npapi.gyp:npapi',
+ ],
+ 'include_dirs': [
+ # all our own includes are relative to src/
+ '..',
+ ],
+ },
+ 'targets': [
+ {
+ # TODO(slightlyoff): de-win23-ify
+ 'target_name': 'xulrunner_sdk',
+ 'type': 'none',
+ 'conditions': [
+ ['OS=="win"', {
+ 'direct_dependent_settings': {
+ 'include_dirs': [
+ '<@(xul_include_directories)',
+ ],
+ 'libraries': [
+ '../third_party/xulrunner-sdk/win/lib/xpcomglue_s.lib',
+ '../third_party/xulrunner-sdk/win/lib/xpcom.lib',
+ '../third_party/xulrunner-sdk/win/lib/nspr4.lib',
+ ],
+ },
+ },],
+ ],
+ },
+ {
+ # build the ICU stubs
+ 'target_name': 'icu_stubs',
+ 'type': 'static_library',
+ 'dependencies': [
+ '../base/base.gyp:base',
+ ],
+ 'sources': [
+ 'icu_stubs.cc'
+ ],
+ },
+ {
+ # TODO(slightlyoff): de-win32-ify
+ #
+ # build the base_noicu.lib.
+ 'target_name': 'base_noicu',
+ 'type': 'none',
+ 'dependencies': [
+ '../base/base.gyp:base',
+ 'icu_stubs',
+ ],
+ 'actions': [
+ {
+ 'action_name': 'combine_libs',
+ 'msvs_cygwin_shell': 0,
+ 'inputs': [
+ '<(PRODUCT_DIR)/lib/base.lib',
+ '<(PRODUCT_DIR)/lib/icu_stubs.lib',
+ ],
+ 'outputs': [
+ '<(PRODUCT_DIR)/lib/base_noicu.lib',
+ ],
+ 'action': [
+ 'python',
+ 'combine_libs.py',
+ '-o <(_outputs)',
+ '-r (icu_|_icu.obj)',
+ '<@(_inputs)'],
+ },
+ ],
+ 'direct_dependent_settings': {
+ # linker_settings
+ 'libraries': [
+ '<(PRODUCT_DIR)/lib/base_noicu.lib',
+ ],
+ },
+ },
+ {
+ 'target_name': 'chrome_frame_unittests',
+ 'type': 'executable',
+ 'dependencies': [
+ '../build/temp_gyp/googleurl.gyp:googleurl',
+ '../chrome/chrome.gyp:common',
+ '../chrome/chrome.gyp:utility',
+ '../testing/gmock.gyp:gmock',
+ '../testing/gtest.gyp:gtest',
+ 'base_noicu',
+ 'icu_stubs',
+ 'chrome_frame_npapi',
+ 'chrome_frame_strings',
+ 'xulrunner_sdk',
+ ],
+ 'sources': [
+ 'chrome_frame_npapi_unittest.cc',
+ 'chrome_frame_unittest_main.cc',
+ 'chrome_launcher_unittest.cc',
+ 'unittest_precompile.h',
+ 'unittest_precompile.cc',
+ 'urlmon_upload_data_stream.cc',
+ 'urlmon_upload_data_stream_unittest.cc',
+ 'chrome_frame_histograms.h',
+ 'chrome_frame_histograms.cc',
+ ],
+ 'include_dirs': [
+ # To allow including "chrome_tab.h"
+ '<(INTERMEDIATE_DIR)',
+ ],
+ 'resource_include_dirs': [
+ '<(INTERMEDIATE_DIR)',
+ ],
+ 'conditions': [
+ ['OS=="win"', {
+ 'msvs_settings': {
+ 'VCLinkerTool': {
+ 'DelayLoadDLLs': ['xpcom.dll', 'nspr4.dll'],
+ },
+ },
+ 'sources': [
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/chrome_frame_resources.rc',
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/chrome_frame_strings.rc',
+ ],
+ 'dependencies': [
+ # TODO(slightlyoff): Get automation targets working on OS X
+ '../chrome/chrome.gyp:automation',
+ '../chrome/installer/installer.gyp:installer_util',
+ '../google_update/google_update.gyp:google_update',
+ ]
+ }],
+ ],
+ },
+ {
+ 'target_name': 'chrome_frame_tests',
+ 'type': 'executable',
+ 'dependencies': [
+ # 'base_noicu',
+ '../build/temp_gyp/googleurl.gyp:googleurl',
+ '../chrome/chrome.gyp:common',
+ '../chrome/chrome.gyp:utility',
+ '../testing/gmock.gyp:gmock',
+ '../testing/gtest.gyp:gtest',
+ '../third_party/libxml/libxml.gyp:libxml',
+ '../third_party/libxslt/libxslt.gyp:libxslt',
+ 'chrome_frame_strings',
+ 'chrome_frame_npapi',
+ # 'npchrome_tab',
+ 'xulrunner_sdk',
+ ],
+ 'sources': [
+ '../base/test_suite.h',
+ 'test/chrome_frame_test_utils.cc',
+ 'test/chrome_frame_test_utils.h',
+ 'test/chrome_frame_automation_mock.cc',
+ 'test/chrome_frame_automation_mock.h',
+ 'test/chrome_frame_unittests.cc',
+ 'test/chrome_frame_unittests.h',
+ 'test/com_message_event_unittest.cc',
+ 'test/function_stub_unittest.cc',
+ 'test/html_util_unittests.cc',
+ 'test/http_server.cc',
+ 'test/http_server.h',
+ 'test/icu_stubs_unittests.cc',
+ 'test/run_all_unittests.cc',
+ 'test/test_server.cc',
+ 'test/test_server.h',
+ 'test/test_server_test.cc',
+ 'test/util_unittests.cc',
+ 'chrome_frame_automation.cc',
+ 'chrome_tab.h',
+ 'chrome_tab.idl',
+ 'com_message_event.cc',
+ 'html_utils.cc',
+ 'sync_msg_reply_dispatcher.cc',
+ 'sync_msg_reply_dispatcher.h',
+ 'test_utils.cc',
+ 'test_utils.h',
+ 'utils.cc',
+ 'utils.h',
+ 'chrome_frame_histograms.h',
+ 'chrome_frame_histograms.cc',
+ ],
+ 'include_dirs': [
+ '<@(xul_include_directories)',
+ '../chrome/third_party/wtl/include',
+ # To allow including "chrome_tab.h"
+ '<(INTERMEDIATE_DIR)',
+ ],
+ 'resource_include_dirs': [
+ '<(INTERMEDIATE_DIR)',
+ ],
+ 'conditions': [
+ ['OS=="win"', {
+ 'msvs_settings': {
+ 'VCLinkerTool': {
+ 'DelayLoadDLLs': ['xpcom.dll', 'nspr4.dll'],
+ },
+ },
+ 'sources': [
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/chrome_frame_resources.rc',
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/chrome_frame_strings.rc',
+ ],
+ 'dependencies': [
+ '../chrome/chrome.gyp:automation',
+ '../chrome/installer/installer.gyp:installer_util',
+ '../google_update/google_update.gyp:google_update',
+ ]
+ }],
+ ],
+ },
+ {
+ 'target_name': 'chrome_frame_perftests',
+ 'type': 'executable',
+ 'dependencies': [
+ '../base/base.gyp:base_gfx',
+ '../base/base.gyp:test_support_base',
+ '../build/temp_gyp/googleurl.gyp:googleurl',
+ '../chrome/chrome.gyp:common',
+ '../chrome/chrome.gyp:utility',
+ '../testing/gmock.gyp:gmock',
+ '../testing/gtest.gyp:gtest',
+ '../third_party/libxml/libxml.gyp:libxml',
+ '../third_party/libxslt/libxslt.gyp:libxslt',
+ 'chrome_frame_strings',
+ 'xulrunner_sdk',
+ ],
+ 'sources': [
+ '../base/perf_test_suite.h',
+ '../base/perftimer.cc',
+ '../base/test_file_util.h',
+ '../chrome/test/chrome_process_util.cc',
+ '../chrome/test/chrome_process_util.h',
+ '../chrome/test/ui/ui_test.cc',
+ 'chrome_tab.h',
+ 'chrome_tab.idl',
+ 'html_utils.cc',
+ 'test/perf/chrome_frame_perftest.cc',
+ 'test/perf/chrome_frame_perftest.h',
+ 'test/perf/run_all.cc',
+ 'test/perf/silverlight.cc',
+ 'test_utils.cc',
+ 'test_utils.h',
+ 'utils.cc',
+ 'utils.h',
+ ],
+ 'include_dirs': [
+ '<@(xul_include_directories)',
+ '../chrome/third_party/wtl/include',
+ # To allow including "chrome_tab.h"
+ '<(INTERMEDIATE_DIR)',
+ ],
+ 'conditions': [
+ ['OS=="win"', {
+ 'dependencies': [
+ '../chrome/chrome.gyp:automation',
+ '../breakpad/breakpad.gyp:breakpad_handler',
+ '../chrome/installer/installer.gyp:installer_util',
+ '../google_update/google_update.gyp:google_update',
+ '../chrome/installer/installer.gyp:installer_util',
+ ],
+ 'sources': [
+ '../chrome/test/perf/mem_usage_win.cc',
+ '../chrome/test/chrome_process_util_win.cc',
+ '../base/test_file_util_win.cc',
+ ]
+ }],
+ ],
+ },
+
+ {
+ 'target_name': 'chrome_frame_net_tests',
+ 'type': 'executable',
+ 'dependencies': [
+ '../base/base.gyp:test_support_base',
+ '../chrome/chrome.gyp:browser',
+ '../chrome/chrome.gyp:chrome_dll_version',
+ '../chrome/chrome.gyp:chrome_resources',
+ '../chrome/chrome.gyp:debugger',
+ '../chrome/chrome.gyp:renderer',
+ '../chrome/chrome.gyp:syncapi',
+ '../skia/skia.gyp:skia',
+ '../testing/gtest.gyp:gtest',
+ '../third_party/icu/icu.gyp:icui18n',
+ '../third_party/icu/icu.gyp:icuuc',
+ 'chrome_frame_npapi',
+ ],
+ 'sources': [
+ '../net/url_request/url_request_unittest.cc',
+ '../net/url_request/url_request_unittest.h',
+ 'test/chrome_frame_test_utils.cc',
+ 'test/chrome_frame_test_utils.h',
+ 'test/test_server.cc',
+ 'test/test_server.h',
+ 'test/net/dialog_watchdog.cc',
+ 'test/net/dialog_watchdog.h',
+ 'test/net/fake_external_tab.cc',
+ 'test/net/fake_external_tab.h',
+ 'test/net/process_singleton_subclass.cc',
+ 'test/net/process_singleton_subclass.h',
+ 'test/net/test_automation_provider.cc',
+ 'test/net/test_automation_provider.h',
+ 'test/net/test_automation_resource_message_filter.cc',
+ 'test/net/test_automation_resource_message_filter.h',
+ ],
+ 'include_dirs': [
+ # To allow including "chrome_tab.h"
+ '<(INTERMEDIATE_DIR)',
+ ],
+ 'conditions': [
+ ['OS=="win"', {
+ 'dependencies': [
+ '../chrome/chrome.gyp:automation',
+ '../breakpad/breakpad.gyp:breakpad_handler',
+ '../chrome/installer/installer.gyp:installer_util',
+ '../google_update/google_update.gyp:google_update',
+ '../chrome/installer/installer.gyp:installer_util',
+ ]
+ }],
+ ],
+ },
+
+ {
+ 'target_name': 'chrome_frame_npapi',
+ 'type': 'static_library',
+ 'dependencies': [
+ 'chrome_frame_strings',
+ '../chrome/chrome.gyp:common',
+ 'xulrunner_sdk',
+ ],
+ 'sources': [
+ 'chrome_frame_automation.cc',
+ 'chrome_frame_automation.h',
+ 'chrome_frame_delegate.cc',
+ 'chrome_frame_delegate.h',
+ 'chrome_frame_plugin.h',
+ 'chrome_frame_npapi.cc',
+ 'chrome_frame_npapi.h',
+ 'chrome_launcher.cc',
+ 'chrome_launcher.h',
+ 'html_utils.cc',
+ 'html_utils.h',
+ 'np_browser_functions.cc',
+ 'np_browser_functions.h',
+ 'np_event_listener.cc',
+ 'np_event_listener.h',
+ 'np_proxy_service.cc',
+ 'np_proxy_service.h',
+ 'npapi_url_request.cc',
+ 'npapi_url_request.h',
+ 'ns_associate_iid_win.h',
+ 'ns_isupports_impl.h',
+ 'plugin_url_request.cc',
+ 'plugin_url_request.h',
+ 'scoped_ns_ptr_win.h',
+ 'sync_msg_reply_dispatcher.cc',
+ 'sync_msg_reply_dispatcher.h',
+ 'utils.cc',
+ 'utils.h',
+ ],
+ },
+ {
+ 'target_name': 'chrome_launcher',
+ 'type': 'executable',
+ 'msvs_guid': 'B7E540C1-49D9-4350-ACBC-FB8306316D16',
+ 'dependencies': [],
+ 'sources': [
+ 'chrome_launcher_main.cc',
+ ],
+ 'msvs_settings': {
+ 'VCLinkerTool': {
+ 'OutputFile':
+ '..\\chrome\\$(ConfigurationName)\\servers\\$(ProjectName).exe',
+ # Set /SUBSYSTEM:WINDOWS since this is not a command-line program.
+ 'SubSystem': '2',
+ # We're going for minimal size, so no standard library (in release
+ # builds).
+ 'IgnoreAllDefaultLibraries': "true",
+ },
+ 'VCCLCompilerTool': {
+ # Requires standard library, so disable it.
+ 'BufferSecurityCheck': "false",
+ },
+ },
+ 'configurations': {
+ # Bring back the standard library in debug buidls.
+ 'Debug': {
+ 'msvs_settings': {
+ 'VCLinkerTool': {
+ 'IgnoreAllDefaultLibraries': "false",
+ },
+ },
+ },
+ },
+ },
+ {
+ 'target_name': 'chrome_frame_strings',
+ 'type': 'none',
+ 'rules': [
+ {
+ 'rule_name': 'grit',
+ 'extension': 'grd',
+ 'inputs': [
+ '../tools/grit/grit.py',
+ ],
+ 'variables': {
+ 'grit_out_dir': '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab',
+ },
+ 'outputs': [
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/grit/<(RULE_INPUT_ROOT).h',
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/<(RULE_INPUT_ROOT).pak',
+ ],
+ 'action': ['python', '<@(_inputs)', '-i',
+ '<(RULE_INPUT_PATH)',
+ 'build', '-o', '<(grit_out_dir)'
+ ],
+ 'message': 'Generating resources from <(RULE_INPUT_PATH)',
+ },
+ ],
+ 'sources': [
+ # Localizable resources.
+ 'resources/chrome_frame_strings.grd',
+ 'resources/chrome_frame_resources.grd',
+ ],
+ 'direct_dependent_settings': {
+ 'include_dirs': [
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab',
+ ],
+ },
+ 'conditions': [
+ ['OS=="win"', {
+ 'dependencies': ['../build/win/system.gyp:cygwin'],
+ }],
+ ],
+ },
+ {
+ 'target_name': 'npchrome_tab',
+ 'type': 'shared_library',
+ 'msvs_guid': 'E3DE7E63-D3B6-4A9F-BCC4-5C8169E9C9F2',
+ 'dependencies': [
+ 'base_noicu',
+ 'chrome_frame_npapi',
+ 'chrome_frame_strings',
+ 'chrome_launcher',
+ 'xulrunner_sdk',
+ '../chrome/chrome.gyp:common',
+ '../chrome/chrome.gyp:utility',
+ '../build/temp_gyp/googleurl.gyp:googleurl',
+ # FIXME(slightlyoff):
+ # gigantic hack to get these to build from main Chrome sln.
+ 'chrome_frame_perftests',
+ 'chrome_frame_tests',
+ 'chrome_frame_unittests',
+ 'chrome_frame_net_tests',
+ ],
+ 'sources': [
+ 'bho.cc',
+ 'bho.h',
+ 'bho.rgs',
+ 'chrome_active_document.bmp',
+ 'chrome_active_document.cc',
+ 'chrome_active_document.h',
+ 'chrome_active_document.rgs',
+ 'chrome_frame_activex.cc',
+ 'chrome_frame_activex.h',
+ 'chrome_frame_activex_base.h',
+ 'chrome_frame_activex.rgs',
+ 'chrome_frame_npapi.rgs',
+ 'chrome_frame_npapi_entrypoints.cc',
+ 'chrome_protocol.cc',
+ 'chrome_protocol.h',
+ 'chrome_protocol.rgs',
+ 'chrome_tab.cc',
+ 'chrome_tab.def',
+ 'chrome_tab.h',
+ 'chrome_tab.idl',
+ # FIXME(slightlyoff): For chrome_tab.tlb. Giant hack until we can
+ # figure out something more gyp-ish.
+ 'resources/tlb_resource.rc',
+ 'chrome_tab.rgs',
+ 'chrome_tab_version.rc.version',
+ 'com_message_event.cc',
+ 'com_message_event.h',
+ 'com_type_info_holder.cc',
+ 'com_type_info_holder.h',
+ 'crash_report.cc',
+ 'crash_report.h',
+ 'ff_30_privilege_check.cc',
+ 'ff_privilege_check.h',
+ 'find_dialog.cc',
+ 'find_dialog.h',
+ 'function_stub.h',
+ 'icu_stubs.cc',
+ 'iids.cc',
+ 'in_place_menu.h',
+ 'ole_document_impl.h',
+ 'protocol_sink_wrap.cc',
+ 'protocol_sink_wrap.h',
+ 'resource.h',
+ 'script_security_manager.h',
+ 'sync_msg_reply_dispatcher.cc',
+ 'sync_msg_reply_dispatcher.h',
+ 'extra_system_apis.h',
+ 'urlmon_url_request.cc',
+ 'urlmon_url_request.h',
+ 'urlmon_upload_data_stream.cc',
+ 'urlmon_upload_data_stream.h',
+ 'vectored_handler-impl.h',
+ 'vectored_handler.h',
+ 'vtable_patch_manager.cc',
+ 'vtable_patch_manager.h',
+ 'chrome_frame_histograms.h',
+ 'chrome_frame_histograms.cc',
+ ],
+ 'include_dirs': [
+ # To allow including "chrome_tab.h"
+ '<(INTERMEDIATE_DIR)',
+ '<(INTERMEDIATE_DIR)/../npchrome_tab',
+ ],
+ 'conditions': [
+ ['OS=="win"', {
+ # NOTE(slightlyoff):
+ # this is a fix for the include dirs length limit on the resource
+ # compiler, tickled by the xul_include_dirs variable
+ 'resource_include_dirs': [
+ '<(INTERMEDIATE_DIR)'
+ ],
+ 'sources': [
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/chrome_frame_resources.rc',
+ '<(SHARED_INTERMEDIATE_DIR)/ie_alt_tab/chrome_frame_strings.rc',
+ ],
+ 'dependencies': [
+ '../breakpad/breakpad.gyp:breakpad_handler',
+ '../chrome/chrome.gyp:automation',
+ # Make the archive build happy.
+ '../chrome/chrome.gyp:syncapi',
+ # Installer
+ '../chrome/installer/installer.gyp:installer_util',
+ '../google_update/google_update.gyp:google_update',
+ ],
+ 'msvs_settings': {
+ 'VCLinkerTool': {
+ 'OutputFile':
+ '..\\chrome\\$(ConfigurationName)\\servers\\$(ProjectName).dll',
+ 'DelayLoadDLLs': ['xpcom.dll', 'nspr4.dll'],
+ 'BaseAddress': '0x33000000',
+ # Set /SUBSYSTEM:WINDOWS (for consistency).
+ 'SubSystem': '2',
+ },
+ },
+ }],
+ ],
+ 'rules': [
+ # Borrowed from chrome.gyp:chrome_dll_version, branding references
+ # removed
+ {
+ 'rule_name': 'version',
+ 'extension': 'version',
+ 'variables': {
+ 'version_py': '../chrome/tools/build/version.py',
+ 'version_path': '../chrome/VERSION',
+ 'template_input_path': 'chrome_tab_version.rc.version',
+ },
+ 'inputs': [
+ '<(template_input_path)',
+ '<(version_path)',
+ ],
+ 'outputs': [
+ 'chrome_tab_version.rc',
+ ],
+ 'action': [
+ 'python',
+ '<(version_py)',
+ '-f', '<(version_path)',
+ '<(template_input_path)',
+ '<@(_outputs)',
+ ],
+ 'process_outputs_as_sources': 1,
+ 'message': 'Generating version information in <(_outputs)'
+ },
+ ],
+ },
+ ],
+}
+
+# vim: shiftwidth=2:et:ai:tabstop=2
diff --git a/chrome_frame/chrome_frame_activex.cc b/chrome_frame/chrome_frame_activex.cc
new file mode 100644
index 0000000..e58a961
--- /dev/null
+++ b/chrome_frame/chrome_frame_activex.cc
@@ -0,0 +1,505 @@
+// Copyright (c) 2009 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 "chrome_frame/chrome_frame_activex.h"
+
+#include <shdeprecated.h> // for IBrowserService2
+#include <wininet.h>
+
+#include <algorithm>
+
+#include "base/basictypes.h"
+#include "base/command_line.h"
+#include "base/file_util.h"
+#include "base/logging.h"
+#include "base/path_service.h"
+#include "base/process_util.h"
+#include "base/scoped_bstr_win.h"
+#include "base/string_util.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/common/chrome_switches.h"
+#include "chrome/test/automation/tab_proxy.h"
+#include "googleurl/src/gurl.h"
+#include "chrome_frame/com_message_event.h"
+#include "chrome_frame/utils.h"
+
+ChromeFrameActivex::ChromeFrameActivex() {
+}
+
+HRESULT ChromeFrameActivex::FinalConstruct() {
+ HRESULT hr = Base::FinalConstruct();
+ if (FAILED(hr))
+ return hr;
+
+ // No need to call FireOnChanged at this point since nobody will be listening.
+ ready_state_ = READYSTATE_LOADING;
+ return S_OK;
+}
+
+ChromeFrameActivex::~ChromeFrameActivex() {
+ // We expect these to be released during a call to SetClientSite(NULL).
+ DCHECK(onmessage_.size() == 0);
+ DCHECK(onloaderror_.size() == 0);
+ DCHECK(onload_.size() == 0);
+ DCHECK(onreadystatechanged_.size() == 0);
+}
+
+LRESULT ChromeFrameActivex::OnCreate(UINT message, WPARAM wparam, LPARAM lparam,
+ BOOL& handled) {
+ Base::OnCreate(message, wparam, lparam, handled);
+ return 0;
+}
+
+void ChromeFrameActivex::OnAcceleratorPressed(int tab_handle,
+ const MSG& accel_message) {
+ DCHECK(m_spInPlaceSite != NULL);
+ // Allow our host a chance to handle the accelerator.
+ // This catches things like Ctrl+F, Ctrl+O etc, but not browser
+ // accelerators such as F11, Ctrl+T etc.
+ // (see AllowFrameToTranslateAccelerator for those).
+ HRESULT hr = TranslateAccelerator(const_cast<MSG*>(&accel_message));
+ if (hr != S_OK)
+ hr = AllowFrameToTranslateAccelerator(accel_message);
+
+ DLOG(INFO) << __FUNCTION__ << " browser response: "
+ << StringPrintf("0x%08x", hr);
+
+ // Last chance to handle the keystroke is to pass it to chromium.
+ // We do this last partially because there's no way for us to tell if
+ // chromium actually handled the keystroke, but also since the browser
+ // should have first dibs anyway.
+ if (hr != S_OK && automation_client_.get()) {
+ TabProxy* tab = automation_client_->tab();
+ if (tab) {
+ tab->ProcessUnhandledAccelerator(accel_message);
+ }
+ }
+}
+
+HRESULT ChromeFrameActivex::GetContainingDocument(IHTMLDocument2** doc) {
+ ScopedComPtr<IOleContainer> container;
+ HRESULT hr = m_spClientSite->GetContainer(container.Receive());
+ if (container)
+ hr = container.QueryInterface(doc);
+ return hr;
+}
+
+HRESULT ChromeFrameActivex::GetDocumentWindow(IHTMLWindow2** window) {
+ ScopedComPtr<IHTMLDocument2> document;
+ HRESULT hr = GetContainingDocument(document.Receive());
+ if (document)
+ hr = document->get_parentWindow(window);
+ return hr;
+}
+
+void ChromeFrameActivex::OnLoad(int tab_handle, const GURL& gurl) {
+ ScopedComPtr<IDispatch> event;
+ std::string url = gurl.spec();
+ if (SUCCEEDED(CreateDomEvent("event", url, "", event.Receive())))
+ Fire_onload(event);
+
+ FireEvent(onload_, url);
+
+ HRESULT hr = InvokeScriptFunction(onload_handler_, url);
+
+ if (ready_state_ < READYSTATE_COMPLETE) {
+ ready_state_ = READYSTATE_COMPLETE;
+ FireOnChanged(DISPID_READYSTATE);
+ }
+}
+
+void ChromeFrameActivex::OnLoadFailed(int error_code, const std::string& url) {
+ ScopedComPtr<IDispatch> event;
+ if (SUCCEEDED(CreateDomEvent("event", url, "", event.Receive())))
+ Fire_onloaderror(event);
+
+ FireEvent(onloaderror_, url);
+
+ HRESULT hr = InvokeScriptFunction(onerror_handler_, url);
+}
+
+void ChromeFrameActivex::OnMessageFromChromeFrame(int tab_handle,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& target) {
+ DLOG(INFO) << __FUNCTION__;
+
+ if (target.compare("*") != 0) {
+ bool drop = true;
+
+ if (is_privileged_) {
+ // Forward messages if the control is in privileged mode.
+ ScopedComPtr<IDispatch> message_event;
+ if (SUCCEEDED(CreateDomEvent("message", message, origin,
+ message_event.Receive()))) {
+ ScopedBstr target_bstr(UTF8ToWide(target).c_str());
+ Fire_onprivatemessage(message_event, target_bstr);
+
+ FireEvent(onprivatemessage_, message_event, target_bstr);
+ }
+ } else {
+ if (HaveSameOrigin(target, document_url_)) {
+ drop = false;
+ } else {
+ DLOG(WARNING) << "Dropping posted message since target doesn't match "
+ "the current document's origin. target=" << target;
+ }
+ }
+
+ if (drop)
+ return;
+ }
+
+ ScopedComPtr<IDispatch> message_event;
+ if (SUCCEEDED(CreateDomEvent("message", message, origin,
+ message_event.Receive()))) {
+ Fire_onmessage(message_event);
+
+ FireEvent(onmessage_, message_event);
+
+ ScopedVariant event_var;
+ event_var.Set(static_cast<IDispatch*>(message_event));
+ InvokeScriptFunction(onmessage_handler_, event_var.AsInput());
+ }
+}
+
+void ChromeFrameActivex::OnAutomationServerLaunchFailed(
+ AutomationLaunchResult reason, const std::string& server_version) {
+ Base::OnAutomationServerLaunchFailed(reason, server_version);
+
+ if (reason == AUTOMATION_VERSION_MISMATCH) {
+ DisplayVersionMismatchWarning(m_hWnd, server_version);
+ }
+}
+
+HRESULT ChromeFrameActivex::InvokeScriptFunction(const VARIANT& script_object,
+ const std::string& param) {
+ ScopedVariant script_arg(UTF8ToWide(param.c_str()).c_str());
+ return InvokeScriptFunction(script_object, script_arg.AsInput());
+}
+
+HRESULT ChromeFrameActivex::InvokeScriptFunction(const VARIANT& script_object,
+ VARIANT* param) {
+ if (V_VT(&script_object) != VT_DISPATCH) {
+ return S_FALSE;
+ }
+
+ CComPtr<IDispatch> script(script_object.pdispVal);
+ HRESULT hr = script.Invoke1(static_cast<DISPID>(DISPID_VALUE), param);
+ // 0x80020101 == SCRIPT_E_REPORTED.
+ // When the script we're invoking has an error, we get this error back.
+ DLOG_IF(ERROR, FAILED(hr) && hr != 0x80020101) << "Failed to invoke script";
+
+ return hr;
+}
+
+HRESULT ChromeFrameActivex::OnDraw(ATL_DRAWINFO& draw_info) { // NO_LINT
+ HRESULT hr = S_OK;
+ int dc_type = ::GetObjectType(draw_info.hicTargetDev);
+ if (dc_type == OBJ_ENHMETADC) {
+ RECT print_bounds = {0};
+ print_bounds.left = draw_info.prcBounds->left;
+ print_bounds.right = draw_info.prcBounds->right;
+ print_bounds.top = draw_info.prcBounds->top;
+ print_bounds.bottom = draw_info.prcBounds->bottom;
+
+ automation_client_->Print(draw_info.hdcDraw, print_bounds);
+ } else {
+ hr = Base::OnDraw(draw_info);
+ }
+
+ return hr;
+}
+
+STDMETHODIMP ChromeFrameActivex::Load(IPropertyBag* bag, IErrorLog* error_log) {
+ DCHECK(bag);
+
+ const wchar_t* event_props[] = {
+ (L"onload"),
+ (L"onloaderror"),
+ (L"onmessage"),
+ (L"onreadystatechanged"),
+ };
+
+ ScopedComPtr<IHTMLObjectElement> obj_element;
+ GetObjectElement(obj_element.Receive());
+
+ ScopedBstr object_id;
+ GetObjectScriptId(obj_element, object_id.Receive());
+
+ ScopedComPtr<IHTMLElement2> element;
+ element.QueryFrom(obj_element);
+ HRESULT hr = S_OK;
+
+ for (int i = 0; SUCCEEDED(hr) && i < arraysize(event_props); ++i) {
+ ScopedBstr prop(event_props[i]);
+ ScopedVariant value;
+ if (SUCCEEDED(bag->Read(prop, value.Receive(), error_log))) {
+ if (value.type() != VT_BSTR ||
+ FAILED(hr = CreateScriptBlockForEvent(element, object_id,
+ V_BSTR(&value), prop))) {
+ DLOG(ERROR) << "Failed to create script block for " << prop
+ << StringPrintf(L"hr=0x%08X, vt=%i", hr, value.type());
+ } else {
+ DLOG(INFO) << "script block created for event " << prop <<
+ StringPrintf(" (0x%08X)", hr) << " connections: " <<
+ ProxyDIChromeFrameEvents<ChromeFrameActivex>::m_vec.GetSize();
+ }
+ } else {
+ DLOG(INFO) << "event property " << prop << " not in property bag";
+ }
+ }
+
+ ScopedVariant src;
+ if (SUCCEEDED(bag->Read(StackBstr(L"src"), src.Receive(), error_log))) {
+ if (src.type() == VT_BSTR) {
+ hr = put_src(V_BSTR(&src));
+ DCHECK(hr != E_UNEXPECTED);
+ }
+ }
+
+ ScopedVariant use_chrome_network;
+ if (SUCCEEDED(bag->Read(StackBstr(L"useChromeNetwork"),
+ use_chrome_network.Receive(), error_log))) {
+ VariantChangeType(use_chrome_network.AsInput(),
+ use_chrome_network.AsInput(),
+ 0, VT_BOOL);
+ if (use_chrome_network.type() == VT_BOOL) {
+ hr = put_useChromeNetwork(V_BOOL(&use_chrome_network));
+ DCHECK(hr != E_UNEXPECTED);
+ }
+ }
+
+ DLOG_IF(ERROR, FAILED(hr))
+ << StringPrintf("Failed to load property bag: 0x%08X", hr);
+
+ return hr;
+}
+
+const wchar_t g_activex_mixed_content_error[] = {
+ L"data:text/html,<html><body><b>ChromeFrame Security Error<br><br>"
+ L"Cannot navigate to HTTP url when document URL is HTTPS</body></html>"};
+
+STDMETHODIMP ChromeFrameActivex::put_src(BSTR src) {
+ GURL document_url(GetDocumentUrl());
+ if (document_url.SchemeIsSecure()) {
+ GURL source_url(src);
+ if (!source_url.SchemeIsSecure()) {
+ Base::put_src(ScopedBstr(g_activex_mixed_content_error));
+ return E_ACCESSDENIED;
+ }
+ }
+ return Base::put_src(src);
+}
+
+HRESULT ChromeFrameActivex::IOleObject_SetClientSite(
+ IOleClientSite* client_site) {
+ HRESULT hr = Base::IOleObject_SetClientSite(client_site);
+ if (FAILED(hr) || !client_site) {
+ EventHandlers* handlers[] = {
+ &onmessage_,
+ &onloaderror_,
+ &onload_,
+ &onreadystatechanged_,
+ };
+
+ for (int i = 0; i < arraysize(handlers); ++i)
+ handlers[i]->clear();
+
+ // Drop privileged mode on uninitialization.
+ is_privileged_ = false;
+ } else {
+ ScopedComPtr<IHTMLDocument2> document;
+ GetContainingDocument(document.Receive());
+ if (document) {
+ ScopedBstr url;
+ if (SUCCEEDED(document->get_URL(url.Receive())))
+ WideToUTF8(url, url.Length(), &document_url_);
+ }
+
+ // Probe to see whether the host implements the privileged service.
+ ScopedComPtr<IChromeFramePrivileged> service;
+ HRESULT service_hr = DoQueryService(SID_ChromeFramePrivileged, client_site,
+ service.Receive());
+ if (SUCCEEDED(service_hr) && service) {
+ // Does the host want privileged mode?
+ boolean wants_privileged = false;
+ service_hr = service->GetWantsPrivileged(&wants_privileged);
+
+ if (SUCCEEDED(service_hr) && wants_privileged)
+ is_privileged_ = true;
+ }
+
+ std::wstring chrome_extra_arguments;
+ std::wstring profile_name(GetHostProcessName(false));
+ if (is_privileged_) {
+ // Does the host want to provide extra arguments?
+ ScopedBstr extra_arguments_arg;
+ service_hr = service->GetChromeExtraArguments(
+ extra_arguments_arg.Receive());
+ if (S_OK == service_hr && extra_arguments_arg)
+ chrome_extra_arguments.assign(extra_arguments_arg,
+ extra_arguments_arg.Length());
+
+ ScopedBstr profile_name_arg;
+ service_hr = service->GetChromeProfileName(profile_name_arg.Receive());
+ if (S_OK == service_hr && profile_name_arg)
+ profile_name.assign(profile_name_arg, profile_name_arg.Length());
+ }
+
+ if (!InitializeAutomation(profile_name, chrome_extra_arguments,
+ IsIEInPrivate())) {
+ return E_FAIL;
+ }
+ }
+
+ return hr;
+}
+
+HRESULT ChromeFrameActivex::GetObjectScriptId(IHTMLObjectElement* object_elem,
+ BSTR* id) {
+ DCHECK(object_elem != NULL);
+ DCHECK(id != NULL);
+
+ HRESULT hr = E_FAIL;
+ if (object_elem) {
+ ScopedComPtr<IHTMLElement> elem;
+ hr = elem.QueryFrom(object_elem);
+ if (elem) {
+ hr = elem->get_id(id);
+ }
+ }
+
+ return hr;
+}
+
+HRESULT ChromeFrameActivex::GetObjectElement(IHTMLObjectElement** element) {
+ DCHECK(m_spClientSite);
+ if (!m_spClientSite)
+ return E_UNEXPECTED;
+
+ ScopedComPtr<IOleControlSite> site;
+ HRESULT hr = site.QueryFrom(m_spClientSite);
+ if (site) {
+ ScopedComPtr<IDispatch> disp;
+ hr = site->GetExtendedControl(disp.Receive());
+ if (disp) {
+ hr = disp.QueryInterface(element);
+ } else {
+ DCHECK(FAILED(hr));
+ }
+ }
+
+ return hr;
+}
+
+HRESULT ChromeFrameActivex::CreateScriptBlockForEvent(
+ IHTMLElement2* insert_after, BSTR instance_id, BSTR script,
+ BSTR event_name) {
+ DCHECK(insert_after);
+ DCHECK(::SysStringLen(event_name) > 0); // should always have this
+
+ // This might be 0 if not specified in the HTML document.
+ if (!::SysStringLen(instance_id)) {
+ // TODO(tommi): Should we give ourselves an ID if this happens?
+ NOTREACHED() << "Need to handle this";
+ return E_INVALIDARG;
+ }
+
+ ScopedComPtr<IHTMLDocument2> document;
+ HRESULT hr = GetContainingDocument(document.Receive());
+ if (SUCCEEDED(hr)) {
+ ScopedComPtr<IHTMLElement> element, new_element;
+ document->createElement(StackBstr(L"script"), element.Receive());
+ if (element) {
+ ScopedComPtr<IHTMLScriptElement> script_element;
+ if (SUCCEEDED(hr = script_element.QueryFrom(element))) {
+ script_element->put_htmlFor(instance_id);
+ script_element->put_event(event_name);
+ script_element->put_text(script);
+
+ hr = insert_after->insertAdjacentElement(StackBstr(L"afterEnd"),
+ element,
+ new_element.Receive());
+ }
+ }
+ }
+
+ return hr;
+}
+
+HRESULT ChromeFrameActivex::CreateDomEvent(const std::string& event_type,
+ const std::string& data,
+ const std::string& origin,
+ IDispatch** event) {
+ DCHECK(event_type.length() > 0);
+ DCHECK(event != NULL);
+
+ CComObject<ComMessageEvent>* ev = NULL;
+ HRESULT hr = CComObject<ComMessageEvent>::CreateInstance(&ev);
+ if (SUCCEEDED(hr)) {
+ ev->AddRef();
+
+ ScopedComPtr<IOleContainer> container;
+ m_spClientSite->GetContainer(container.Receive());
+ if (ev->Initialize(container, data, origin, event_type)) {
+ *event = ev;
+ } else {
+ NOTREACHED() << "event->Initialize";
+ ev->Release();
+ hr = E_UNEXPECTED;
+ }
+ }
+
+ return hr;
+}
+
+void ChromeFrameActivex::FireEvent(const EventHandlers& handlers,
+ const std::string& arg) {
+ if (handlers.size()) {
+ ScopedComPtr<IDispatch> event;
+ if (SUCCEEDED(CreateDomEvent("event", arg, "", event.Receive()))) {
+ FireEvent(handlers, event);
+ }
+ }
+}
+
+void ChromeFrameActivex::FireEvent(const EventHandlers& handlers,
+ IDispatch* event) {
+ DCHECK(event != NULL);
+ VARIANT arg = { VT_DISPATCH };
+ arg.pdispVal = event;
+ DISPPARAMS params = { &arg, NULL, 1, 0 };
+ for (EventHandlers::const_iterator it = handlers.begin();
+ it != handlers.end();
+ ++it) {
+ HRESULT hr = (*it)->Invoke(DISPID_VALUE, IID_NULL, LOCALE_USER_DEFAULT,
+ DISPATCH_METHOD, &params, NULL, NULL, NULL);
+ // 0x80020101 == SCRIPT_E_REPORTED.
+ // When the script we're invoking has an error, we get this error back.
+ DLOG_IF(ERROR, FAILED(hr) && hr != 0x80020101)
+ << StringPrintf(L"Failed to invoke script: 0x%08X", hr);
+ }
+}
+
+void ChromeFrameActivex::FireEvent(const EventHandlers& handlers,
+ IDispatch* event, BSTR target) {
+ DCHECK(event != NULL);
+ // Arguments in reverse order to event handler function declaration,
+ // because that's what DISPPARAMS requires.
+ VARIANT args[2] = { { VT_BSTR }, { VT_DISPATCH }, };
+ args[0].bstrVal = target;
+ args[1].pdispVal = event;
+ DISPPARAMS params = { args, NULL, arraysize(args), 0 };
+ for (EventHandlers::const_iterator it = handlers.begin();
+ it != handlers.end();
+ ++it) {
+ HRESULT hr = (*it)->Invoke(DISPID_VALUE, IID_NULL, LOCALE_USER_DEFAULT,
+ DISPATCH_METHOD, &params, NULL, NULL, NULL);
+ // 0x80020101 == SCRIPT_E_REPORTED.
+ // When the script we're invoking has an error, we get this error back.
+ DLOG_IF(ERROR, FAILED(hr) && hr != 0x80020101)
+ << StringPrintf(L"Failed to invoke script: 0x%08X", hr);
+ }
+}
diff --git a/chrome_frame/chrome_frame_activex.h b/chrome_frame/chrome_frame_activex.h
new file mode 100644
index 0000000..07b8122
--- /dev/null
+++ b/chrome_frame/chrome_frame_activex.h
@@ -0,0 +1,140 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_FRAME_ACTIVEX_H_
+#define CHROME_FRAME_CHROME_FRAME_ACTIVEX_H_
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <atlctl.h>
+
+#include <set>
+#include <string>
+
+#include "base/scoped_bstr_win.h"
+#include "base/scoped_comptr_win.h"
+#include "base/scoped_variant_win.h"
+#include "chrome_frame/chrome_frame_activex_base.h"
+#include "chrome_frame/com_type_info_holder.h"
+#include "grit/chrome_frame_resources.h"
+
+// Include without path to make GYP build see it.
+#include "chrome_tab.h" // NOLINT
+
+// ChromeFrameActivex: Implementation of the ActiveX control that is
+// responsible for hosting a chrome frame, i.e. an iframe like widget which
+// hosts the the chrome window. This object delegates to Chrome.exe
+// (via the Chrome IPC-based automation mechanism) for the actual rendering.
+class ATL_NO_VTABLE ChromeFrameActivex
+ : public ChromeFrameActivexBase<ChromeFrameActivex, CLSID_ChromeFrame>,
+ public IObjectSafetyImpl<ChromeFrameActivex,
+ INTERFACESAFE_FOR_UNTRUSTED_CALLER |
+ INTERFACESAFE_FOR_UNTRUSTED_DATA>,
+ public IPersistPropertyBag {
+ public:
+ typedef ChromeFrameActivexBase<ChromeFrameActivex, CLSID_ChromeFrame> Base;
+ ChromeFrameActivex();
+ ~ChromeFrameActivex();
+
+DECLARE_REGISTRY_RESOURCEID(IDR_CHROMEFRAME)
+
+BEGIN_COM_MAP(ChromeFrameActivex)
+ COM_INTERFACE_ENTRY(IObjectSafety)
+ COM_INTERFACE_ENTRY(IPersistPropertyBag)
+ COM_INTERFACE_ENTRY(IConnectionPointContainer)
+ COM_INTERFACE_ENTRY_CHAIN(Base)
+END_COM_MAP()
+
+BEGIN_MSG_MAP(ChromeFrameActivex)
+ MESSAGE_HANDLER(WM_CREATE, OnCreate)
+ CHAIN_MSG_MAP(Base)
+END_MSG_MAP()
+
+ HRESULT FinalConstruct();
+
+ virtual HRESULT OnDraw(ATL_DRAWINFO& draw_info);
+
+ // IPersistPropertyBag implementation
+ STDMETHOD(GetClassID)(CLSID* class_id) {
+ if (class_id != NULL)
+ *class_id = GetObjectCLSID();
+ return S_OK;
+ }
+
+ STDMETHOD(InitNew)() {
+ return S_OK;
+ }
+
+ STDMETHOD(Load)(IPropertyBag* bag, IErrorLog* error_log);
+
+ STDMETHOD(Save)(IPropertyBag* bag, BOOL clear_dirty, BOOL save_all) {
+ return E_NOTIMPL;
+ }
+
+ // Used to setup the document_url_ member needed for completing navigation.
+ // Create external tab (possibly in incognito mode).
+ HRESULT IOleObject_SetClientSite(IOleClientSite *pClientSite);
+
+ // Overridden to perform security checks.
+ STDMETHOD(put_src)(BSTR src);
+
+ protected:
+ virtual void OnAcceleratorPressed(int tab_handle, const MSG& accel_message);
+ virtual void OnLoad(int tab_handle, const GURL& url);
+ virtual void OnMessageFromChromeFrame(int tab_handle,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& target);
+
+ private:
+
+ LRESULT OnCreate(UINT message, WPARAM wparam, LPARAM lparam,
+ BOOL& handled); // NO_LINT
+
+ // ChromeFrameDelegate overrides
+ virtual void ChromeFrameActivex::OnAutomationServerLaunchFailed(
+ AutomationLaunchResult reason, const std::string& server_version);
+ virtual void OnLoadFailed(int error_code, const std::string& url);
+
+ // Helper function to execute a function on a script IDispatch interface.
+ HRESULT InvokeScriptFunction(const VARIANT& script, const std::string& param);
+ HRESULT InvokeScriptFunction(const VARIANT& script, VARIANT* param);
+ HRESULT GetContainingDocument(IHTMLDocument2** doc);
+ HRESULT GetDocumentWindow(IHTMLWindow2** window);
+
+ // Gets the value of the 'id' attribute of the object element.
+ HRESULT GetObjectScriptId(IHTMLObjectElement* object_elem, BSTR* id);
+
+ // Returns the object element in the HTML page.
+ // Note that if we're not being hosted inside an HTML
+ // document, then this call will fail.
+ HRESULT GetObjectElement(IHTMLObjectElement** element);
+
+ HRESULT CreateScriptBlockForEvent(IHTMLElement2* insert_after,
+ BSTR instance_id, BSTR script,
+ BSTR event_name);
+
+ // Creates a new event object that supports the |data| property.
+ // Note: you should supply an empty string for |origin| unless you're
+ // creating a "message" event.
+ HRESULT CreateDomEvent(const std::string& event_type, const std::string& data,
+ const std::string& origin, IDispatch** event);
+
+ // Utility function that checks the size of the vector and if > 0 creates
+ // a variant for the string argument and forwards the call to the other
+ // FireEvent method.
+ void FireEvent(const EventHandlers& handlers, const std::string& arg);
+
+ // Invokes all registered handlers in a vector of event handlers.
+ void FireEvent(const EventHandlers& handlers, IDispatch* event);
+
+ // This variant is used for the privatemessage handler only.
+ void FireEvent(const EventHandlers& handlers, IDispatch* event,
+ BSTR target);
+
+};
+
+OBJECT_ENTRY_AUTO(__uuidof(ChromeFrame), ChromeFrameActivex)
+
+#endif // CHROME_FRAME_CHROME_FRAME_ACTIVEX_H_
diff --git a/chrome_frame/chrome_frame_activex.rgs b/chrome_frame/chrome_frame_activex.rgs
new file mode 100644
index 0000000..27edbcf
--- /dev/null
+++ b/chrome_frame/chrome_frame_activex.rgs
@@ -0,0 +1,79 @@
+HKLM {
+ NoRemove Software {
+ NoRemove Classes {
+ ChromeTab.ChromeFrame.1 = s 'Chrome Frame' {
+ CLSID = s '{E0A900DF-9611-4446-86BD-4B1D47E7DB2A}'
+ }
+ ChromeTab.ChromeFrame = s 'Chrome Frame' {
+ CLSID = s '{E0A900DF-9611-4446-86BD-4B1D47E7DB2A}'
+ CurVer = s 'ChromeTab.ChromeFrame.1'
+ }
+ NoRemove CLSID {
+ ForceRemove {E0A900DF-9611-4446-86BD-4B1D47E7DB2A} = s 'Chrome Frame' {
+ ProgID = s 'ChromeTab.ChromeFrame.1'
+ VersionIndependentProgID = s 'ChromeTab.ChromeFrame'
+ ForceRemove 'Programmable'
+ InprocServer32 = s '%MODULE%' {
+ val ThreadingModel = s 'Apartment'
+ }
+ val AppID = s '%APPID%'
+ ForceRemove 'Control'
+ ForceRemove 'Programmable'
+ ForceRemove 'Insertable'
+ ForceRemove 'ToolboxBitmap32' = s '%MODULE%, 1'
+ 'MiscStatus' = s '0'
+ {
+ '1' = s '%OLEMISC%'
+ }
+ 'TypeLib' = s '{6F2664E1-FF6E-488A-BCD1-F4CA6001DFCC}'
+ 'Version' = s '1.0'
+ }
+ }
+ }
+
+ NoRemove Microsoft {
+ NoRemove Windows {
+ NoRemove CurrentVersion {
+ NoRemove Ext {
+ NoRemove PreApproved {
+ ForceRemove '{E0A900DF-9611-4446-86BD-4B1D47E7DB2A}' = s '' {
+ }
+ }
+ NoRemove Stats {
+ ForceRemove {E0A900DF-9611-4446-86BD-4B1D47E7DB2A} {
+ ForceRemove 'iexplore' {
+ val Type = d '1'
+ val Flags = d '4'
+ val Count = d '0'
+ val Time = b '%SYSTIME%'
+ ForceRemove AllowedDomains {
+ ForceRemove '*' {
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+ NoRemove 'Internet Explorer' {
+ NoRemove 'Low Rights' {
+ NoRemove ElevationPolicy {
+ ForceRemove '{E0A900DF-9611-4446-86BD-4B1D47E7DB2A}' = s '' {
+ val Policy = d '3'
+ val AppName = s '%CHROME_LAUNCHER_APPNAME%'
+ val AppPath = s '%CHROME_LAUNCHER_APPPATH%'
+ }
+ }
+ NoRemove DragDrop {
+ ForceRemove '{E0A900DF-9611-4446-86BD-4B1D47E7DB2A}' = s '' {
+ val Policy = d '3'
+ val AppName = s '%CHROME_APPNAME%'
+ val AppPath = s '%CHROME_APPPATH%'
+ }
+ }
+ }
+ }
+ }
+ }
+} \ No newline at end of file
diff --git a/chrome_frame/chrome_frame_activex_base.h b/chrome_frame/chrome_frame_activex_base.h
new file mode 100644
index 0000000..7a1993a
--- /dev/null
+++ b/chrome_frame/chrome_frame_activex_base.h
@@ -0,0 +1,848 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_FRAME_ACTIVEX_BASE_H_
+#define CHROME_FRAME_CHROME_FRAME_ACTIVEX_BASE_H_
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <atlctl.h>
+
+// Copied min/max defs from windows headers to appease atlimage.h.
+// TODO(slightlyoff): Figure out of more recent platform SDK's (> 6.1)
+// undo the janky "#define NOMINMAX" train wreck. See:
+// http://connect.microsoft.com/VisualStudio/feedback/ViewFeedback.aspx?FeedbackID=100703
+#ifndef max
+#define max(a,b) (((a) > (b)) ? (a) : (b))
+#endif
+#ifndef min
+#define min(a,b) (((a) < (b)) ? (a) : (b))
+#endif
+#include <atlimage.h>
+#undef max
+#undef min
+
+#include <shdeprecated.h> // for IBrowserService2
+#include <shlguid.h>
+
+#include <set>
+#include <string>
+
+#include "base/histogram.h"
+#include "base/scoped_bstr_win.h"
+#include "base/scoped_comptr_win.h"
+#include "base/scoped_variant_win.h"
+#include "base/string_util.h"
+#include "grit/chrome_frame_resources.h"
+#include "grit/chrome_frame_strings.h"
+#include "chrome_frame/chrome_frame_plugin.h"
+#include "chrome_frame/com_type_info_holder.h"
+#include "chrome_frame/urlmon_url_request.h"
+
+// Include without path to make GYP build see it.
+#include "chrome_tab.h" // NOLINT
+
+// Connection point class to support firing IChromeFrameEvents (dispinterface).
+template<class T>
+class ATL_NO_VTABLE ProxyDIChromeFrameEvents
+ : public IConnectionPointImpl<T, &DIID_DIChromeFrameEvents> {
+ public:
+ void FireMethodWithParams(ChromeFrameEventDispId dispid,
+ const VARIANT* params, size_t num_params) {
+ T* me = static_cast<T*>(this);
+ int connections = m_vec.GetSize();
+
+ for (int connection = 0; connection < connections; ++connection) {
+ me->Lock();
+ CComPtr<IUnknown> sink(m_vec.GetAt(connection));
+ me->Unlock();
+
+ DIChromeFrameEvents* events = static_cast<DIChromeFrameEvents*>(sink.p);
+ if (events) {
+ DISPPARAMS disp_params = {
+ const_cast<VARIANT*>(params),
+ NULL,
+ num_params,
+ 0};
+ HRESULT hr = events->Invoke(static_cast<DISPID>(dispid),
+ DIID_DIChromeFrameEvents,
+ LOCALE_USER_DEFAULT, DISPATCH_METHOD,
+ &disp_params, NULL, NULL, NULL);
+ DLOG_IF(ERROR, FAILED(hr)) << "invoke(" << dispid << ") failed" <<
+ StringPrintf("0x%08X", hr);
+ }
+ }
+ }
+
+ void FireMethodWithParam(ChromeFrameEventDispId dispid,
+ const VARIANT& param) {
+ FireMethodWithParams(dispid, &param, 1);
+ }
+
+ void Fire_onload(IDispatch* event) {
+ VARIANT var = { VT_DISPATCH };
+ var.pdispVal = event;
+ FireMethodWithParam(CF_EVENT_DISPID_ONLOAD, var);
+ }
+
+ void Fire_onloaderror(IDispatch* event) {
+ VARIANT var = { VT_DISPATCH };
+ var.pdispVal = event;
+ FireMethodWithParam(CF_EVENT_DISPID_ONLOADERROR, var);
+ }
+
+ void Fire_onmessage(IDispatch* event) {
+ VARIANT var = { VT_DISPATCH };
+ var.pdispVal = event;
+ FireMethodWithParam(CF_EVENT_DISPID_ONMESSAGE, var);
+ }
+
+ void Fire_onreadystatechanged(long readystate) {
+ VARIANT var = { VT_I4 };
+ var.lVal = readystate;
+ FireMethodWithParam(CF_EVENT_DISPID_ONREADYSTATECHANGED, var);
+ }
+
+ void Fire_onprivatemessage(IDispatch* event, BSTR target) {
+ // Arguments in reverse order to the function declaration, because
+ // that's what DISPPARAMS requires.
+ VARIANT args[2] = { { VT_BSTR, }, {VT_DISPATCH, } };
+ args[0].bstrVal = target;
+ args[1].pdispVal = event;
+
+ FireMethodWithParams(CF_EVENT_DISPID_ONPRIVATEMESSAGE,
+ args,
+ arraysize(args));
+ }
+};
+
+extern bool g_first_launch_by_process_;
+
+// Common implementation for ActiveX and Active Document
+template <class T, const CLSID& class_id>
+class ATL_NO_VTABLE ChromeFrameActivexBase :
+ public CComObjectRootEx<CComSingleThreadModel>,
+ public IOleControlImpl<T>,
+ public IOleObjectImpl<T>,
+ public IOleInPlaceActiveObjectImpl<T>,
+ public IViewObjectExImpl<T>,
+ public IOleInPlaceObjectWindowlessImpl<T>,
+ public ISupportErrorInfo,
+ public IQuickActivateImpl<T>,
+ public com_util::IProvideClassInfo2Impl<class_id,
+ DIID_DIChromeFrameEvents>,
+ public com_util::IDispatchImpl<IChromeFrame>,
+ public IConnectionPointContainerImpl<T>,
+ public ProxyDIChromeFrameEvents<T>,
+ public IPropertyNotifySinkCP<T>,
+ public CComCoClass<T, &class_id>,
+ public CComControl<T>,
+ public ChromeFramePlugin<T> {
+ protected:
+ typedef std::set<ScopedComPtr<IDispatch> > EventHandlers;
+
+ public:
+ ChromeFrameActivexBase()
+ : ready_state_(READYSTATE_UNINITIALIZED) {
+ m_bWindowOnly = TRUE;
+ }
+
+ ~ChromeFrameActivexBase() {
+ }
+
+DECLARE_OLEMISC_STATUS(OLEMISC_RECOMPOSEONRESIZE | OLEMISC_CANTLINKINSIDE |
+ OLEMISC_INSIDEOUT | OLEMISC_ACTIVATEWHENVISIBLE |
+ OLEMISC_SETCLIENTSITEFIRST)
+
+DECLARE_NOT_AGGREGATABLE(T)
+
+BEGIN_COM_MAP(ChromeFrameActivexBase)
+ COM_INTERFACE_ENTRY(IChromeFrame)
+ COM_INTERFACE_ENTRY(IDispatch)
+ COM_INTERFACE_ENTRY(IViewObjectEx)
+ COM_INTERFACE_ENTRY(IViewObject2)
+ COM_INTERFACE_ENTRY(IViewObject)
+ COM_INTERFACE_ENTRY(IOleInPlaceObjectWindowless)
+ COM_INTERFACE_ENTRY(IOleInPlaceObject)
+ COM_INTERFACE_ENTRY2(IOleWindow, IOleInPlaceObjectWindowless)
+ COM_INTERFACE_ENTRY(IOleInPlaceActiveObject)
+ COM_INTERFACE_ENTRY(IOleControl)
+ COM_INTERFACE_ENTRY(IOleObject)
+ COM_INTERFACE_ENTRY(ISupportErrorInfo)
+ COM_INTERFACE_ENTRY(IQuickActivate)
+ COM_INTERFACE_ENTRY(IProvideClassInfo)
+ COM_INTERFACE_ENTRY(IProvideClassInfo2)
+ COM_INTERFACE_ENTRY_FUNC_BLIND(0, InterfaceNotSupported)
+END_COM_MAP()
+
+BEGIN_CONNECTION_POINT_MAP(T)
+ CONNECTION_POINT_ENTRY(IID_IPropertyNotifySink)
+ CONNECTION_POINT_ENTRY(DIID_DIChromeFrameEvents)
+END_CONNECTION_POINT_MAP()
+
+BEGIN_MSG_MAP(ChromeFrameActivexBase)
+ MESSAGE_HANDLER(WM_CREATE, OnCreate)
+ CHAIN_MSG_MAP(ChromeFramePlugin<T>)
+ CHAIN_MSG_MAP(CComControl<T>)
+ DEFAULT_REFLECTION_HANDLER()
+END_MSG_MAP()
+
+ // IViewObjectEx
+ DECLARE_VIEW_STATUS(VIEWSTATUS_SOLIDBKGND | VIEWSTATUS_OPAQUE)
+
+ inline HRESULT IViewObject_Draw(DWORD draw_aspect, LONG index,
+ void* aspect_info, DVTARGETDEVICE* ptd, HDC info_dc, HDC dc,
+ LPCRECTL bounds, LPCRECTL win_bounds) {
+ // ATL ASSERTs if dwDrawAspect is DVASPECT_DOCPRINT, so we cheat.
+ DWORD aspect = draw_aspect;
+ if (aspect == DVASPECT_DOCPRINT)
+ aspect = DVASPECT_CONTENT;
+
+ return CComControl<T>::IViewObject_Draw(aspect, index, aspect_info, ptd,
+ info_dc, dc, bounds, win_bounds);
+ }
+
+ DECLARE_PROTECT_FINAL_CONSTRUCT()
+
+ HRESULT FinalConstruct() {
+ if (!Initialize())
+ return E_OUTOFMEMORY;
+
+ // Set to true if this is the first launch by this process.
+ // Used to perform one time tasks.
+ if (g_first_launch_by_process_) {
+ g_first_launch_by_process_ = false;
+ UMA_HISTOGRAM_CUSTOM_COUNTS("ChromeFrame.IEVersion",
+ GetIEVersion(),
+ IE_INVALID,
+ IE_8,
+ IE_8 + 1);
+ }
+ return S_OK;
+ }
+
+ void FinalRelease() {
+ }
+
+ static HRESULT WINAPI InterfaceNotSupported(void* pv, REFIID riid, void** ppv,
+ DWORD dw) {
+#ifndef NDEBUG
+ wchar_t buffer[64] = {0};
+ ::StringFromGUID2(riid, buffer, arraysize(buffer));
+ DLOG(INFO) << "E_NOINTERFACE: " << buffer;
+#endif
+ return E_NOINTERFACE;
+ }
+
+ // ISupportsErrorInfo
+ STDMETHOD(InterfaceSupportsErrorInfo)(REFIID riid) {
+ static const IID* interfaces[] = {
+ &IID_IChromeFrame,
+ &IID_IDispatch
+ };
+
+ for (int i = 0; i < arraysize(interfaces); ++i) {
+ if (InlineIsEqualGUID(*interfaces[i], riid))
+ return S_OK;
+ }
+ return S_FALSE;
+ }
+
+ // Called to draw our control when chrome hasn't been initialized.
+ virtual HRESULT OnDraw(ATL_DRAWINFO& draw_info) { // NO_LINT
+ if (NULL == draw_info.prcBounds) {
+ NOTREACHED();
+ return E_FAIL;
+ }
+ // Don't draw anything.
+ return S_OK;
+ }
+
+
+ // Used to setup the document_url_ member needed for completing navigation.
+ // Create external tab (possibly in incognito mode).
+ HRESULT IOleObject_SetClientSite(IOleClientSite* client_site) {
+ // If we currently have a document site pointer, release it.
+ doc_site_.Release();
+ if (client_site) {
+ doc_site_.QueryFrom(client_site);
+ }
+
+ return CComControlBase::IOleObject_SetClientSite(client_site);
+ }
+
+ bool HandleContextMenuCommand(UINT cmd) {
+ if (cmd == IDC_ABOUT_CHROME_FRAME) {
+ int tab_handle = automation_client_->tab()->handle();
+ OnOpenURL(tab_handle, GURL("about:version"), NEW_WINDOW);
+ return true;
+ }
+
+ return false;
+ }
+
+ // Should connections initiated by this class try to block
+ // responses served with the X-Frame-Options header?
+ // ActiveX controls genereally will want to do this,
+ // returning true, while true top-level documents
+ // (ActiveDocument servers) will not. Your specialization
+ // of this template should implement this method based on how
+ // it "feels" from a security perspective. If it's hosted in another
+ // scriptable document, return true, else false.
+ virtual bool is_frame_busting_enabled() const {
+ return true;
+ }
+
+ protected:
+ virtual void OnTabbedOut(int tab_handle, bool reverse) {
+ DCHECK(m_bInPlaceActive);
+
+ HWND parent = ::GetParent(m_hWnd);
+ ::SetFocus(parent);
+ ScopedComPtr<IOleControlSite> control_site;
+ control_site.QueryFrom(m_spClientSite);
+ if (control_site)
+ control_site->OnFocus(FALSE);
+ }
+
+ virtual void OnOpenURL(int tab_handle, const GURL& url_to_open,
+ int open_disposition) {
+ ScopedComPtr<IWebBrowser2> web_browser2;
+ DoQueryService(SID_SWebBrowserApp, m_spClientSite, web_browser2.Receive());
+ DCHECK(web_browser2);
+
+ ScopedVariant url;
+ // Check to see if the URL uses a "view-source:" prefix, if so, open it
+ // using chrome frame full tab mode by using 'cf:' protocol handler.
+ // Also change the disposition to NEW_WINDOW since IE6 doesn't have tabs.
+ if (url_to_open.has_scheme() && (url_to_open.SchemeIs("view-source") ||
+ url_to_open.SchemeIs("about"))) {
+ std::string chrome_url;
+ chrome_url.append("cf:");
+ chrome_url.append(url_to_open.spec());
+ url.Set(UTF8ToWide(chrome_url).c_str());
+ open_disposition = NEW_WINDOW;
+ } else {
+ url.Set(UTF8ToWide(url_to_open.spec()).c_str());
+ }
+
+ VARIANT flags = { VT_I4 };
+ V_I4(&flags) = 0;
+
+ IEVersion ie_version = GetIEVersion();
+ DCHECK(ie_version != NON_IE && ie_version != IE_UNSUPPORTED);
+ // Since IE6 doesn't support tabs, so we just use window instead.
+ if (ie_version == IE_6) {
+ if (open_disposition == NEW_FOREGROUND_TAB ||
+ open_disposition == NEW_BACKGROUND_TAB ||
+ open_disposition == NEW_WINDOW) {
+ V_I4(&flags) = navOpenInNewWindow;
+ } else if (open_disposition != CURRENT_TAB) {
+ NOTREACHED() << "Unsupported open disposition in IE6";
+ }
+ } else {
+ switch (open_disposition) {
+ case NEW_FOREGROUND_TAB:
+ V_I4(&flags) = navOpenInNewTab;
+ break;
+ case NEW_BACKGROUND_TAB:
+ V_I4(&flags) = navOpenInBackgroundTab;
+ break;
+ case NEW_WINDOW:
+ V_I4(&flags) = navOpenInNewWindow;
+ break;
+ default:
+ break;
+ }
+ }
+
+ // TODO(sanjeevr): The navOpenInNewWindow flag causes IE to open this
+ // in a new window ONLY if the user has specified
+ // "Always open popups in a new window". Otherwise it opens in a new tab.
+ // We need to investigate more and see if we can force IE to display the
+ // link in a new window. MSHTML uses the below code to force an open in a
+ // new window. But this logic also fails for us. Perhaps this flag is not
+ // honoured if the ActiveDoc is not MSHTML.
+ // Even the HLNF_DISABLEWINDOWRESTRICTIONS flag did not work.
+ // Start of MSHTML-like logic.
+ // CComQIPtr<ITargetFramePriv2> target_frame = web_browser2;
+ // if (target_frame) {
+ // CComPtr<IUri> uri;
+ // CreateUri(UTF8ToWide(open_url_command->url_.spec()).c_str(),
+ // Uri_CREATE_IE_SETTINGS, 0, &uri);
+ // CComPtr<IBindCtx> bind_ctx;
+ // CreateBindCtx(0, &bind_ctx);
+ // target_frame->AggregatedNavigation2(
+ // HLNF_TRUSTFIRSTDOWNLOAD|HLNF_OPENINNEWWINDOW, bind_ctx, NULL,
+ // L"No_Name", uri, L"");
+ // }
+ // End of MSHTML-like logic
+ VARIANT empty = ScopedVariant::kEmptyVariant;
+ web_browser2->Navigate2(url.AsInput(), &flags, &empty, &empty, &empty);
+ web_browser2->put_Visible(VARIANT_TRUE);
+ }
+
+ virtual void OnRequestStart(int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request_info) {
+ scoped_refptr<CComObject<UrlmonUrlRequest> > request;
+ if (base_url_request_.get() &&
+ GURL(base_url_request_->url()) == GURL(request_info.url)) {
+ request.swap(base_url_request_);
+ } else {
+ CComObject<UrlmonUrlRequest>* new_request = NULL;
+ CComObject<UrlmonUrlRequest>::CreateInstance(&new_request);
+ request = new_request;
+ }
+
+ DCHECK(request.get() != NULL);
+
+ if (request->Initialize(automation_client_.get(), tab_handle, request_id,
+ request_info.url, request_info.method,
+ request_info.referrer,
+ request_info.extra_request_headers,
+ request_info.upload_data.get(),
+ static_cast<T*>(this)->is_frame_busting_enabled())) {
+ // If Start is successful, it will add a self reference.
+ request->Start();
+ request->set_parent_window(m_hWnd);
+ }
+ }
+
+ virtual void OnRequestRead(int tab_handle, int request_id,
+ int bytes_to_read) {
+ automation_client_->ReadRequest(request_id, bytes_to_read);
+ }
+
+ virtual void OnRequestEnd(int tab_handle, int request_id,
+ const URLRequestStatus& status) {
+ automation_client_->RemoveRequest(request_id, status.status(), true);
+ }
+
+ virtual void OnSetCookieAsync(int tab_handle, const GURL& url,
+ const std::string& cookie) {
+ std::string name;
+ std::string data;
+ size_t name_end = cookie.find('=');
+ if (std::string::npos != name_end) {
+ name = cookie.substr(0, name_end);
+ data = cookie.substr(name_end + 1);
+ } else {
+ data = cookie;
+ }
+
+ BOOL ret = InternetSetCookieA(url.spec().c_str(), name.c_str(),
+ data.c_str());
+ DCHECK(ret) << "InternetSetcookie failed. Error: " << GetLastError();
+ }
+
+ virtual void OnAttachExternalTab(int tab_handle,
+ intptr_t cookie,
+ int disposition) {
+ std::string url;
+ url = StringPrintf("cf:attach_external_tab&%d&%d",
+ cookie, disposition);
+ OnOpenURL(tab_handle, GURL(url), disposition);
+ }
+
+ LRESULT OnCreate(UINT message, WPARAM wparam, LPARAM lparam,
+ BOOL& handled) { // NO_LINT
+ ModifyStyle(0, WS_CLIPCHILDREN | WS_CLIPSIBLINGS, 0);
+ automation_client_->SetParentWindow(m_hWnd);
+ // Only fire the 'interactive' ready state if we aren't there already.
+ if (ready_state_ < READYSTATE_INTERACTIVE) {
+ ready_state_ = READYSTATE_INTERACTIVE;
+ FireOnChanged(DISPID_READYSTATE);
+ }
+ return 0;
+ }
+
+ // ChromeFrameDelegate override
+ virtual void OnAutomationServerReady() {
+ ChromeFramePlugin<T>::OnAutomationServerReady();
+
+ ready_state_ = READYSTATE_COMPLETE;
+ FireOnChanged(DISPID_READYSTATE);
+ }
+
+ // ChromeFrameDelegate override
+ virtual void OnAutomationServerLaunchFailed(
+ AutomationLaunchResult reason, const std::string& server_version) {
+ ready_state_ = READYSTATE_UNINITIALIZED;
+ FireOnChanged(DISPID_READYSTATE);
+ }
+
+ // Overridden to take advantage of readystate prop changes and send those
+ // to potential listeners.
+ HRESULT FireOnChanged(DISPID dispid) {
+ if (dispid == DISPID_READYSTATE) {
+ Fire_onreadystatechanged(ready_state_);
+ }
+ return __super::FireOnChanged(dispid);
+ }
+
+ // IChromeFrame
+ // Property getter/setters for the src attribute, which contains a URL.
+ // The ChromeFrameActivex control initiates navigation to this URL
+ // when instantiated.
+ STDMETHOD(get_src)(BSTR* src) {
+ if (NULL == src) {
+ return E_POINTER;
+ }
+
+ *src = SysAllocString(url_);
+ return S_OK;
+ }
+
+ STDMETHOD(put_src)(BSTR src) {
+ if (src == NULL)
+ return E_INVALIDARG;
+
+ // Switch the src to UTF8 and try to expand to full URL
+ std::string src_utf8;
+ WideToUTF8(src, SysStringLen(src), &src_utf8);
+ std::string full_url = ResolveURL(GetDocumentUrl(), src_utf8);
+ if (full_url.empty()) {
+ return E_INVALIDARG;
+ }
+
+ // We can initiate navigation here even if ready_state is not complete.
+ // We do not have to set proxy, and AutomationClient will take care
+ // of navigation just after CreateExternalTab is done.
+ if (!automation_client_->InitiateNavigation(full_url)) {
+ // TODO(robertshield): Make InitiateNavigation return more useful
+ // error information.
+ return E_INVALIDARG;
+ }
+
+ // Save full URL in BSTR member
+ url_.Reset(::SysAllocString(UTF8ToWide(full_url).c_str()));
+
+ return S_OK;
+ }
+
+ STDMETHOD(get_onload)(VARIANT* onload_handler) {
+ if (NULL == onload_handler)
+ return E_INVALIDARG;
+
+ *onload_handler = onload_handler_.Copy();
+
+ return S_OK;
+ }
+
+ // Property setter for the onload attribute, which contains a
+ // javascript function to be invoked on successful navigation.
+ STDMETHOD(put_onload)(VARIANT onload_handler) {
+ if (V_VT(&onload_handler) != VT_DISPATCH) {
+ DLOG(WARNING) << "Invalid onload handler type: "
+ << onload_handler.vt
+ << " specified";
+ return E_INVALIDARG;
+ }
+
+ onload_handler_ = onload_handler;
+
+ return S_OK;
+ }
+
+ // Property getter/setters for the onloaderror attribute, which contains a
+ // javascript function to be invoked on navigation failure.
+ STDMETHOD(get_onloaderror)(VARIANT* onerror_handler) {
+ if (NULL == onerror_handler)
+ return E_INVALIDARG;
+
+ *onerror_handler = onerror_handler_.Copy();
+
+ return S_OK;
+ }
+
+ STDMETHOD(put_onloaderror)(VARIANT onerror_handler) {
+ if (V_VT(&onerror_handler) != VT_DISPATCH) {
+ DLOG(WARNING) << "Invalid onloaderror handler type: "
+ << onerror_handler.vt
+ << " specified";
+ return E_INVALIDARG;
+ }
+
+ onerror_handler_ = onerror_handler;
+
+ return S_OK;
+ }
+
+ // Property getter/setters for the onmessage attribute, which contains a
+ // javascript function to be invoked when we receive a message from the
+ // chrome frame.
+ STDMETHOD(put_onmessage)(VARIANT onmessage_handler) {
+ if (V_VT(&onmessage_handler) != VT_DISPATCH) {
+ DLOG(WARNING) << "Invalid onmessage handler type: "
+ << onmessage_handler.vt
+ << " specified";
+ return E_INVALIDARG;
+ }
+
+ onmessage_handler_ = onmessage_handler;
+
+ return S_OK;
+ }
+
+ STDMETHOD(get_onmessage)(VARIANT* onmessage_handler) {
+ if (NULL == onmessage_handler)
+ return E_INVALIDARG;
+
+ *onmessage_handler = onmessage_handler_.Copy();
+
+ return S_OK;
+ }
+
+ STDMETHOD(get_readyState)(long* ready_state) { // NOLINT
+ DLOG(INFO) << __FUNCTION__;
+ DCHECK(ready_state);
+
+ if (!ready_state)
+ return E_INVALIDARG;
+
+ *ready_state = ready_state_;
+
+ return S_OK;
+ }
+
+ // Property getter/setters for use_chrome_network flag. This flag
+ // indicates if chrome network stack is to be used for fetching
+ // network requests.
+ STDMETHOD(get_useChromeNetwork)(VARIANT_BOOL* use_chrome_network) {
+ if (!use_chrome_network)
+ return E_INVALIDARG;
+
+ *use_chrome_network =
+ automation_client_->use_chrome_network() ? VARIANT_TRUE : VARIANT_FALSE;
+ return S_OK;
+ }
+
+ STDMETHOD(put_useChromeNetwork)(VARIANT_BOOL use_chrome_network) {
+ if (!is_privileged_) {
+ DLOG(ERROR) << "Attempt to set useChromeNetwork in non-privileged mode";
+ return E_ACCESSDENIED;
+ }
+
+ automation_client_->set_use_chrome_network(
+ (VARIANT_FALSE != use_chrome_network));
+ return S_OK;
+ }
+
+ // Posts a message to the chrome frame.
+ STDMETHOD(postMessage)(BSTR message, VARIANT target) {
+ if (NULL == message) {
+ return E_INVALIDARG;
+ }
+
+ if (!automation_client_.get())
+ return E_FAIL;
+
+ std::string utf8_target;
+ if (target.vt == VT_BSTR) {
+ int len = ::SysStringLen(target.bstrVal);
+ if (len == 1 && target.bstrVal[0] == L'*') {
+ utf8_target = "*";
+ } else {
+ GURL resolved(target.bstrVal);
+ if (!resolved.is_valid()) {
+ Error(L"Unable to parse the specified target URL.");
+ return E_INVALIDARG;
+ }
+
+ utf8_target = resolved.spec();
+ }
+ } else {
+ utf8_target = "*";
+ }
+
+ std::string utf8_message;
+ WideToUTF8(message, ::SysStringLen(message), &utf8_message);
+
+ GURL url(GURL(document_url_).GetOrigin());
+ std::string origin(url.is_empty() ? "null" : url.spec());
+ if (!automation_client_->ForwardMessageFromExternalHost(utf8_message,
+ origin,
+ utf8_target)) {
+ Error(L"Failed to post message to chrome frame");
+ return E_FAIL;
+ }
+
+ return S_OK;
+ }
+
+ STDMETHOD(addEventListener)(BSTR event_type, IDispatch* listener,
+ VARIANT use_capture) {
+ EventHandlers* handlers = NULL;
+ HRESULT hr = GetHandlersForEvent(event_type, &handlers);
+ if (FAILED(hr))
+ return hr;
+
+ DCHECK(handlers != NULL);
+
+ handlers->insert(ScopedComPtr<IDispatch>(listener));
+
+ return hr;
+ }
+
+ STDMETHOD(removeEventListener)(BSTR event_type, IDispatch* listener,
+ VARIANT use_capture) {
+ EventHandlers* handlers = NULL;
+ HRESULT hr = GetHandlersForEvent(event_type, &handlers);
+ if (FAILED(hr))
+ return hr;
+
+ DCHECK(handlers != NULL);
+ std::remove(handlers->begin(), handlers->end(), listener);
+
+ return hr;
+ }
+
+ STDMETHOD(get_version)(BSTR* version) {
+ if (!automation_client_.get()) {
+ NOTREACHED();
+ return E_FAIL;
+ }
+
+ if (version == NULL) {
+ return E_INVALIDARG;
+ }
+
+ *version = SysAllocString(automation_client_->GetVersion().c_str());
+ return S_OK;
+ }
+
+ STDMETHOD(postPrivateMessage)(BSTR message, BSTR origin, BSTR target) {
+ if (NULL == message)
+ return E_INVALIDARG;
+
+ if (!is_privileged_) {
+ DLOG(ERROR) << "Attempt to postPrivateMessage in non-privileged mode";
+ return E_ACCESSDENIED;
+ }
+
+ DCHECK(automation_client_.get());
+ std::string utf8_message, utf8_origin, utf8_target;
+ WideToUTF8(message, ::SysStringLen(message), &utf8_message);
+ WideToUTF8(origin, ::SysStringLen(origin), &utf8_origin);
+ WideToUTF8(target, ::SysStringLen(target), &utf8_target);
+
+ if (!automation_client_->ForwardMessageFromExternalHost(utf8_message,
+ utf8_origin,
+ utf8_target)) {
+ Error(L"Failed to post message to chrome frame");
+ return E_FAIL;
+ }
+
+ return S_OK;
+ }
+
+ // Returns the vector of event handlers for a given event (e.g. "load").
+ // If the event type isn't recognized, the function fills in a descriptive
+ // error (IErrorInfo) and returns E_INVALIDARG.
+ HRESULT GetHandlersForEvent(BSTR event_type, EventHandlers** handlers) {
+ DCHECK(handlers != NULL);
+
+ HRESULT hr = S_OK;
+ const wchar_t* event_type_end = event_type + ::SysStringLen(event_type);
+ if (LowerCaseEqualsASCII(event_type, event_type_end, "message")) {
+ *handlers = &onmessage_;
+ } else if (LowerCaseEqualsASCII(event_type, event_type_end, "load")) {
+ *handlers = &onload_;
+ } else if (LowerCaseEqualsASCII(event_type, event_type_end, "loaderror")) {
+ *handlers = &onloaderror_;
+ } else if (LowerCaseEqualsASCII(event_type, event_type_end,
+ "readystatechanged")) {
+ *handlers = &onreadystatechanged_;
+ } else if (LowerCaseEqualsASCII(event_type, event_type_end,
+ "privatemessage")) {
+ // This event handler is only available in privileged mode.
+ if (!is_privileged_) {
+ Error("Event type 'privatemessage' is privileged");
+ return E_ACCESSDENIED;
+ }
+ *handlers = &onprivatemessage_;
+ } else {
+ Error(StringPrintf("Event type '%ls' not found", event_type).c_str());
+ hr = E_INVALIDARG;
+ }
+
+ return hr;
+ }
+
+ // Gives the browser a chance to handle an accelerator that was
+ // sent to the out of proc chromium instance.
+ // Returns S_OK iff the accelerator was handled by the browser.
+ HRESULT AllowFrameToTranslateAccelerator(const MSG& msg) {
+ // Although IBrowserService2 is officially deprecated, it's still alive
+ // and well in IE7 and earlier. We have to use it here to correctly give
+ // the browser a chance to handle keyboard shortcuts.
+ // This happens automatically for activex components that have windows that
+ // belong to the current thread. In that circumstance IE owns the message
+ // loop and can walk the line of components allowing each participant the
+ // chance to handle the keystroke and eventually falls back to
+ // v_MayTranslateAccelerator. However in our case, the message loop is
+ // owned by the out-of-proc chromium instance so IE doesn't have a chance to
+ // fall back on its default behavior. Instead we give IE a chance to
+ // handle the shortcut here.
+
+ HRESULT hr = S_FALSE;
+ ScopedComPtr<IBrowserService2> bs2;
+ if (S_OK == DoQueryService(SID_STopLevelBrowser, m_spInPlaceSite,
+ bs2.Receive())) {
+ hr = bs2->v_MayTranslateAccelerator(const_cast<MSG*>(&msg));
+ } else {
+ // IE8 doesn't support IBrowserService2 unless you enable a special,
+ // undocumented flag with CoInternetSetFeatureEnabled and even then,
+ // the object you get back implements only a couple of methods of
+ // that interface... all the other entries in the vtable are NULL.
+ // In addition, the class that implements it is called
+ // ImpostorBrowserService2 :)
+ // IE8 does have a new interface though, presumably called
+ // ITabBrowserService or something that can be abbreviated to TBS.
+ // That interface has a method, TranslateAcceleratorTBS that does
+ // call the root MayTranslateAccelerator function, but alas the
+ // first argument to MayTranslateAccelerator is hard coded to FALSE
+ // which means that global accelerators are not handled and we're
+ // out of luck.
+ // A third thing that's notable with regards to IE8 is that
+ // none of the *MayTranslate* functions exist in a vtable inside
+ // ieframe.dll. I checked this by scanning for the address of
+ // those functions inside the dll and found none, which means that
+ // all calls to those functions are relative.
+ // So, for IE8, our approach is very simple. Just post the message
+ // to our parent window and IE will pick it up if it's an
+ // accelerator. We won't know for sure if the browser handled the
+ // keystroke or not.
+ ::PostMessage(::GetParent(m_hWnd), msg.message, msg.wParam,
+ msg.lParam);
+ }
+
+ return hr;
+ }
+
+ protected:
+ ScopedBstr url_;
+ ScopedComPtr<IOleDocumentSite> doc_site_;
+
+ // For more information on the ready_state_ property see:
+ // http://msdn.microsoft.com/en-us/library/aa768179(VS.85).aspx#
+ READYSTATE ready_state_;
+
+ // The following members contain IDispatch interfaces representing the
+ // onload/onerror/onmessage handlers on the page.
+ ScopedVariant onload_handler_;
+ ScopedVariant onerror_handler_;
+ ScopedVariant onmessage_handler_;
+
+ EventHandlers onmessage_;
+ EventHandlers onloaderror_;
+ EventHandlers onload_;
+ EventHandlers onreadystatechanged_;
+ EventHandlers onprivatemessage_;
+
+ // The UrlmonUrlRequest instance instantiated for downloading the base URL.
+ scoped_refptr<CComObject<UrlmonUrlRequest> > base_url_request_;
+};
+
+#endif // CHROME_FRAME_CHROME_FRAME_ACTIVEX_BASE_H_
diff --git a/chrome_frame/chrome_frame_automation.cc b/chrome_frame/chrome_frame_automation.cc
new file mode 100644
index 0000000..9da5e43
--- /dev/null
+++ b/chrome_frame/chrome_frame_automation.cc
@@ -0,0 +1,975 @@
+// Copyright (c) 2009 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 "chrome_frame/chrome_frame_automation.h"
+
+#include "base/command_line.h"
+#include "base/compiler_specific.h"
+#include "base/file_util.h"
+#include "base/file_version_info.h"
+#include "base/logging.h"
+#include "base/path_service.h"
+#include "base/process_util.h"
+#include "base/singleton.h"
+#include "base/string_util.h"
+#include "base/sys_info.h"
+#include "chrome/app/client_util.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/common/chrome_paths.h"
+#include "chrome/common/chrome_switches.h"
+#include "chrome/test/automation/tab_proxy.h"
+#include "chrome_frame/chrome_launcher.h"
+#include "chrome_frame/utils.h"
+#include "chrome_frame/sync_msg_reply_dispatcher.h"
+
+#ifdef NDEBUG
+int64 kAutomationServerReasonableLaunchDelay = 1000; // in milliseconds
+#else
+int64 kAutomationServerReasonableLaunchDelay = 1000 * 10;
+#endif
+
+int kDefaultSendUMADataInterval = 20000; // in milliseconds.
+
+static const wchar_t kUmaSendIntervalValue[] = L"UmaSendInterval";
+
+class TabProxyNotificationMessageFilter
+ : public IPC::ChannelProxy::MessageFilter {
+ public:
+ explicit TabProxyNotificationMessageFilter(AutomationHandleTracker* tracker)
+ : tracker_(tracker) {
+ }
+
+ virtual bool OnMessageReceived(const IPC::Message& message) {
+ if (message.is_reply())
+ return false;
+
+ int tab_handle = 0;
+ if (!ChromeFrameDelegateImpl::IsTabMessage(message, &tab_handle))
+ return false;
+
+ // Get AddRef-ed pointer to corresponding TabProxy object
+ TabProxy* tab = static_cast<TabProxy*>(tracker_->GetResource(tab_handle));
+ if (tab) {
+ tab->OnMessageReceived(message);
+ tab->Release();
+ }
+ return true;
+ }
+
+
+ private:
+ AutomationHandleTracker* tracker_;
+};
+
+class ChromeFrameAutomationProxyImpl::CFMsgDispatcher
+ : public SyncMessageReplyDispatcher {
+ public:
+ CFMsgDispatcher() : SyncMessageReplyDispatcher() {}
+ protected:
+ virtual bool HandleMessageType(const IPC::Message& msg,
+ const MessageSent& origin) {
+ switch (origin.type) {
+ case AutomationMsg_CreateExternalTab::ID:
+ case AutomationMsg_ConnectExternalTab::ID:
+ InvokeCallback<Tuple3<HWND, HWND, int> >(msg, origin);
+ break;
+ case AutomationMsg_NavigateInExternalTab::ID:
+ InvokeCallback<Tuple1<AutomationMsg_NavigationResponseValues> >(msg,
+ origin);
+ break;
+ default:
+ NOTREACHED();
+ }
+ return true;
+ }
+};
+
+ChromeFrameAutomationProxyImpl::ChromeFrameAutomationProxyImpl(
+ int launch_timeout)
+ : AutomationProxy(launch_timeout) {
+ sync_ = new CFMsgDispatcher();
+ // Order of filters is not important.
+ channel_->AddFilter(new TabProxyNotificationMessageFilter(tracker_.get()));
+ channel_->AddFilter(sync_.get());
+}
+
+ChromeFrameAutomationProxyImpl::~ChromeFrameAutomationProxyImpl() {
+}
+
+void ChromeFrameAutomationProxyImpl::SendAsAsync(IPC::SyncMessage* msg,
+ void* callback, void* key) {
+ sync_->Push(msg, callback, key);
+ channel_->ChannelProxy::Send(msg);
+}
+
+void ChromeFrameAutomationProxyImpl::CancelAsync(void* key) {
+ sync_->Cancel(key);
+}
+
+scoped_refptr<TabProxy> ChromeFrameAutomationProxyImpl::CreateTabProxy(
+ int handle) {
+ DCHECK(tracker_->GetResource(handle) == NULL);
+ return new TabProxy(this, tracker_.get(), handle);
+}
+
+struct LaunchTimeStats {
+#ifndef NDEBUG
+ LaunchTimeStats() {
+ launch_time_begin_ = base::Time::Now();
+ }
+
+ void Dump() {
+ base::TimeDelta launch_time = base::Time::Now() - launch_time_begin_;
+ HISTOGRAM_TIMES("ChromeFrame.AutomationServerLaunchTime", launch_time);
+ const int64 launch_milliseconds = launch_time.InMilliseconds();
+ if (launch_milliseconds > kAutomationServerReasonableLaunchDelay) {
+ LOG(WARNING) << "Automation server launch took longer than expected: " <<
+ launch_milliseconds << " ms.";
+ }
+ }
+
+ base::Time launch_time_begin_;
+#else
+ void Dump() {}
+#endif
+};
+
+
+ProxyFactory::ProxyCacheEntry::ProxyCacheEntry(const std::wstring& profile)
+ : proxy(NULL), profile_name(profile), ref_count(1),
+ launch_result(AutomationLaunchResult(-1)) {
+ thread.reset(new base::Thread(WideToASCII(profile_name).c_str()));
+ thread->Start();
+}
+
+template <> struct RunnableMethodTraits<ProxyFactory> {
+ static void RetainCallee(ProxyFactory* obj) {}
+ static void ReleaseCallee(ProxyFactory* obj) {}
+};
+
+ProxyFactory::ProxyFactory()
+ : uma_send_interval_(0) {
+ uma_send_interval_ = GetConfigInt(kDefaultSendUMADataInterval,
+ kUmaSendIntervalValue);
+}
+
+ProxyFactory::~ProxyFactory() {
+ DCHECK_EQ(proxies_.container().size(), 0);
+}
+
+void* ProxyFactory::GetAutomationServer(int launch_timeout,
+ const std::wstring& profile_name,
+ const std::wstring& extra_argument,
+ bool perform_version_check,
+ LaunchDelegate* delegate) {
+ ProxyCacheEntry* entry = NULL;
+ // Find already existing launcher thread for given profile
+ AutoLock lock(lock_);
+ for (size_t i = 0; i < proxies_.container().size(); ++i) {
+ if (!lstrcmpiW(proxies_[i]->profile_name.c_str(), profile_name.c_str())) {
+ entry = proxies_[i];
+ DCHECK(entry->thread.get() != NULL);
+ break;
+ }
+ }
+
+ if (entry == NULL) {
+ entry = new ProxyCacheEntry(profile_name);
+ proxies_.container().push_back(entry);
+ } else {
+ entry->ref_count++;
+ }
+
+
+ // Note we always queue request to the launch thread, even if we already
+ // have established proxy object. A simple lock around entry->proxy = proxy
+ // would allow calling LaunchDelegate directly from here if
+ // entry->proxy != NULL. Drawback is that callback may be invoked either in
+ // main thread or in background thread, which may confuse the client.
+ entry->thread->message_loop()->PostTask(FROM_HERE, NewRunnableMethod(this,
+ &ProxyFactory::CreateProxy, entry,
+ launch_timeout, extra_argument,
+ perform_version_check, delegate));
+
+ entry->thread->message_loop()->PostDelayedTask(FROM_HERE,
+ NewRunnableMethod(this, &ProxyFactory::SendUMAData, entry),
+ uma_send_interval_);
+
+ return entry;
+}
+
+void ProxyFactory::CreateProxy(ProxyFactory::ProxyCacheEntry* entry,
+ int launch_timeout,
+ const std::wstring& extra_chrome_arguments,
+ bool perform_version_check,
+ LaunchDelegate* delegate) {
+ DCHECK(entry->thread->thread_id() == PlatformThread::CurrentId());
+ if (entry->proxy) {
+ delegate->LaunchComplete(entry->proxy, entry->launch_result);
+ return;
+ }
+
+ // We *must* create automationproxy in a thread that has message loop,
+ // since SyncChannel::Context construction registers event to be watched
+ // through ObjectWatcher which subscribes for the current thread message loop
+ // destruction notification.
+
+ // At same time we must destroy/stop the thread from another thread.
+ ChromeFrameAutomationProxyImpl* proxy =
+ new ChromeFrameAutomationProxyImpl(launch_timeout);
+
+ // Launch browser
+ scoped_ptr<CommandLine> command_line(
+ chrome_launcher::CreateLaunchCommandLine());
+ command_line->AppendSwitchWithValue(switches::kAutomationClientChannelID,
+ ASCIIToWide(proxy->channel_id()));
+
+ // The metrics bug out because they attempt to use URLFetcher with a
+ // null URLRequestContext::default_request_context_. Turn them off for now.
+ // TODO(robertshield): Figure out why this is. It appears to have something
+ // to do with an improperly set up profile...
+ command_line->AppendSwitch(switches::kDisableMetrics);
+
+ // Chrome Frame never wants Chrome to start up with a First Run UI.
+ command_line->AppendSwitch(switches::kNoFirstRun);
+
+ // Place the profile directory in
+ // "<chrome_exe_path>\..\User Data\<profile-name>"
+ if (!entry->profile_name.empty()) {
+ std::wstring profile_path;
+ if (GetUserProfileBaseDirectory(&profile_path)) {
+ file_util::AppendToPath(&profile_path, entry->profile_name);
+ command_line->AppendSwitchWithValue(switches::kUserDataDir,
+ profile_path);
+ } else {
+ // Can't get the profile dir :-( We need one to work, so fail.
+ // We have no code for launch failure.
+ entry->launch_result = AutomationLaunchResult(-1);
+ }
+ }
+
+ std::wstring command_line_string(command_line->command_line_string());
+ // If there are any extra arguments, append them to the command line.
+ if (!extra_chrome_arguments.empty()) {
+ command_line_string += L' ' + extra_chrome_arguments;
+ }
+
+ automation_server_launch_start_time_ = base::TimeTicks::Now();
+
+ if (!base::LaunchApp(command_line_string, false, false, NULL)) {
+ // We have no code for launch failure.
+ entry->launch_result = AutomationLaunchResult(-1);
+ } else {
+ // Launch timeout may happen if the new instance tries to communicate
+ // with an existing Chrome instance that is hung and displays msgbox
+ // asking to kill the previous one. This could be easily observed if the
+ // already running Chrome instance is running as high-integrity process
+ // (started with "Run as Administrator" or launched by another high
+ // integrity process) hence our medium-integrity process
+ // cannot SendMessage to it with request to activate itself.
+
+ // TODO(stoyan) AutomationProxy eats Hello message, hence installing
+ // message filter is pointless, we can leverage ObjectWatcher and use
+ // system thread pool to notify us when proxy->AppLaunch event is signaled.
+ LaunchTimeStats launch_stats;
+ // Wait for the automation server launch result, then stash away the
+ // version string it reported.
+ entry->launch_result = proxy->WaitForAppLaunch();
+ launch_stats.Dump();
+
+ base::TimeDelta delta =
+ base::TimeTicks::Now() - automation_server_launch_start_time_;
+
+ if (entry->launch_result == AUTOMATION_SUCCESS) {
+ UMA_HISTOGRAM_TIMES("ChromeFrame.AutomationServerLaunchSuccessTime",
+ delta);
+ } else {
+ UMA_HISTOGRAM_TIMES("ChromeFrame.AutomationServerLaunchFailedTime",
+ delta);
+ }
+
+ UMA_HISTOGRAM_CUSTOM_COUNTS("ChromeFrame.LaunchResult",
+ entry->launch_result,
+ AUTOMATION_SUCCESS,
+ AUTOMATION_CREATE_TAB_FAILED,
+ AUTOMATION_CREATE_TAB_FAILED + 1);
+ }
+
+ // Finally set the proxy.
+ entry->proxy = proxy;
+ delegate->LaunchComplete(proxy, entry->launch_result);
+}
+
+bool ProxyFactory::ReleaseAutomationServer(void* server_id) {
+ DLOG(INFO) << __FUNCTION__;
+
+ if (!server_id) {
+ NOTREACHED();
+ return false;
+ }
+
+ ProxyCacheEntry* entry = reinterpret_cast<ProxyCacheEntry*>(server_id);
+
+ lock_.Acquire();
+ Vector::ContainerType::iterator it = std::find(proxies_.container().begin(),
+ proxies_.container().end(),
+ entry);
+ DCHECK(it != proxies_.container().end());
+ DCHECK(entry->thread->thread_id() != PlatformThread::CurrentId());
+ if (--entry->ref_count == 0) {
+ proxies_.container().erase(it);
+ }
+
+ lock_.Release();
+
+ // Destroy it.
+ if (entry->ref_count == 0) {
+ entry->thread->message_loop()->PostTask(FROM_HERE, NewRunnableMethod(this,
+ &ProxyFactory::DestroyProxy, entry));
+ // Wait until thread exits
+ entry->thread.reset();
+ DCHECK(entry->proxy == NULL);
+ delete entry;
+ }
+
+ return true;
+}
+
+void ProxyFactory::DestroyProxy(ProxyCacheEntry* entry) {
+ DCHECK(entry->thread->thread_id() == PlatformThread::CurrentId());
+ // Send pending UMA data if any.
+ SendUMAData(entry);
+ delete entry->proxy;
+ entry->proxy = NULL;
+}
+
+Singleton<ProxyFactory> g_proxy_factory;
+
+void ProxyFactory::SendUMAData(ProxyCacheEntry* proxy_entry) {
+ if (!proxy_entry) {
+ NOTREACHED() << __FUNCTION__ << " Invalid proxy entry";
+ return;
+ }
+
+ DCHECK(proxy_entry->thread->thread_id() == PlatformThread::CurrentId());
+
+ if (proxy_entry->proxy) {
+ ChromeFrameHistogramSnapshots::HistogramPickledList histograms =
+ chrome_frame_histograms_.GatherAllHistograms();
+
+ if (!histograms.empty()) {
+ proxy_entry->proxy->Send(
+ new AutomationMsg_RecordHistograms(0, histograms));
+ }
+ } else {
+ DLOG(INFO) << __FUNCTION__ << " No proxy available to service the request";
+ return;
+ }
+
+ MessageLoop::current()->PostDelayedTask(FROM_HERE, NewRunnableMethod(
+ this, &ProxyFactory::SendUMAData, proxy_entry), uma_send_interval_);
+}
+
+template <> struct RunnableMethodTraits<ChromeFrameAutomationClient> {
+ static void RetainCallee(ChromeFrameAutomationClient* obj) {}
+ static void ReleaseCallee(ChromeFrameAutomationClient* obj) {}
+};
+
+ChromeFrameAutomationClient::ChromeFrameAutomationClient()
+ : chrome_frame_delegate_(NULL),
+ chrome_window_(NULL),
+ tab_window_(NULL),
+ parent_window_(NULL),
+ automation_server_(NULL),
+ automation_server_id_(NULL),
+ ui_thread_id_(NULL),
+ incognito_(false),
+ init_state_(UNINITIALIZED),
+ use_chrome_network_(false),
+ proxy_factory_(g_proxy_factory.get()),
+ handle_top_level_requests_(false),
+ tab_handle_(-1),
+ external_tab_cookie_(NULL) {
+}
+
+ChromeFrameAutomationClient::~ChromeFrameAutomationClient() {
+ // Uninitialize must be called prior to the destructor
+ DCHECK(automation_server_ == NULL);
+}
+
+bool ChromeFrameAutomationClient::Initialize(
+ ChromeFrameDelegate* chrome_frame_delegate,
+ int automation_server_launch_timeout,
+ bool perform_version_check,
+ const std::wstring& profile_name,
+ const std::wstring& extra_chrome_arguments,
+ bool incognito) {
+ DCHECK(!IsWindow());
+ chrome_frame_delegate_ = chrome_frame_delegate;
+ incognito_ = incognito;
+ ui_thread_id_ = PlatformThread::CurrentId();
+
+#ifndef NDEBUG
+ // In debug mode give more time to work with a debugger.
+ if (automation_server_launch_timeout != INFINITE)
+ automation_server_launch_timeout *= 2;
+#endif // NDEBUG
+
+ // Create a window on the UI thread for marshaling messages back and forth
+ // from the IPC thread. This window cannot be a message only window as the
+ // external chrome tab window is created as a child of this window. This
+ // window is eventually reparented to the ActiveX/NPAPI plugin window.
+ if (!Create(GetDesktopWindow(), NULL, NULL,
+ WS_CHILDWINDOW | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
+ WS_EX_TOOLWINDOW)) {
+ NOTREACHED();
+ return false;
+ }
+
+ // Mark our state as initializing. We'll reach initialized once
+ // InitializeComplete is called successfully.
+ init_state_ = INITIALIZING;
+
+ automation_server_id_ = proxy_factory_->GetAutomationServer(
+ automation_server_launch_timeout,
+ profile_name, extra_chrome_arguments, perform_version_check,
+ static_cast<ProxyFactory::LaunchDelegate*>(this));
+
+ return true;
+}
+
+void ChromeFrameAutomationClient::Uninitialize() {
+ DLOG(INFO) << __FUNCTION__;
+
+ init_state_ = UNINITIALIZING;
+
+ // Called from client's FinalRelease() / destructor
+ // ChromeFrameAutomationClient may wait for the initialization (launch)
+ // to complete while Uninitialize is called.
+ // We either have to:
+ // 1) Make ReleaseAutomationServer blocking call (wait until thread exits)
+ // 2) Behave like a COM object i.e. increase module lock count.
+ // Otherwise the DLL may get unloaded while we have running threads.
+ // Unfortunately in NPAPI case we cannot increase module lock count, hence
+ // we stick with blocking/waiting
+ if (tab_.get()) {
+ tab_->RemoveObserver(this);
+ tab_ = NULL; // scoped_refptr::Release
+ }
+
+ // Clean up any outstanding requests
+ CleanupRequests();
+
+ // Wait for the background thread to exit.
+ ReleaseAutomationServer();
+
+ // We must destroy the window, since if there are pending tasks
+ // window procedure may be invoked after DLL is unloaded.
+ // Unfortunately pending tasks are leaked.
+ if (m_hWnd)
+ DestroyWindow();
+
+ chrome_frame_delegate_ = NULL;
+ init_state_ = UNINITIALIZED;
+}
+
+bool ChromeFrameAutomationClient::InitiateNavigation(const std::string& url) {
+ if (url.empty())
+ return false;
+
+ url_ = GURL(url);
+
+ // Catch invalid URLs early.
+ if (!url_.is_valid()) {
+ DLOG(ERROR) << "Invalid URL passed to InitiateNavigation: " << url;
+ return false;
+ }
+
+ if (is_initialized()) {
+ BeginNavigate(GURL(url));
+ }
+
+ return true;
+}
+
+bool ChromeFrameAutomationClient::NavigateToIndex(int index) {
+ // Could be NULL if we failed to launch Chrome in LaunchAutomationServer()
+ if (!automation_server_ || !tab_.get() || !tab_->is_valid()) {
+ return false;
+ }
+
+ DCHECK(::IsWindow(chrome_window_));
+
+ IPC::SyncMessage* msg = new AutomationMsg_NavigateExternalTabAtIndex(
+ 0, tab_->handle(), index, NULL);
+ automation_server_->SendAsAsync(msg, NewCallback(this,
+ &ChromeFrameAutomationClient::BeginNavigateCompleted), this);
+ return true;
+}
+
+bool ChromeFrameAutomationClient::ForwardMessageFromExternalHost(
+ const std::string& message, const std::string& origin,
+ const std::string& target) {
+ // Could be NULL if we failed to launch Chrome in LaunchAutomationServer()
+ if (!is_initialized())
+ return false;
+
+ tab_->HandleMessageFromExternalHost(message, origin, target);
+ return true;
+}
+
+bool ChromeFrameAutomationClient::SetProxySettings(
+ const std::string& json_encoded_proxy_settings) {
+ if (!is_initialized())
+ return false;
+ automation_server_->SendProxyConfig(json_encoded_proxy_settings);
+ return true;
+}
+
+void ChromeFrameAutomationClient::BeginNavigate(const GURL& url) {
+ // Could be NULL if we failed to launch Chrome in LaunchAutomationServer()
+ if (!automation_server_ || !tab_.get()) {
+ DLOG(WARNING) << "BeginNavigate - can't navigate.";
+ ReportNavigationError(AUTOMATION_MSG_NAVIGATION_ERROR, url_.spec());
+ return;
+ }
+
+ DCHECK(::IsWindow(chrome_window_));
+
+ if (!tab_->is_valid()) {
+ DLOG(WARNING) << "BeginNavigate - tab isn't valid.";
+ return;
+ }
+
+ IPC::SyncMessage* msg =
+ new AutomationMsg_NavigateInExternalTab(0, tab_->handle(), url, NULL);
+ automation_server_->SendAsAsync(msg, NewCallback(this,
+ &ChromeFrameAutomationClient::BeginNavigateCompleted), this);
+
+ RECT client_rect = {0};
+ chrome_frame_delegate_->GetBounds(&client_rect);
+ Resize(client_rect.right - client_rect.left,
+ client_rect.bottom - client_rect.top,
+ SWP_NOACTIVATE | SWP_NOZORDER);
+}
+
+void ChromeFrameAutomationClient::BeginNavigateCompleted(
+ AutomationMsg_NavigationResponseValues result) {
+ if (result == AUTOMATION_MSG_NAVIGATION_ERROR)
+ ReportNavigationError(AUTOMATION_MSG_NAVIGATION_ERROR, url_.spec());
+}
+
+void ChromeFrameAutomationClient::FindInPage(const std::wstring& search_string,
+ FindInPageDirection forward,
+ FindInPageCase match_case,
+ bool find_next) {
+ DCHECK(tab_.get());
+
+ // What follows is quite similar to TabProxy::FindInPage() but uses
+ // the SyncMessageReplyDispatcher to avoid concerns about blocking
+ // synchronous messages.
+ AutomationMsg_Find_Params params;
+ params.unused = 0;
+ params.search_string = WideToUTF16Hack(search_string);
+ params.find_next = find_next;
+ params.match_case = (match_case == CASE_SENSITIVE);
+ params.forward = (forward == FWD);
+
+ IPC::SyncMessage* msg =
+ new AutomationMsg_Find(0, tab_->handle(), params, NULL, NULL);
+ automation_server_->SendAsAsync(msg, NULL, this);
+}
+
+void ChromeFrameAutomationClient::CreateExternalTab() {
+ AutomationLaunchResult launch_result = AUTOMATION_SUCCESS;
+ DCHECK(IsWindow());
+ DCHECK(automation_server_ != NULL);
+
+ const IPC::ExternalTabSettings settings = {
+ m_hWnd,
+ gfx::Rect(),
+ WS_CHILD,
+ incognito_,
+ !use_chrome_network_,
+ handle_top_level_requests_,
+ GURL(url_)
+ };
+
+ UMA_HISTOGRAM_CUSTOM_COUNTS(
+ "ChromeFrame.HostNetworking", !use_chrome_network_, 0, 1, 2);
+
+ UMA_HISTOGRAM_CUSTOM_COUNTS(
+ "ChromeFrame.HandleTopLevelRequests", handle_top_level_requests_, 0, 1,
+ 2);
+
+ IPC::SyncMessage* message =
+ new AutomationMsg_CreateExternalTab(0, settings, NULL, NULL, NULL);
+ automation_server_->SendAsAsync(message, NewCallback(this,
+ &ChromeFrameAutomationClient::CreateExternalTabComplete), this);
+}
+
+void ChromeFrameAutomationClient::CreateExternalTabComplete(HWND chrome_window,
+ HWND tab_window, int tab_handle) {
+ if (!automation_server_) {
+ // If we receive this notification while shutting down, do nothing.
+ DLOG(ERROR) << "CreateExternalTabComplete called when automation server "
+ << "was null!";
+ return;
+ }
+
+ AutomationLaunchResult launch_result = AUTOMATION_SUCCESS;
+ if (tab_handle == 0 || !::IsWindow(chrome_window) ||
+ !::IsWindow(chrome_window)) {
+ launch_result = AUTOMATION_CREATE_TAB_FAILED;
+ } else {
+ chrome_window_ = chrome_window;
+ tab_window_ = tab_window;
+ tab_ = automation_server_->CreateTabProxy(tab_handle);
+ tab_->AddObserver(this);
+ tab_handle_ = tab_handle;
+ }
+
+ PostTask(FROM_HERE, NewRunnableMethod(this,
+ &ChromeFrameAutomationClient::InitializeComplete, launch_result));
+}
+
+void ChromeFrameAutomationClient::SetEnableExtensionAutomation(
+ bool enable_automation) {
+ if (!is_initialized())
+ return;
+
+ automation_server_->SetEnableExtensionAutomation(enable_automation);
+}
+
+// Invoked in launch background thread.
+void ChromeFrameAutomationClient::LaunchComplete(
+ ChromeFrameAutomationProxy* proxy,
+ AutomationLaunchResult result) {
+ // If we're shutting down we don't keep a pointer to the automation server.
+ if (init_state_ != UNINITIALIZING) {
+ DCHECK(init_state_ == INITIALIZING);
+ automation_server_ = proxy;
+ } else {
+ DLOG(INFO) << "Not storing automation server pointer due to shutting down";
+ }
+
+ if (result == AUTOMATION_SUCCESS) {
+ // NOTE: A potential problem here is that Uninitialize() may just have
+ // been called so we need to be careful and check the automation_server_
+ // pointer.
+ if (automation_server_ != NULL) {
+ // If we have a valid tab_handle here it means that we are attaching to
+ // an existing ExternalTabContainer instance, in which case we don't
+ // want to create an external tab instance in Chrome.
+ if (external_tab_cookie_ == NULL) {
+ // Continue with Initialization - Create external tab
+ CreateExternalTab();
+ } else {
+ // Send a notification to Chrome that we are ready to connect to the
+ // ExternalTab.
+ IPC::SyncMessage* message =
+ new AutomationMsg_ConnectExternalTab(0, external_tab_cookie_, NULL,
+ NULL, NULL);
+ automation_server_->SendAsAsync(message, NewCallback(this,
+ &ChromeFrameAutomationClient::CreateExternalTabComplete), this);
+ }
+ }
+ } else {
+ // Launch failed. Note, we cannot delete proxy here.
+ PostTask(FROM_HERE, NewRunnableMethod(this,
+ &ChromeFrameAutomationClient::InitializeComplete, result));
+ }
+}
+
+void ChromeFrameAutomationClient::InitializeComplete(
+ AutomationLaunchResult result) {
+ DCHECK(PlatformThread::CurrentId() == ui_thread_id_);
+ std::string version = automation_server_->server_version();
+
+ if (result != AUTOMATION_SUCCESS) {
+ DLOG(WARNING) << "InitializeComplete: failure " << result;
+ ReleaseAutomationServer();
+ } else {
+ init_state_ = INITIALIZED;
+
+ // If the host already have a window, ask Chrome to re-parent.
+ if (parent_window_)
+ SetParentWindow(parent_window_);
+ }
+
+ if (chrome_frame_delegate_) {
+ if (result == AUTOMATION_SUCCESS) {
+ chrome_frame_delegate_->OnAutomationServerReady();
+ } else {
+ chrome_frame_delegate_->OnAutomationServerLaunchFailed(result, version);
+ }
+ }
+}
+
+// This is invoked in channel's background thread.
+// Cannot call any method of the activex/npapi here since they are STA
+// kind of beings.
+// By default we marshal the IPC message to the main/GUI thread and from there
+// we safely invoke chrome_frame_delegate_->OnMessageReceived(msg).
+void ChromeFrameAutomationClient::OnMessageReceived(TabProxy* tab,
+ const IPC::Message& msg) {
+ DCHECK(tab == tab_.get());
+
+ // Early check to avoid needless marshaling
+ if (chrome_frame_delegate_ == NULL)
+ return;
+
+ CallDelegate(FROM_HERE, NewRunnableMethod(chrome_frame_delegate_,
+ &ChromeFrameDelegate::OnMessageReceived, msg));
+}
+
+void ChromeFrameAutomationClient::ReportNavigationError(
+ AutomationMsg_NavigationResponseValues error_code,
+ const std::string& url) {
+ CallDelegate(FROM_HERE, NewRunnableMethod(chrome_frame_delegate_,
+ &ChromeFrameDelegate::OnLoadFailed,
+ error_code,
+ url));
+}
+
+void ChromeFrameAutomationClient::Resize(int width, int height,
+ int flags) {
+ if (tab_.get() && ::IsWindow(chrome_window())) {
+ SetWindowPos(HWND_TOP, 0, 0, width, height, flags);
+ tab_->Reposition(chrome_window(), HWND_TOP, 0, 0, width, height,
+ flags, m_hWnd);
+ }
+}
+
+void ChromeFrameAutomationClient::SetParentWindow(HWND parent_window) {
+ parent_window_ = parent_window;
+ // If we're done with the initialization step, go ahead
+ if (is_initialized()) {
+ if (parent_window == NULL) {
+ // Hide and reparent the automation window. This window will get
+ // reparented to the new ActiveX/Active document window when it gets
+ // created.
+ ShowWindow(SW_HIDE);
+ SetParent(GetDesktopWindow());
+ } else {
+ if (!::IsWindow(chrome_window())) {
+ DLOG(WARNING) << "Invalid Chrome Window handle in SetParentWindow";
+ return;
+ }
+
+ if (!SetParent(parent_window)) {
+ NOTREACHED();
+ DLOG(WARNING) << "Failed to set parent window for automation window. "
+ << "Error = "
+ << GetLastError();
+ return;
+ }
+
+ RECT parent_client_rect = {0};
+ ::GetClientRect(parent_window, &parent_client_rect);
+ int width = parent_client_rect.right - parent_client_rect.left;
+ int height = parent_client_rect.bottom - parent_client_rect.top;
+
+ Resize(width, height, SWP_SHOWWINDOW | SWP_NOZORDER);
+ }
+ }
+}
+
+void ChromeFrameAutomationClient::ReleaseAutomationServer() {
+ DLOG(INFO) << __FUNCTION__;
+ if (automation_server_id_) {
+ // Cache the server id and clear the automation_server_id_ before
+ // calling ReleaseAutomationServer. The reason we do this is that
+ // we must cancel pending messages before we release the automation server.
+ // Furthermore, while ReleaseAutomationServer is running, we could get
+ // a callback to LaunchComplete which is where we normally get our pointer
+ // to the automation server and there we check the server id for NULLness
+ // and do nothing if it is NULL.
+ void* server_id = automation_server_id_;
+ automation_server_id_ = NULL;
+
+ if (automation_server_) {
+ // Make sure to clean up any pending sync messages before we go away.
+ automation_server_->CancelAsync(this);
+ automation_server_ = NULL;
+ }
+
+ proxy_factory_->ReleaseAutomationServer(server_id);
+
+ // automation_server_ must not have been set to non NULL.
+ // (if this regresses, start by looking at LaunchComplete()).
+ DCHECK(automation_server_ == NULL);
+ } else {
+ DCHECK(automation_server_ == NULL);
+ }
+}
+
+void ChromeFrameAutomationClient::SendContextMenuCommandToChromeFrame(
+ int selected_command) {
+ DCHECK(tab_ != NULL);
+ tab_->SendContextMenuCommand(selected_command);
+}
+
+std::wstring ChromeFrameAutomationClient::GetVersion() const {
+ static FileVersionInfo* version_info =
+ FileVersionInfo::CreateFileVersionInfoForCurrentModule();
+
+ std::wstring version;
+ if (version_info)
+ version = version_info->product_version();
+
+ return version;
+}
+
+void ChromeFrameAutomationClient::CallDelegate(
+ const tracked_objects::Location& from_here, Task* delegate_task ) {
+ delegate_task->SetBirthPlace(from_here);
+ PostTask(FROM_HERE, NewRunnableMethod(this,
+ &ChromeFrameAutomationClient::CallDelegateImpl,
+ delegate_task));
+}
+
+void ChromeFrameAutomationClient::CallDelegateImpl(Task* delegate_task) {
+ if (chrome_frame_delegate_) {
+ // task's object should be == chrome_frame_delegate_
+ delegate_task->Run();
+ }
+
+ delete delegate_task;
+}
+
+void ChromeFrameAutomationClient::Print(HDC print_dc,
+ const RECT& print_bounds) {
+ if (!tab_window_) {
+ NOTREACHED();
+ return;
+ }
+
+ HDC window_dc = ::GetDC(tab_window_);
+
+ BitBlt(print_dc, print_bounds.left, print_bounds.top,
+ print_bounds.right - print_bounds.left,
+ print_bounds.bottom - print_bounds.top,
+ window_dc, print_bounds.left, print_bounds.top,
+ SRCCOPY);
+
+ ::ReleaseDC(tab_window_, window_dc);
+}
+
+void ChromeFrameAutomationClient::PrintTab() {
+ tab_->PrintAsync();
+}
+
+// IPC:Message::Sender implementation
+bool ChromeFrameAutomationClient::Send(IPC::Message* msg) {
+ return automation_server_->Send(msg);
+}
+
+bool ChromeFrameAutomationClient::AddRequest(PluginUrlRequest* request) {
+ if (!request) {
+ NOTREACHED();
+ return false;
+ }
+
+ DCHECK(request_map_.end() == request_map_.find(request->id()));
+ request_map_[request->id()] = request;
+ return true;
+}
+
+bool ChromeFrameAutomationClient::ReadRequest(
+ int request_id, int bytes_to_read) {
+ bool result = false;
+ PluginUrlRequest* request = LookupRequest(request_id);
+ if (request)
+ result = request->Read(bytes_to_read);
+
+ return result;
+}
+
+void ChromeFrameAutomationClient::RemoveRequest(PluginUrlRequest* request) {
+ DCHECK(request_map_.end() != request_map_.find(request->id()));
+ request_map_.erase(request->id());
+}
+
+void ChromeFrameAutomationClient::RemoveRequest(
+ int request_id, int reason, bool abort) {
+ PluginUrlRequest* request = LookupRequest(request_id);
+ if (request) {
+ if (abort) {
+ request->Stop();
+ DCHECK(request_map_.end() == request_map_.find(request_id));
+ } else {
+ request_map_.erase(request_id);
+ }
+ }
+}
+
+PluginUrlRequest* ChromeFrameAutomationClient::LookupRequest(
+ int request_id) const {
+ PluginUrlRequest* request = NULL;
+ RequestMap::const_iterator it = request_map_.find(request_id);
+ if (request_map_.end() != it)
+ request = (*it).second;
+ return request;
+}
+
+bool ChromeFrameAutomationClient::IsValidRequest(
+ PluginUrlRequest* request) const {
+ bool is_valid = false;
+ // if request is invalid then request->id() won't work
+ // hence perform reverse map lookup for validity of the
+ // request pointer.
+ if (request) {
+ for (RequestMap::const_iterator it = request_map_.begin();
+ it != request_map_.end(); it++) {
+ if (request == (*it).second) {
+ is_valid = true;
+ break;
+ }
+ }
+ }
+
+ return is_valid;
+}
+
+void ChromeFrameAutomationClient::CleanupRequests() {
+ while (request_map_.size()) {
+ PluginUrlRequest* request = request_map_.begin()->second;
+ if (request) {
+ int request_id = request->id();
+ request->Stop();
+ DCHECK(request_map_.end() == request_map_.find(request_id));
+ }
+ }
+
+ DCHECK(request_map_.empty());
+ request_map_.clear();
+}
+
+bool ChromeFrameAutomationClient::Reinitialize(
+ ChromeFrameDelegate* delegate) {
+ if (!tab_.get() || !::IsWindow(chrome_window_)) {
+ NOTREACHED();
+ DLOG(WARNING) << "ChromeFrameAutomationClient instance reused "
+ << "with invalid tab";
+ return false;
+ }
+
+ if (!delegate) {
+ NOTREACHED();
+ return false;
+ }
+
+ chrome_frame_delegate_ = delegate;
+ SetParentWindow(NULL);
+ return true;
+}
+
+void ChromeFrameAutomationClient::AttachExternalTab(
+ intptr_t external_tab_cookie) {
+ DCHECK(tab_.get() == NULL);
+ DCHECK(tab_handle_ == -1);
+
+ external_tab_cookie_ = external_tab_cookie;
+}
diff --git a/chrome_frame/chrome_frame_automation.h b/chrome_frame/chrome_frame_automation.h
new file mode 100644
index 0000000..796facb
--- /dev/null
+++ b/chrome_frame/chrome_frame_automation.h
@@ -0,0 +1,356 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_FRAME_AUTOMATION_H_
+#define CHROME_FRAME_CHROME_FRAME_AUTOMATION_H_
+
+#include <atlbase.h>
+#include <atlwin.h>
+#include <string>
+#include <map>
+
+#include "base/lock.h"
+#include "base/ref_counted.h"
+#include "base/scoped_handle.h"
+#include "base/stack_container.h"
+#include "base/task.h"
+#include "base/timer.h"
+#include "base/thread.h"
+#include "chrome/test/automation/automation_proxy.h"
+#include "chrome/test/automation/tab_proxy.h"
+#include "chrome_frame/chrome_frame_delegate.h"
+#include "chrome_frame/chrome_frame_histograms.h"
+#include "chrome_frame/plugin_url_request.h"
+
+const unsigned long kCommandExecutionTimeout = 4000; // NOLINT, 4 seconds
+
+class ProxyFactory;
+
+struct DECLSPEC_NOVTABLE ChromeFrameAutomationProxy {
+ virtual bool Send(IPC::Message* msg) = 0;
+
+ virtual void SendAsAsync(IPC::SyncMessage* msg, void* callback,
+ void* key) = 0;
+ virtual void CancelAsync(void* key) = 0;
+ virtual scoped_refptr<TabProxy> CreateTabProxy(int handle) = 0;
+ virtual std::string server_version() = 0;
+
+ virtual void SendProxyConfig(const std::string&) = 0;
+ virtual void SetEnableExtensionAutomation(bool enable) = 0;
+ protected:
+ ~ChromeFrameAutomationProxy() {}
+};
+
+// We extend the AutomationProxy class to handle our custom
+// IPC messages
+class ChromeFrameAutomationProxyImpl : public ChromeFrameAutomationProxy,
+ // We have to derive from automationproxy since we want access to some members
+ // (tracker_ & channel_) - simple aggregation wont work;
+ // .. and non-public inheritance is verboten.
+ public AutomationProxy {
+ public:
+ virtual void SendAsAsync(IPC::SyncMessage* msg, void* callback, void* key);
+
+ virtual void CancelAsync(void* key);
+
+ virtual scoped_refptr<TabProxy> CreateTabProxy(int handle);
+ virtual std::string server_version() {
+ return AutomationProxy::server_version();
+ }
+
+
+ virtual bool Send(IPC::Message* msg) {
+ return AutomationProxy::Send(msg);
+ }
+
+ virtual void SendProxyConfig(const std::string& p) {
+ AutomationProxy::SendProxyConfig(p);
+ }
+
+ virtual void SetEnableExtensionAutomation(bool e) {
+ AutomationProxy::SetEnableExtensionAutomation(e);
+ }
+
+ protected:
+ explicit ChromeFrameAutomationProxyImpl(int launch_timeout);
+ ~ChromeFrameAutomationProxyImpl();
+ class CFMsgDispatcher;
+ scoped_refptr<CFMsgDispatcher> sync_;
+ friend class ProxyFactory;
+};
+
+// We must create and destroy automation proxy in a thread with a message loop.
+// Hence thread cannot be a member of the proxy.
+class ProxyFactory {
+ public:
+ // Callback when chrome process launch is complete and automation handshake
+ // (Hello message) is established.
+ struct DECLSPEC_NOVTABLE LaunchDelegate {
+ virtual void LaunchComplete(ChromeFrameAutomationProxy* proxy,
+ AutomationLaunchResult result) = 0;
+ };
+
+ ProxyFactory();
+ ~ProxyFactory();
+ // FIXME: we should pass the result as output parameter, not as return value
+ // since, LaunchDelegate can be invoked before this function returns.
+ virtual void* GetAutomationServer(int launch_timeout,
+ const std::wstring& profile_name,
+ // Extra command line argument when launching Chrome
+ const std::wstring& extra_argument,
+ bool perform_version_check,
+ LaunchDelegate* delegate);
+ virtual bool ReleaseAutomationServer(void* server_id);
+
+ private:
+ struct ProxyCacheEntry {
+ std::wstring profile_name;
+ int ref_count;
+ scoped_ptr<base::Thread> thread;
+ ChromeFrameAutomationProxyImpl* proxy;
+ AutomationLaunchResult launch_result;
+ explicit ProxyCacheEntry(const std::wstring& profile);
+ };
+
+ void CreateProxy(ProxyCacheEntry* entry,
+ int launch_timeout,
+ const std::wstring& extra_chrome_arguments,
+ bool perform_version_check,
+ LaunchDelegate* delegate);
+ void DestroyProxy(ProxyCacheEntry* entry);
+
+ void SendUMAData(ProxyCacheEntry* proxy_entry);
+
+ typedef StackVector<ProxyCacheEntry*, 4> Vector;
+ Vector proxies_;
+ // Lock if we are going to call GetAutomationServer from more than one thread.
+ Lock lock_;
+
+ // Used for UMA histogram logging to measure the time for the chrome
+ // automation server to start;
+ base::TimeTicks automation_server_launch_start_time_;
+
+ // Gathers histograms to be sent to Chrome.
+ ChromeFrameHistogramSnapshots chrome_frame_histograms_;
+
+ // Interval for sending UMA data
+ int uma_send_interval_;
+};
+
+// T is expected to be something CWindowImpl derived, or at least to have
+// PostMessage(UINT, WPARAM) method. Do not forget to CHAIN_MSG_MAP
+template <class T> class TaskMarshallerThroughWindowsMessages {
+ public:
+ void PostTask(const tracked_objects::Location& from_here, Task* task) {
+ task->SetBirthPlace(from_here);
+ T* this_ptr = static_cast<T*>(this);
+ if (this_ptr->IsWindow()) {
+ this_ptr->PostMessage(MSG_EXECUTE_TASK, reinterpret_cast<WPARAM>(task));
+ } else {
+ DLOG(INFO) << "Dropping MSG_EXECUTE_TASK message for destroyed window.";
+ }
+ }
+
+ BEGIN_MSG_MAP(PostMessageMarshaller)
+ MESSAGE_HANDLER(MSG_EXECUTE_TASK, ExecuteTask)
+ END_MSG_MAP()
+
+ private:
+ enum { MSG_EXECUTE_TASK = WM_APP + 6 };
+ inline LRESULT ExecuteTask(UINT, WPARAM wparam, LPARAM,
+ BOOL& handled) { // NOLINT
+ Task* task = reinterpret_cast<Task*>(wparam);
+ task->Run();
+ delete task;
+ return 0;
+ }
+};
+
+// Handles all automation requests initiated from the chrome frame objects.
+// These include the chrome tab/chrome frame activex/chrome frame npapi
+// plugin objects.
+class ChromeFrameAutomationClient
+ : public CWindowImpl<ChromeFrameAutomationClient>,
+ public TaskMarshallerThroughWindowsMessages<ChromeFrameAutomationClient>,
+ public PluginRequestHandler,
+ public TabProxy::TabProxyDelegate,
+ public ProxyFactory::LaunchDelegate {
+ public:
+ ChromeFrameAutomationClient();
+ ~ChromeFrameAutomationClient();
+
+ // Called from UI thread.
+ virtual bool Initialize(ChromeFrameDelegate* chrome_frame_delegate,
+ int automation_server_launch_timeout,
+ bool perform_version_check,
+ const std::wstring& profile_name,
+ const std::wstring& extra_chrome_arguments,
+ bool incognito);
+ void Uninitialize();
+
+ virtual bool InitiateNavigation(const std::string& url);
+ virtual bool NavigateToIndex(int index);
+ bool ForwardMessageFromExternalHost(const std::string& message,
+ const std::string& origin,
+ const std::string& target);
+ bool SetProxySettings(const std::string& json_encoded_proxy_settings);
+
+ virtual void SetEnableExtensionAutomation(bool enable_automation);
+
+ void FindInPage(const std::wstring& search_string,
+ FindInPageDirection forward,
+ FindInPageCase match_case,
+ bool find_next);
+
+ TabProxy* tab() const { return tab_.get(); }
+
+ BEGIN_MSG_MAP(ChromeFrameAutomationClient)
+ CHAIN_MSG_MAP(
+ TaskMarshallerThroughWindowsMessages<ChromeFrameAutomationClient>)
+ END_MSG_MAP()
+
+ void set_delegate(ChromeFrameDelegate* d) {
+ chrome_frame_delegate_ = d;
+ }
+
+ // Resizes the hosted chrome window. This is brokered to the chrome
+ // automation instance as the host browser could be running under low IL,
+ // which would cause the SetWindowPos call to fail.
+ void Resize(int width, int height, int flags);
+
+ // Sets the passed in window as the parent of the external tab.
+ void SetParentWindow(HWND parent_window);
+
+ void SendContextMenuCommandToChromeFrame(int selected_command);
+
+ HWND tab_window() const {
+ return tab_window_;
+ }
+
+ void ReleaseAutomationServer();
+
+ // Returns the version number of plugin dll.
+ std::wstring GetVersion() const;
+
+ // BitBlts the contents of the chrome window to the print dc.
+ void Print(HDC print_dc, const RECT& print_bounds);
+
+ // Called in full tab mode and indicates a request to chrome to print
+ // the whole tab.
+ void PrintTab();
+
+ // PluginRequestHandler
+ bool AddRequest(PluginUrlRequest* request);
+ void RemoveRequest(PluginUrlRequest* request);
+ virtual bool Send(IPC::Message* msg);
+
+ // URL request related
+ bool ReadRequest(int request_id, int bytes_to_read);
+ void RemoveRequest(int request_id, int reason, bool abort);
+ PluginUrlRequest* LookupRequest(int request_id) const;
+ bool IsValidRequest(PluginUrlRequest* request) const;
+ void CleanupRequests();
+
+ void set_use_chrome_network(bool use_chrome_network) {
+ use_chrome_network_ = use_chrome_network;
+ }
+ bool use_chrome_network() const {
+ return use_chrome_network_;
+ }
+
+#ifdef UNIT_TEST
+ void set_proxy_factory(ProxyFactory* factory) {
+ proxy_factory_ = factory;
+ }
+#endif
+
+ void set_handle_top_level_requests(bool handle_top_level_requests) {
+ handle_top_level_requests_ = handle_top_level_requests;
+ }
+
+ // Called if the same instance of the ChromeFrameAutomationClient object
+ // is reused.
+ bool Reinitialize(ChromeFrameDelegate* chrome_frame_delegate);
+
+ // Attaches an existing external tab to this automation client instance.
+ void AttachExternalTab(intptr_t external_tab_cookie);
+
+ protected:
+ // ChromeFrameAutomationProxy::LaunchDelegate implementation.
+ virtual void LaunchComplete(ChromeFrameAutomationProxy* proxy,
+ AutomationLaunchResult result);
+ // TabProxyDelegate implementation
+ virtual void OnMessageReceived(TabProxy* tab, const IPC::Message& msg);
+
+ void CreateExternalTab();
+ void CreateExternalTabComplete(HWND chrome_window, HWND tab_window,
+ int tab_handle);
+ // Called in UI thread. Here we fire event to the client notifying for
+ // the result of Initialize() method call.
+ void InitializeComplete(AutomationLaunchResult result);
+
+ private:
+ typedef std::map<int, scoped_refptr<PluginUrlRequest> > RequestMap;
+
+ // Usage: From bkgnd thread invoke:
+ // CallDelegate(FROM_HERE, NewRunnableMethod(chrome_frame_delegate_,
+ // ChromeFrameDelegate::Something,
+ // param1,
+ // param2));
+ void CallDelegate(const tracked_objects::Location& from_here,
+ Task* delegate_task);
+ // The workhorse method called in main/GUI thread which is going to
+ // execute ChromeFrameDelegate method encapsulated in delegate_task.
+ void CallDelegateImpl(Task* delegate_task);
+
+ HWND chrome_window() const { return chrome_window_; }
+ void BeginNavigate(const GURL& url);
+ void BeginNavigateCompleted(AutomationMsg_NavigationResponseValues result);
+
+ // Helpers
+ void ReportNavigationError(AutomationMsg_NavigationResponseValues error_code,
+ const std::string& url);
+
+ bool is_initialized() const {
+ return init_state_ == INITIALIZED;
+ }
+
+ bool incognito_;
+ HWND parent_window_;
+ PlatformThreadId ui_thread_id_;
+
+ void* automation_server_id_;
+ ChromeFrameAutomationProxy* automation_server_;
+ HWND chrome_window_;
+ scoped_refptr<TabProxy> tab_;
+ ChromeFrameDelegate* chrome_frame_delegate_;
+ GURL url_;
+
+ // Handle to the underlying chrome window. This is a child of the external
+ // tab window.
+ HWND tab_window_;
+
+ // Keeps track of the version of Chrome we're talking to.
+ std::string automation_server_version_;
+
+ // Map of outstanding requests
+ RequestMap request_map_;
+
+ typedef enum InitializationState {
+ UNINITIALIZED = 0,
+ INITIALIZING,
+ INITIALIZED,
+ UNINITIALIZING,
+ };
+
+ InitializationState init_state_;
+ bool use_chrome_network_;
+ bool handle_top_level_requests_;
+ ProxyFactory* proxy_factory_;
+ int tab_handle_;
+ // Only used if we attach to an existing tab.
+ intptr_t external_tab_cookie_;
+};
+
+#endif // CHROME_FRAME_CHROME_FRAME_AUTOMATION_H_
diff --git a/chrome_frame/chrome_frame_delegate.cc b/chrome_frame/chrome_frame_delegate.cc
new file mode 100644
index 0000000..9a1069b
--- /dev/null
+++ b/chrome_frame/chrome_frame_delegate.cc
@@ -0,0 +1,67 @@
+// Copyright (c) 2009 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 "chrome_frame/chrome_frame_delegate.h"
+
+bool ChromeFrameDelegateImpl::IsTabMessage(const IPC::Message& message,
+ int* tab_handle) {
+ bool is_tab_message = true;
+ IPC_BEGIN_MESSAGE_MAP(ChromeFrameDelegateImpl, message)
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_NavigationStateChanged, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_UpdateTargetUrl, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_HandleAccelerator, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_TabbedOut, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_OpenURL, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_NavigationFailed, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_DidNavigate, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_TabLoaded, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_ForwardMessageToExternalHost, )
+ IPC_MESSAGE_HANDLER_GENERIC(
+ AutomationMsg_ForwardContextMenuToExternalHost, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_RequestStart, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_RequestRead, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_RequestEnd, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_SetCookieAsync, )
+ IPC_MESSAGE_HANDLER_GENERIC(AutomationMsg_AttachExternalTab, )
+ IPC_MESSAGE_UNHANDLED(is_tab_message = false);
+ IPC_END_MESSAGE_MAP()
+
+ if (is_tab_message) {
+ // Read tab handle from the message.
+ void* iter = NULL;
+ is_tab_message = message.ReadInt(&iter, tab_handle);
+ }
+
+ return is_tab_message;
+}
+
+void ChromeFrameDelegateImpl::OnMessageReceived(const IPC::Message& msg) {
+ if (!IsValid()) {
+ DLOG(WARNING) << __FUNCTION__
+ << " Msgs received for a NULL automation client instance";
+ return;
+ }
+
+ IPC_BEGIN_MESSAGE_MAP(ChromeFrameDelegateImpl, msg)
+ IPC_MESSAGE_HANDLER(AutomationMsg_NavigationStateChanged,
+ OnNavigationStateChanged)
+ IPC_MESSAGE_HANDLER(AutomationMsg_UpdateTargetUrl, OnUpdateTargetUrl)
+ IPC_MESSAGE_HANDLER(AutomationMsg_HandleAccelerator,
+ OnAcceleratorPressed)
+ IPC_MESSAGE_HANDLER(AutomationMsg_TabbedOut, OnTabbedOut)
+ IPC_MESSAGE_HANDLER(AutomationMsg_OpenURL, OnOpenURL)
+ IPC_MESSAGE_HANDLER(AutomationMsg_NavigationFailed, OnNavigationFailed)
+ IPC_MESSAGE_HANDLER(AutomationMsg_DidNavigate, OnDidNavigate)
+ IPC_MESSAGE_HANDLER(AutomationMsg_TabLoaded, OnLoad)
+ IPC_MESSAGE_HANDLER(AutomationMsg_ForwardMessageToExternalHost,
+ OnMessageFromChromeFrame)
+ IPC_MESSAGE_HANDLER(AutomationMsg_ForwardContextMenuToExternalHost,
+ OnHandleContextMenu)
+ IPC_MESSAGE_HANDLER(AutomationMsg_RequestStart, OnRequestStart)
+ IPC_MESSAGE_HANDLER(AutomationMsg_RequestRead, OnRequestRead)
+ IPC_MESSAGE_HANDLER(AutomationMsg_RequestEnd, OnRequestEnd)
+ IPC_MESSAGE_HANDLER(AutomationMsg_SetCookieAsync, OnSetCookieAsync)
+ IPC_MESSAGE_HANDLER(AutomationMsg_AttachExternalTab, OnAttachExternalTab)
+ IPC_END_MESSAGE_MAP()
+}
diff --git a/chrome_frame/chrome_frame_delegate.h b/chrome_frame/chrome_frame_delegate.h
new file mode 100644
index 0000000..a3302da
--- /dev/null
+++ b/chrome_frame/chrome_frame_delegate.h
@@ -0,0 +1,99 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_FRAME_DELEGATE_H_
+#define CHROME_FRAME_CHROME_FRAME_DELEGATE_H_
+
+#include "chrome/test/automation/automation_messages.h"
+#include "ipc/ipc_message.h"
+
+// A common interface supported by all the browser specific ChromeFrame
+// implementations.
+class ChromeFrameDelegate {
+ public:
+
+ typedef HWND WindowType;
+
+ virtual WindowType GetWindow() const = 0;
+ virtual void GetBounds(RECT* bounds) = 0;
+ virtual std::string GetDocumentUrl() = 0;
+ virtual void OnAutomationServerReady() = 0;
+ virtual void OnAutomationServerLaunchFailed(
+ AutomationLaunchResult reason, const std::string& server_version) = 0;
+ virtual void OnMessageReceived(const IPC::Message& msg) = 0;
+
+ // This remains in interface since we call it if Navigate()
+ // returns immediate error.
+ virtual void OnLoadFailed(int error_code, const std::string& url) = 0;
+
+ // Returns true if this instance is alive and well for processing automation
+ // messages.
+ virtual bool IsValid() const = 0;
+
+ protected:
+ ~ChromeFrameDelegate() {}
+};
+
+// Template specialization
+template <> struct RunnableMethodTraits<ChromeFrameDelegate> {
+ static void RetainCallee(ChromeFrameDelegate* obj) {
+ }
+
+ static void ReleaseCallee(ChromeFrameDelegate* obj) {
+ }
+};
+
+extern UINT kAutomationServerReady;
+extern UINT kMessageFromChromeFrame;
+
+class ChromeFrameDelegateImpl : public ChromeFrameDelegate {
+ public:
+ virtual WindowType GetWindow() { return NULL; }
+ virtual void GetBounds(RECT* bounds) {}
+ virtual std::string GetDocumentUrl() { return std::string(); }
+ virtual void OnAutomationServerReady() {}
+ virtual void OnAutomationServerLaunchFailed(
+ AutomationLaunchResult reason, const std::string& server_version) {}
+ virtual void OnLoadFailed(int error_code, const std::string& url) {}
+ virtual void OnMessageReceived(const IPC::Message& msg);
+ static bool IsTabMessage(const IPC::Message& message, int* tab_handle);
+
+ virtual bool IsValid() const {
+ return true;
+ }
+
+ protected:
+ // Protected methods to be overriden.
+ virtual void OnNavigationStateChanged(int tab_handle, int flags,
+ const IPC::NavigationInfo& nav_info) {}
+ virtual void OnUpdateTargetUrl(int tab_handle,
+ const std::wstring& new_target_url) {}
+ virtual void OnAcceleratorPressed(int tab_handle, const MSG& accel_message) {}
+ virtual void OnTabbedOut(int tab_handle, bool reverse) {}
+ virtual void OnOpenURL(int tab_handle, const GURL& url,
+ int open_disposition) {}
+ virtual void OnDidNavigate(int tab_handle,
+ const IPC::NavigationInfo& navigation_info) {}
+ virtual void OnNavigationFailed(int tab_handle, int error_code,
+ const GURL& gurl) {}
+ virtual void OnLoad(int tab_handle, const GURL& url) {}
+ virtual void OnMessageFromChromeFrame(int tab_handle,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& target) {}
+ virtual void OnHandleContextMenu(int tab_handle, HANDLE menu_handle,
+ int x_pos, int y_pos, int align_flags) {}
+ virtual void OnRequestStart(int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request) {}
+ virtual void OnRequestRead(int tab_handle, int request_id,
+ int bytes_to_read) {}
+ virtual void OnRequestEnd(int tab_handle, int request_id,
+ const URLRequestStatus& status) {}
+ virtual void OnSetCookieAsync(int tab_handle, const GURL& url,
+ const std::string& cookie) {}
+ virtual void OnAttachExternalTab(int tab_handle, intptr_t cookie,
+ int disposition) {}
+};
+
+#endif // CHROME_FRAME_CHROME_FRAME_DELEGATE_H_
diff --git a/chrome_frame/chrome_frame_histograms.cc b/chrome_frame/chrome_frame_histograms.cc
new file mode 100644
index 0000000..e1ea548
--- /dev/null
+++ b/chrome_frame/chrome_frame_histograms.cc
@@ -0,0 +1,81 @@
+// Copyright (c) 2009 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 "chrome_frame/chrome_frame_histograms.h"
+
+#include "base/histogram.h"
+#include "base/lazy_instance.h"
+#include "base/logging.h"
+#include "base/message_loop.h"
+#include "base/pickle.h"
+
+ // Initialize histogram statistics gathering system.
+base::LazyInstance<StatisticsRecorder>
+ g_statistics_recorder_(base::LINKER_INITIALIZED);
+
+ChromeFrameHistogramSnapshots::ChromeFrameHistogramSnapshots() {
+ // Ensure that an instance of the StatisticsRecorder object is created.
+ g_statistics_recorder_.Get();
+}
+
+ChromeFrameHistogramSnapshots::HistogramPickledList
+ ChromeFrameHistogramSnapshots::GatherAllHistograms() {
+
+ StatisticsRecorder::Histograms histograms;
+ StatisticsRecorder::GetHistograms(&histograms);
+
+ HistogramPickledList pickled_histograms;
+
+ for (StatisticsRecorder::Histograms::iterator it = histograms.begin();
+ histograms.end() != it;
+ it++) {
+ GatherHistogram(**it, &pickled_histograms);
+ }
+
+ return pickled_histograms;
+}
+
+void ChromeFrameHistogramSnapshots::GatherHistogram(
+ const Histogram& histogram,
+ HistogramPickledList* pickled_histograms) {
+
+ // Get up-to-date snapshot of sample stats.
+ Histogram::SampleSet snapshot;
+ histogram.SnapshotSample(&snapshot);
+ const std::string& histogram_name = histogram.histogram_name();
+
+ // Check if we already have a log of this histogram and if not create an
+ // empty set.
+ LoggedSampleMap::iterator it = logged_samples_.find(histogram_name);
+ Histogram::SampleSet* already_logged;
+ if (logged_samples_.end() == it) {
+ // Add new entry.
+ already_logged = &logged_samples_[histogram.histogram_name()];
+ already_logged->Resize(histogram); // Complete initialization.
+ } else {
+ already_logged = &(it->second);
+ // Deduct any stats we've already logged from our snapshot.
+ snapshot.Subtract(*already_logged);
+ }
+
+ // Snapshot now contains only a delta to what we've already_logged.
+ if (snapshot.TotalCount() > 0) {
+ GatherHistogramDelta(histogram, snapshot, pickled_histograms);
+ // Add new data into our running total.
+ already_logged->Add(snapshot);
+ }
+}
+
+void ChromeFrameHistogramSnapshots::GatherHistogramDelta(
+ const Histogram& histogram,
+ const Histogram::SampleSet& snapshot,
+ HistogramPickledList* pickled_histograms) {
+
+ DCHECK(0 != snapshot.TotalCount());
+ snapshot.CheckSize(histogram);
+
+ std::string histogram_info =
+ Histogram::SerializeHistogramInfo(histogram, snapshot);
+ pickled_histograms->push_back(histogram_info);
+}
diff --git a/chrome_frame/chrome_frame_histograms.h b/chrome_frame/chrome_frame_histograms.h
new file mode 100644
index 0000000..03aaab5
--- /dev/null
+++ b/chrome_frame/chrome_frame_histograms.h
@@ -0,0 +1,54 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_HISTOGRAM_SNAPSHOTS_H_
+#define CHROME_FRAME_HISTOGRAM_SNAPSHOTS_H_
+
+#include <map>
+#include <string>
+#include <vector>
+
+#include "base/basictypes.h"
+#include "base/histogram.h"
+#include "base/process.h"
+#include "base/scoped_ptr.h"
+#include "base/task.h"
+
+// This class gathers histogram data in the host browser process and
+// serializes the data into a vector of strings to be uploaded to the
+// Chrome browser process. It records the histogram data which has been
+// logged and only uploads the delta with the next log.
+// TODO(iyengar)
+// This class does not contain any ChromeFrame specific stuff. It should
+// be moved to base.
+class ChromeFrameHistogramSnapshots {
+ public:
+ // Maintain a map of histogram names to the sample stats we've sent.
+ typedef std::map<std::string, Histogram::SampleSet> LoggedSampleMap;
+ typedef std::vector<std::string> HistogramPickledList;
+
+ ChromeFrameHistogramSnapshots();
+ ~ChromeFrameHistogramSnapshots() {}
+
+ // Extract snapshot data and return it to be sent off to the Chrome browser
+ // process.
+ // Return only a delta to what we have already sent.
+ HistogramPickledList GatherAllHistograms();
+
+ private:
+ void GatherHistogram(const Histogram& histogram,
+ HistogramPickledList* histograms);
+
+ void GatherHistogramDelta(const Histogram& histogram,
+ const Histogram::SampleSet& snapshot,
+ HistogramPickledList* histograms);
+
+ // For histograms, record what we've already logged (as a sample for each
+ // histogram) so that we can send only the delta with the next log.
+ LoggedSampleMap logged_samples_;
+
+ DISALLOW_COPY_AND_ASSIGN(ChromeFrameHistogramSnapshots);
+};
+
+#endif // CHROME_RENDERER_HISTOGRAM_SNAPSHOTS_H_
diff --git a/chrome_frame/chrome_frame_npapi.cc b/chrome_frame/chrome_frame_npapi.cc
new file mode 100644
index 0000000..0c58cff
--- /dev/null
+++ b/chrome_frame/chrome_frame_npapi.cc
@@ -0,0 +1,1462 @@
+// Copyright (c) 2009 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 "chrome_frame/chrome_frame_npapi.h"
+
+#include "base/basictypes.h"
+#include "base/logging.h"
+#include "base/message_loop.h"
+#include "base/string_util.h"
+#include "base/win_util.h"
+#include "chrome/test/automation/tab_proxy.h"
+#include "chrome_frame/ff_privilege_check.h"
+#include "chrome_frame/utils.h"
+
+MessageLoop* ChromeFrameNPAPI::message_loop_ = NULL;
+int ChromeFrameNPAPI::instance_count_ = 0;
+
+static const char* kNpEventNames[] = {
+ "focus",
+ "blur",
+};
+
+NPClass ChromeFrameNPAPI::plugin_class_ = {
+ NP_CLASS_STRUCT_VERSION,
+ ChromeFrameNPAPI::AllocateObject,
+ ChromeFrameNPAPI::DeallocateObject,
+ ChromeFrameNPAPI::Invalidate,
+ ChromeFrameNPAPI::HasMethod,
+ ChromeFrameNPAPI::Invoke,
+ NULL, // invokeDefault
+ ChromeFrameNPAPI::HasProperty,
+ ChromeFrameNPAPI::GetProperty,
+ ChromeFrameNPAPI::SetProperty,
+ NULL, // remove property
+ NULL, // enumeration
+ NULL, // construct
+};
+
+NPIdentifier
+ ChromeFrameNPAPI::plugin_property_identifiers_[PLUGIN_PROPERTY_COUNT]
+ = {0};
+
+const NPUTF8* ChromeFrameNPAPI::plugin_property_identifier_names_[] = {
+ "version",
+ "src",
+ "onload",
+ "onloaderror",
+ "onmessage",
+ "readystate",
+ "onprivatemessage",
+ "usechromenetwork",
+};
+
+const NPUTF8* ChromeFrameNPAPI::plugin_method_identifier_names_[] = {
+ "postMessage",
+ "postPrivateMessage",
+};
+
+ChromeFrameNPAPI::PluginMethod ChromeFrameNPAPI::plugin_methods_[] = {
+ &ChromeFrameNPAPI::postMessage,
+ &ChromeFrameNPAPI::postPrivateMessage,
+};
+
+NPIdentifier
+ ChromeFrameNPAPI::plugin_method_identifiers_[arraysize(plugin_methods_)]
+ = {0};
+
+
+void ChromeFrameNPAPI::CompileAsserts() {
+ NOTREACHED(); // This function should never be invoked.
+
+ COMPILE_ASSERT(arraysize(plugin_method_identifier_names_) ==
+ arraysize(plugin_methods_),
+ you_must_add_both_plugin_method_and_name);
+
+ COMPILE_ASSERT(arraysize(plugin_property_identifier_names_) ==
+ arraysize(plugin_property_identifiers_),
+ you_must_add_both_plugin_property_and_name);
+}
+
+static const int kMaxBytesForPluginConsumption = 0x7FFFFFFF;
+
+static const char kPluginSrcAttribute[] = "src";
+static const char kPluginForceFullPageAttribute[] = "force_full_page";
+static const char kPluginOnloadAttribute[] = "onload";
+static const char kPluginOnErrorAttribute[] = "onloaderror";
+static const char kPluginOnMessageAttribute[] = "onmessage";
+static const char kPluginOnPrivateMessageAttribute[] = "onprivatemessage";
+// These properties can only be set in arguments at control instantiation.
+// When the privileged_mode property is provided and set to true, the control
+// will probe for whether its hosting document has the system principal, in
+// which case privileged mode will be enabled.
+static const char kPluginPrivilegedModeAttribute[] = "privileged_mode";
+// If privileged mode is enabled, the string value of this argument will
+// be appended to the chrome.exe command line.
+static const char kPluginChromeExtraArguments[] = "chrome_extra_arguments";
+// If privileged mode is enabled, the string value of this argument will
+// be used as the profile name for our chrome.exe instance.
+static const char kPluginChromeProfileName[] = "chrome_profile_name";
+// If chrome network stack is to be used
+static const char kPluginUseChromeNetwork[] = "usechromenetwork";
+
+
+NPError NPP_New(NPMIMEType plugin_type, NPP instance, uint16 mode, int16 argc,
+ char* argn[], char* argv[], NPSavedData* saved) {
+ if (instance == NULL)
+ return NPERR_INVALID_INSTANCE_ERROR;
+
+ ChromeFrameNPAPI::ChromeFrameNPObject* chrome_frame_npapi_obj =
+ reinterpret_cast<ChromeFrameNPAPI::ChromeFrameNPObject*>(
+ npapi::CreateObject(instance, ChromeFrameNPAPI::PluginClass()));
+ DCHECK(chrome_frame_npapi_obj != NULL);
+
+ ChromeFrameNPAPI* plugin_instance =
+ chrome_frame_npapi_obj->chrome_frame_plugin_instance;
+ DCHECK(plugin_instance != NULL);
+
+ // Note that we MUST set instance->pdata BEFORE calling Initialize. This is
+ // because Initialize can call back into the NPAPI host which will need the
+ // pdata field to be set.
+ chrome_frame_npapi_obj->chrome_frame_plugin_instance =
+ plugin_instance;
+ instance->pdata = chrome_frame_npapi_obj;
+
+ bool init = plugin_instance->Initialize(plugin_type, instance,
+ mode, argc, argn, argv);
+ DCHECK(init);
+
+ return NPERR_NO_ERROR;
+}
+
+NPError NPP_Destroy(NPP instance, NPSavedData** save) {
+ // Takes ownership and releases the object at the end of scope.
+ ScopedNpObject<ChromeFrameNPAPI::ChromeFrameNPObject> chrome_frame_npapi_obj(
+ reinterpret_cast<ChromeFrameNPAPI::ChromeFrameNPObject*>(
+ instance->pdata));
+
+ if (chrome_frame_npapi_obj.get()) {
+ ChromeFrameNPAPI* plugin_instance =
+ ChromeFrameNPAPI::ChromeFrameInstanceFromPluginInstance(instance);
+
+ plugin_instance->Uninitialize();
+ instance->pdata = NULL;
+ }
+
+ return NPERR_NO_ERROR;
+}
+
+NPError NPP_SetWindow(NPP instance, NPWindow* window_info) {
+ if (window_info == NULL) {
+ NOTREACHED();
+ return NPERR_GENERIC_ERROR;
+ }
+
+ ChromeFrameNPAPI* plugin_instance =
+ ChromeFrameNPAPI::ChromeFrameInstanceFromPluginInstance(instance);
+
+ if (plugin_instance == NULL) {
+ return NPERR_INVALID_INSTANCE_ERROR;
+ }
+
+ plugin_instance->SetWindow(window_info);
+ return NPERR_NO_ERROR;
+}
+
+NPError NPP_NewStream(NPP instance, NPMIMEType type, NPStream* stream,
+ NPBool seekable, uint16* stream_type) {
+ NPAPIUrlRequest* url_request = ChromeFrameNPAPI::ValidateRequest(
+ instance, stream->notifyData);
+ if (url_request) {
+ if (!url_request->OnStreamCreated(type, stream))
+ return NPERR_GENERIC_ERROR;
+ }
+
+ // We need to return the requested stream mode if we are returning a success
+ // code. If we don't do this it causes Opera to blow up.
+ *stream_type = NP_NORMAL;
+ return NPERR_NO_ERROR;
+}
+
+NPError NPP_DestroyStream(NPP instance, NPStream* stream, NPReason reason) {
+ NPAPIUrlRequest* url_request = ChromeFrameNPAPI::ValidateRequest(
+ instance, stream->notifyData);
+ if (url_request) {
+ url_request->OnStreamDestroyed(reason);
+ }
+
+ return NPERR_NO_ERROR;
+}
+
+NPError NPP_GetValue(NPP instance, NPPVariable variable, void* value) {
+ if (variable == NPPVpluginScriptableNPObject) {
+ void** plugin = reinterpret_cast<void**>(value);
+ ChromeFrameNPAPI::ChromeFrameNPObject* chrome_frame_npapi_obj =
+ reinterpret_cast<ChromeFrameNPAPI::ChromeFrameNPObject*>(
+ instance->pdata);
+ // Return value is expected to be retained
+ npapi::RetainObject(reinterpret_cast<NPObject*>(chrome_frame_npapi_obj));
+ *plugin = chrome_frame_npapi_obj;
+ return NPERR_NO_ERROR;
+ }
+ return NPERR_GENERIC_ERROR;
+}
+
+NPError NPP_SetValue(NPP instance, NPNVariable variable, void* value) {
+ return NPERR_GENERIC_ERROR;
+}
+
+int32 NPP_WriteReady(NPP instance, NPStream* stream) {
+ NPAPIUrlRequest* url_request = ChromeFrameNPAPI::ValidateRequest(
+ instance, stream->notifyData);
+ if (url_request) {
+ return url_request->OnWriteReady();
+ }
+
+ return kMaxBytesForPluginConsumption;
+}
+
+int32 NPP_Write(NPP instance, NPStream* stream, int32 offset, int32 len,
+ void* buffer) {
+ NPAPIUrlRequest* url_request = ChromeFrameNPAPI::ValidateRequest(
+ instance, stream->notifyData);
+ if (url_request) {
+ return url_request->OnWrite(buffer, len);
+ }
+
+ return len;
+}
+
+void NPP_URLNotify(NPP instance, const char* url, NPReason reason,
+ void* notifyData) {
+ ChromeFrameNPAPI* plugin_instance =
+ ChromeFrameNPAPI::ChromeFrameInstanceFromPluginInstance(instance);
+ if (plugin_instance) {
+ plugin_instance->UrlNotify(url, reason, notifyData);
+ }
+}
+
+void NPP_Print(NPP instance, NPPrint* print_info) {
+ ChromeFrameNPAPI* plugin_instance =
+ ChromeFrameNPAPI::ChromeFrameInstanceFromPluginInstance(instance);
+
+ if (plugin_instance == NULL) {
+ NOTREACHED();
+ return;
+ }
+
+ plugin_instance->Print(print_info);
+}
+
+// ChromeFrameNPAPI member defines.
+
+// TODO(tommi): remove ignore_setfocus_ since that's not how focus is
+// handled anymore.
+
+ChromeFrameNPAPI::ChromeFrameNPAPI()
+ : instance_(NULL),
+ mode_(NP_EMBED),
+ force_full_page_plugin_(false),
+ ready_state_(READYSTATE_LOADING),
+ enabled_popups_(false) {
+}
+
+ChromeFrameNPAPI::~ChromeFrameNPAPI() {
+ if (IsWindow()) {
+ if (!UnsubclassWindow()) {
+ // TODO(tommi): Figure out why this can sometimes happen in the
+ // WidgetModeFF_Resize unittest.
+ DLOG(ERROR) << "Couldn't unsubclass safely!";
+ UnsubclassWindow(TRUE);
+ }
+ }
+ m_hWnd = NULL;
+
+ instance_count_--;
+ if (instance_count_ <= 0) {
+ delete message_loop_;
+ message_loop_ = NULL;
+ }
+
+ Uninitialize();
+}
+
+std::string GetLocation(NPP instance, NPObject* window) {
+ if (!window) {
+ // Can fail if the browser is closing (seen in Opera).
+ return "";
+ }
+
+ std::string result;
+ ScopedNpVariant href;
+ ScopedNpVariant location;
+
+ bool ok = npapi::GetProperty(instance, window,
+ npapi::GetStringIdentifier("location"), &location);
+ DCHECK(ok);
+ DCHECK(location.type == NPVariantType_Object);
+
+ if (ok) {
+ ok = npapi::GetProperty(instance,
+ location.value.objectValue,
+ npapi::GetStringIdentifier("href"),
+ &href);
+ DCHECK(ok);
+ DCHECK(href.type == NPVariantType_String);
+ if (ok) {
+ result.assign(href.value.stringValue.UTF8Characters,
+ href.value.stringValue.UTF8Length);
+ }
+ }
+
+ return result;
+}
+
+std::string ChromeFrameNPAPI::GetLocation() {
+ // Note that GetWindowObject() will cache the window object here.
+ return ::GetLocation(instance_, GetWindowObject());
+}
+
+bool ChromeFrameNPAPI::Initialize(NPMIMEType mime_type, NPP instance,
+ uint16 mode, int16 argc, char* argn[],
+ char* argv[]) {
+ if (!Base::Initialize())
+ return false;
+
+ instance_ = instance;
+ mime_type_ = mime_type;
+ mode_ = mode;
+ document_url_ = GetLocation();
+
+ if (instance_count_ == 0) {
+ DCHECK(message_loop_ == NULL);
+ message_loop_ = new MessageLoop();
+ }
+
+ instance_count_++;
+
+ // Create our prefs service wrapper here.
+ DCHECK(!pref_service_.get());
+ pref_service_ = CreatePrefService();
+ if (!pref_service_.get()) {
+ NOTREACHED() << "new NpProxyService";
+ return false;
+ }
+
+ // Temporary variables for privileged only parameters
+ const char* onprivatemessage_arg = NULL;
+ const char* chrome_extra_arguments_arg = NULL;
+ const char* chrome_profile_name_arg = NULL;
+ bool chrome_network_arg_set = false;
+ bool chrome_network_arg = false;
+ bool wants_privileged = false;
+
+ for (int i = 0; i < argc; ++i) {
+ if (LowerCaseEqualsASCII(argn[i], kPluginSrcAttribute)) {
+ src_ = ResolveURL(GetDocumentUrl(), argv[i]);
+ } else if (LowerCaseEqualsASCII(argn[i], kPluginForceFullPageAttribute)) {
+ force_full_page_plugin_ = atoi(argv[i]) ? true : false;
+ } else if (LowerCaseEqualsASCII(argn[i], kPluginOnErrorAttribute)) {
+ onerror_handler_ = JavascriptToNPObject(argv[i]);
+ } else if (LowerCaseEqualsASCII(argn[i], kPluginOnMessageAttribute)) {
+ onmessage_handler_ = JavascriptToNPObject(argv[i]);
+ } else if (LowerCaseEqualsASCII(argn[i],
+ kPluginPrivilegedModeAttribute)) {
+ // Test for the FireFox privileged mode if the user requests it
+ // in initialization parameters.
+ wants_privileged = atoi(argv[i]) ? true : false;
+ } else if (LowerCaseEqualsASCII(argn[i],
+ kPluginOnPrivateMessageAttribute)) {
+ onprivatemessage_arg = argv[i];
+ } else if (LowerCaseEqualsASCII(argn[i], kPluginChromeExtraArguments)) {
+ chrome_extra_arguments_arg = argv[i];
+ } else if (LowerCaseEqualsASCII(argn[i], kPluginChromeProfileName)) {
+ chrome_profile_name_arg = argv[i];
+ } else if (LowerCaseEqualsASCII(argn[i], kPluginUseChromeNetwork)) {
+ chrome_network_arg_set = true;
+ chrome_network_arg = atoi(argv[i]) ? true : false;
+ }
+ }
+
+ // Is the privileged mode requested?
+ if (wants_privileged) {
+ is_privileged_ = IsFireFoxPrivilegedInvocation(instance);
+ if (!is_privileged_) {
+ DLOG(WARNING) << "Privileged mode requested in non-privileged context";
+ }
+ }
+
+ std::wstring extra_arguments;
+ std::wstring profile_name(GetHostProcessName(false));
+ if (is_privileged_) {
+ // Process any privileged mode-only arguments we were handed.
+ if (onprivatemessage_arg)
+ onprivatemessage_handler_ = JavascriptToNPObject(onprivatemessage_arg);
+
+ if (chrome_extra_arguments_arg)
+ extra_arguments = UTF8ToWide(chrome_extra_arguments_arg);
+
+ if (chrome_profile_name_arg)
+ profile_name = UTF8ToWide(chrome_profile_name_arg);
+
+ if (chrome_network_arg_set)
+ automation_client_->set_use_chrome_network(chrome_network_arg);
+
+ }
+
+ // TODO(joshia): Initialize navigation here and send proxy config as
+ // part of LaunchSettings
+ /*
+ if (!src_.empty())
+ automation_client_->InitiateNavigation(src_);
+
+ std::string proxy_settings;
+ bool has_prefs = pref_service_->Initialize(instance_,
+ automation_client_.get());
+ if (has_prefs && pref_service_->GetProxyValueJSONString(&proxy_settings)) {
+ automation_client_->SetProxySettings(proxy_settings);
+ }
+ */
+
+ // We can't call SubscribeToFocusEvents here since
+ // when Initialize gets called, Opera is in a state where
+ // it can't handle calls back and the thread will hang.
+ // Instead, we call SubscribeToFocusEvents when we initialize
+ // our plugin window.
+
+ // TODO(stoyan): Ask host for specific interface whether to honor
+ // host's in-private mode.
+ return InitializeAutomation(profile_name, extra_arguments,
+ GetBrowserIncognitoMode());
+}
+
+void ChromeFrameNPAPI::Uninitialize() {
+ // Don't call SetReadyState as it will end up calling FireEvent.
+ // We are in the context of NPP_DESTROY.
+ ready_state_ = READYSTATE_UNINITIALIZED;
+
+ UnsubscribeFromFocusEvents();
+
+ if (pref_service_) {
+ pref_service_->UnInitialize();
+ pref_service_ = NULL;
+ }
+
+ window_object_.Free();
+ onerror_handler_.Free();
+ onmessage_handler_.Free();
+ onprivatemessage_handler_.Free();
+
+ Base::Uninitialize();
+}
+
+void ChromeFrameNPAPI::OnFinalMessage(HWND window) {
+ // The automation server should be gone by now.
+ Uninitialize();
+}
+
+void ChromeFrameNPAPI::SubscribeToFocusEvents() {
+ DCHECK(focus_listener_.get() == NULL);
+
+ focus_listener_ = new DomEventListener(this);
+ if (!focus_listener_->Subscribe(instance_, kNpEventNames,
+ arraysize(kNpEventNames))) {
+ focus_listener_ = NULL;
+ focus_listener_ = new NPObjectEventListener(this);
+ if (!focus_listener_->Subscribe(instance_, kNpEventNames,
+ arraysize(kNpEventNames))) {
+ DLOG(ERROR) << "Failed to subscribe to focus events";
+ focus_listener_ = NULL;
+ }
+ }
+}
+
+void ChromeFrameNPAPI::UnsubscribeFromFocusEvents() {
+ if (!focus_listener_.get())
+ return;
+
+ bool ret = focus_listener_->Unsubscribe(instance_, kNpEventNames,
+ arraysize(kNpEventNames));
+ DLOG_IF(WARNING, !ret) << "focus_listener_->Unsubscribe failed";
+ focus_listener_ = NULL;
+}
+
+bool ChromeFrameNPAPI::SetWindow(NPWindow* window_info) {
+ if (!window_info || !automation_client_.get()) {
+ NOTREACHED();
+ return false;
+ }
+
+ HWND window = reinterpret_cast<HWND>(window_info->window);
+ if (!::IsWindow(window)) {
+ // No window created yet. Ignore this call.
+ return false;
+ }
+
+ if (IsWindow()) {
+ // We've already subclassed, make sure that SetWindow doesn't get called
+ // with an HWND other than the one we subclassed during our lifetime.
+ DCHECK(window == m_hWnd);
+ return true;
+ }
+
+ automation_client_->SetParentWindow(window);
+
+ SubscribeToFocusEvents();
+
+ if (force_full_page_plugin_) {
+ // By default full page mode is only enabled when the plugin is loaded off
+ // a separate file, i.e. it is the primary content in the window. Even if
+ // we specify the width/height attributes for the plugin as 100% each, FF
+ // instantiates the plugin passing in a width/height of 100px each. To
+ // workaround this we resize the plugin window passed in by FF to the size
+ // of its parent.
+ HWND plugin_parent_window = ::GetParent(window);
+ RECT plugin_parent_rect = {0};
+ ::GetClientRect(plugin_parent_window, &plugin_parent_rect);
+ ::SetWindowPos(window, NULL, plugin_parent_rect.left,
+ plugin_parent_rect.top,
+ plugin_parent_rect.right - plugin_parent_rect.left,
+ plugin_parent_rect.bottom - plugin_parent_rect.top, 0);
+ }
+
+ // Subclass the browser's plugin window here.
+ if (SubclassWindow(window)) {
+ DWORD new_style_flags = WS_CLIPCHILDREN;
+ ModifyStyle(0, new_style_flags, 0);
+
+ if (ready_state_ < READYSTATE_INTERACTIVE) {
+ SetReadyState(READYSTATE_INTERACTIVE);
+ }
+ }
+
+ return true;
+}
+
+void ChromeFrameNPAPI::Print(NPPrint* print_info) {
+ if (!print_info) {
+ NOTREACHED();
+ return;
+ }
+
+ // We dont support full tab mode yet.
+ if (print_info->mode != NP_EMBED) {
+ NOTREACHED();
+ return;
+ }
+
+ NPWindow window = print_info->print.embedPrint.window;
+
+ RECT print_bounds = {0};
+ print_bounds.left = window.x;
+ print_bounds.top = window.y;
+ print_bounds.right = window.x + window.width;
+ print_bounds.bottom = window.x + window.height;
+
+ automation_client_->Print(
+ reinterpret_cast<HDC>(print_info->print.embedPrint.platformPrint),
+ print_bounds);
+}
+
+void ChromeFrameNPAPI::UrlNotify(const char* url, NPReason reason,
+ void* notify_data) {
+ if (enabled_popups_) {
+ // We have opened the URL so tell the browser to restore popup settings
+ enabled_popups_ = false;
+ npapi::PopPopupsEnabledState(instance_);
+ }
+
+ // It is now safe to release the additional reference on the request
+ NPAPIUrlRequest* request = RequestFromNotifyData(notify_data);
+ if (request) {
+ request->Stop();
+ request->Release();
+ }
+}
+
+void ChromeFrameNPAPI::OnAcceleratorPressed(int tab_handle,
+ const MSG& accel_message) {
+ DLOG(INFO) << __FUNCTION__ << " msg:"
+ << StringPrintf("0x%04X", accel_message.message) << " key:"
+ << accel_message.wParam;
+
+ // The host browser does call TranslateMessage on messages like WM_KEYDOWN
+ // WM_KEYUP, etc, which will result in messages like WM_CHAR, WM_SYSCHAR, etc
+ // being posted to the message queue. We don't post these messages here to
+ // avoid these messages from getting handled twice.
+ if (accel_message.message != WM_CHAR &&
+ accel_message.message != WM_DEADCHAR &&
+ accel_message.message != WM_SYSCHAR &&
+ accel_message.message != WM_SYSDEADCHAR) {
+ // A very primitive way to handle keystrokes.
+ // TODO(tommi): When we've implemented a way for chrome to
+ // know when keystrokes are handled (deterministically) on that side,
+ // then this function should get called and not otherwise.
+ ::PostMessage(::GetParent(m_hWnd), accel_message.message,
+ accel_message.wParam, accel_message.lParam);
+ }
+
+ if (automation_client_.get()) {
+ TabProxy* tab = automation_client_->tab();
+ if (tab) {
+ tab->ProcessUnhandledAccelerator(accel_message);
+ }
+ }
+}
+
+void ChromeFrameNPAPI::OnTabbedOut(int tab_handle, bool reverse) {
+ DLOG(INFO) << __FUNCTION__;
+
+ ignore_setfocus_ = true;
+ HWND parent = ::GetParent(m_hWnd);
+ ::SetFocus(parent);
+
+ INPUT input = {0};
+ input.type = INPUT_KEYBOARD;
+ input.ki.wVk = VK_TAB;
+ SendInput(1, &input, sizeof(input));
+ input.ki.dwFlags = KEYEVENTF_KEYUP;
+ SendInput(1, &input, sizeof(input));
+
+ ignore_setfocus_ = false;
+}
+
+void ChromeFrameNPAPI::OnOpenURL(int tab_handle,
+ const GURL& url, int open_disposition) {
+ std::string target;
+ switch (open_disposition) {
+ case NEW_FOREGROUND_TAB:
+ target = "_blank";
+ break;
+ case NEW_BACKGROUND_TAB:
+ target = "_blank";
+ break;
+ case NEW_WINDOW:
+ target = "_new";
+ break;
+ default:
+ break;
+ }
+
+ // Tell the browser to temporarily allow popups
+ enabled_popups_ = true;
+ npapi::PushPopupsEnabledState(instance_, TRUE);
+ npapi::GetURLNotify(instance_, url.spec().c_str(), target.c_str(), NULL);
+}
+
+void ChromeFrameNPAPI::OnRequestStart(int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request) {
+ scoped_refptr<NPAPIUrlRequest> new_request(new NPAPIUrlRequest(instance_));
+ DCHECK(new_request);
+ if (new_request->Initialize(automation_client_.get(), tab_handle,
+ request_id, request.url, request.method,
+ request.referrer, request.extra_request_headers,
+ request.upload_data.get(), true)) {
+ if (new_request->Start()) {
+ // Keep additional reference on request for NPSTREAM
+ // This will be released in NPP_UrlNotify
+ new_request->AddRef();
+ }
+ }
+}
+
+void ChromeFrameNPAPI::OnRequestRead(int tab_handle, int request_id,
+ int bytes_to_read) {
+ automation_client_->ReadRequest(request_id, bytes_to_read);
+}
+
+void ChromeFrameNPAPI::OnRequestEnd(int tab_handle, int request_id,
+ const URLRequestStatus& status) {
+ automation_client_->RemoveRequest(request_id, status.status(), true);
+}
+
+void ChromeFrameNPAPI::OnSetCookieAsync(int tab_handle, const GURL& url,
+ const std::string& cookie) {
+ // Use the newer NPAPI way if available
+ if (npapi::VersionMinor() >= NPVERS_HAS_URL_AND_AUTH_INFO) {
+ npapi::SetValueForURL(instance_, NPNURLVCookie, url.spec().c_str(),
+ cookie.c_str(), cookie.length());
+ } else if (url == GURL(document_url_)) {
+ std::string script = "javascript:document.cookie=";
+ script.append(cookie);
+ script.append(1, ';');
+ ExecuteScript(script, NULL);
+ } else {
+ // Third party cookie, use nsICookieService to set the cookie.
+ NOTREACHED();
+ }
+}
+
+bool ChromeFrameNPAPI::HasMethod(NPObject* obj, NPIdentifier name) {
+ for (int i = 0; i < arraysize(plugin_methods_); ++i) {
+ if (name == plugin_method_identifiers_[i])
+ return true;
+ }
+
+ return false;
+}
+
+bool ChromeFrameNPAPI::Invoke(NPObject* header, NPIdentifier name,
+ const NPVariant* args, uint32_t arg_count,
+ NPVariant* result) {
+ ChromeFrameNPAPI* plugin_instance = ChromeFrameInstanceFromNPObject(header);
+ if (!plugin_instance && (plugin_instance->automation_client_.get()))
+ return false;
+
+ bool success = false;
+ for (int i = 0; i < arraysize(plugin_methods_); ++i) {
+ if (name == plugin_method_identifiers_[i]) {
+ PluginMethod method = plugin_methods_[i];
+ success = (plugin_instance->*method)(header, args, arg_count, result);
+ break;
+ }
+ }
+
+ return success;
+}
+
+void ChromeFrameNPAPI::InitializeIdentifiers() {
+ npapi::GetStringIdentifiers(plugin_method_identifier_names_,
+ arraysize(plugin_methods_),
+ plugin_method_identifiers_);
+
+ npapi::GetStringIdentifiers(plugin_property_identifier_names_,
+ PLUGIN_PROPERTY_COUNT,
+ plugin_property_identifiers_);
+}
+
+NPObject* ChromeFrameNPAPI::AllocateObject(NPP instance, NPClass* class_name) {
+ static bool identifiers_initialized = false;
+
+ ChromeFrameNPObject* plugin_object = new ChromeFrameNPObject();
+ DCHECK(plugin_object != NULL);
+
+ plugin_object->chrome_frame_plugin_instance = new ChromeFrameNPAPI();
+ DCHECK(plugin_object->chrome_frame_plugin_instance != NULL);
+
+ plugin_object->npp = NULL;
+
+ COMPILE_ASSERT(arraysize(plugin_method_identifiers_) ==
+ arraysize(plugin_method_identifier_names_),
+ method_count_mismatch);
+
+ COMPILE_ASSERT(arraysize(plugin_method_identifiers_) ==
+ arraysize(plugin_methods_),
+ method_count_mismatch);
+
+ if (!identifiers_initialized) {
+ InitializeIdentifiers();
+ identifiers_initialized = true;
+ }
+
+ return reinterpret_cast<NPObject*>(plugin_object);
+}
+
+void ChromeFrameNPAPI::DeallocateObject(NPObject* header) {
+ ChromeFrameNPObject* plugin_object =
+ reinterpret_cast<ChromeFrameNPObject*>(header);
+ DCHECK(plugin_object != NULL);
+
+ if (plugin_object) {
+ delete plugin_object->chrome_frame_plugin_instance;
+ delete plugin_object;
+ }
+}
+
+void ChromeFrameNPAPI::Invalidate(NPObject* header) {
+ DCHECK(header);
+ ChromeFrameNPObject* plugin_object =
+ reinterpret_cast<ChromeFrameNPObject*>(header);
+ if (plugin_object) {
+ DCHECK(plugin_object->chrome_frame_plugin_instance);
+ plugin_object->chrome_frame_plugin_instance->Uninitialize();
+ }
+}
+
+ChromeFrameNPAPI* ChromeFrameNPAPI::ChromeFrameInstanceFromPluginInstance(
+ NPP instance) {
+ if ((instance == NULL) || (instance->pdata == NULL)) {
+ NOTREACHED();
+ return NULL;
+ }
+
+ return ChromeFrameInstanceFromNPObject(instance->pdata);
+}
+
+ChromeFrameNPAPI* ChromeFrameNPAPI::ChromeFrameInstanceFromNPObject(
+ void* object) {
+ ChromeFrameNPObject* plugin_object =
+ reinterpret_cast<ChromeFrameNPObject*>(object);
+ if (!plugin_object) {
+ NOTREACHED();
+ return NULL;
+ }
+
+ DCHECK(plugin_object->chrome_frame_plugin_instance);
+ return plugin_object->chrome_frame_plugin_instance;
+}
+
+bool ChromeFrameNPAPI::HasProperty(NPObject* obj, NPIdentifier name) {
+ for (int i = 0; i < PLUGIN_PROPERTY_COUNT; ++i) {
+ if (name == plugin_property_identifiers_[i])
+ return true;
+ }
+ return false;
+}
+
+bool ChromeFrameNPAPI::GetProperty(NPIdentifier name,
+ NPVariant* variant) {
+ if (name == plugin_property_identifiers_[PLUGIN_PROPERTY_ONERROR]) {
+ if (onerror_handler_) {
+ variant->type = NPVariantType_Object;
+ variant->value.objectValue = onerror_handler_.Copy();
+ return true;
+ }
+ } else if (name == plugin_property_identifiers_[PLUGIN_PROPERTY_ONMESSAGE]) {
+ if (onmessage_handler_) {
+ variant->type = NPVariantType_Object;
+ variant->value.objectValue = onmessage_handler_.Copy();
+ return true;
+ }
+ } else if (name ==
+ plugin_property_identifiers_[PLUGIN_PROPERTY_ONPRIVATEMESSAGE]) {
+ if (!is_privileged_) {
+ DLOG(WARNING) << "Attempt to read onprivatemessage property while not "
+ "privileged";
+ } else {
+ if (onprivatemessage_handler_) {
+ variant->type = NPVariantType_Object;
+ variant->value.objectValue =
+ onprivatemessage_handler_.Copy();
+ return true;
+ }
+ }
+ } else if (name == plugin_property_identifiers_[PLUGIN_PROPERTY_SRC]) {
+ AllocateStringVariant(src_, variant);
+ return true;
+ } else if (name == plugin_property_identifiers_[PLUGIN_PROPERTY_VERSION]) {
+ const std::wstring version =
+ automation_client_->GetVersion();
+ AllocateStringVariant(WideToUTF8(version), variant);
+ return true;
+ } else if (name == plugin_property_identifiers_[PLUGIN_PROPERTY_READYSTATE]) {
+ INT32_TO_NPVARIANT(ready_state_, *variant);
+ return true;
+ } else if (name ==
+ plugin_property_identifiers_[PLUGIN_PROPERTY_USECHROMENETWORK]) {
+ BOOLEAN_TO_NPVARIANT(automation_client_->use_chrome_network(), *variant);
+ return true;
+ }
+
+ return false;
+}
+
+bool ChromeFrameNPAPI::GetProperty(NPObject* object, NPIdentifier name,
+ NPVariant* variant) {
+ if (!object || !variant) {
+ NOTREACHED();
+ return false;
+ }
+
+ ChromeFrameNPAPI* plugin_instance = ChromeFrameInstanceFromNPObject(object);
+ if (!plugin_instance) {
+ NOTREACHED();
+ return false;
+ }
+
+ return plugin_instance->GetProperty(name, variant);
+}
+
+bool ChromeFrameNPAPI::SetProperty(NPIdentifier name,
+ const NPVariant* variant) {
+ if (NPVARIANT_IS_OBJECT(*variant)) {
+ if (name == plugin_property_identifiers_[PLUGIN_PROPERTY_ONERROR]) {
+ onerror_handler_.Free();
+ onerror_handler_ = variant->value.objectValue;
+ return true;
+ } else if (
+ name == plugin_property_identifiers_[PLUGIN_PROPERTY_ONMESSAGE]) {
+ onmessage_handler_.Free();
+ onmessage_handler_ = variant->value.objectValue;
+ return true;
+ } else if (name ==
+ plugin_property_identifiers_[PLUGIN_PROPERTY_ONPRIVATEMESSAGE]) {
+ if (!is_privileged_) {
+ DLOG(WARNING) << "Attempt to set onprivatemessage while not privileged";
+ } else {
+ onprivatemessage_handler_.Free();
+ onprivatemessage_handler_ = variant->value.objectValue;
+ return true;
+ }
+ }
+ } else if (NPVARIANT_IS_STRING(*variant) || NPVARIANT_IS_NULL(*variant)) {
+ if (name == plugin_property_identifiers_[PLUGIN_PROPERTY_SRC]) {
+ return NavigateToURL(variant, 1, NULL);
+ }
+ } else if (NPVARIANT_IS_BOOLEAN(*variant)) {
+ if (name ==
+ plugin_property_identifiers_[PLUGIN_PROPERTY_USECHROMENETWORK]) {
+ automation_client_->set_use_chrome_network(
+ NPVARIANT_TO_BOOLEAN(*variant));
+ }
+ }
+
+ return false;
+}
+
+bool ChromeFrameNPAPI::SetProperty(NPObject* object, NPIdentifier name,
+ const NPVariant* variant) {
+ if (!object || !variant) {
+ DLOG(ERROR) << "Cannot set property: " << npapi::StringFromIdentifier(name);
+ return false;
+ }
+
+ ChromeFrameNPAPI* plugin_instance = ChromeFrameInstanceFromNPObject(object);
+ if (!plugin_instance) {
+ NOTREACHED();
+ return false;
+ }
+
+ return plugin_instance->SetProperty(name, variant);
+}
+
+void ChromeFrameNPAPI::OnFocus() {
+ DLOG(INFO) << __FUNCTION__;
+ PostMessage(WM_SETFOCUS, 0, 0);
+}
+
+void ChromeFrameNPAPI::OnEvent(const char* event_name) {
+ DCHECK(event_name);
+ DLOG(INFO) << event_name;
+
+ if (lstrcmpiA(event_name, "focus") == 0) {
+ OnFocus();
+ } else if (lstrcmpiA(event_name, "blur") == 0) {
+ OnBlur();
+ } else {
+ NOTREACHED() << event_name;
+ }
+}
+
+LRESULT CALLBACK ChromeFrameNPAPI::DropKillFocusHook(int code, WPARAM wparam,
+ LPARAM lparam) {
+ LRESULT ret = 0;
+ CWPSTRUCT* wp = reinterpret_cast<CWPSTRUCT*>(lparam);
+ if ((code < 0) || (wp->message != WM_KILLFOCUS))
+ ret = ::CallNextHookEx(NULL, code, wparam, lparam);
+
+ return ret;
+}
+
+LRESULT ChromeFrameNPAPI::OnSetFocus(UINT message, WPARAM wparam,
+ LPARAM lparam, BOOL& handled) { // NO_LINT
+ // Opera has a WH_CALLWNDPROC hook that handles WM_KILLFOCUS and
+ // prevents us from setting the focus to the tab.
+ // To work around that, we set a temporary hook here that does nothing
+ // (not even call other hooks) when it sees WM_KILLFOCUS.
+ HHOOK hook = NULL;
+ hook = ::SetWindowsHookEx(WH_CALLWNDPROC, DropKillFocusHook, NULL,
+ ::GetCurrentThreadId());
+ // Since we chain message maps, make sure we are not calling base class
+ // twice for WM_SETFOCUS.
+ BOOL handled_by_base = TRUE;
+ LRESULT ret = Base::OnSetFocus(message, wparam, lparam, handled_by_base);
+ if (hook)
+ ::UnhookWindowsHookEx(hook);
+
+ return ret;
+}
+
+void ChromeFrameNPAPI::OnBlur() {
+ DLOG(INFO) << __FUNCTION__;
+}
+
+void ChromeFrameNPAPI::OnLoad(int, const GURL& gurl) {
+ DLOG(INFO) << "Firing onload";
+ FireEvent("load", gurl.spec());
+}
+
+void ChromeFrameNPAPI::OnLoadFailed(int error_code, const std::string& url) {
+ FireEvent("loaderror", url);
+
+ ScopedNpVariant result;
+ InvokeDefault(onerror_handler_, url, &result);
+}
+
+void ChromeFrameNPAPI::OnMessageFromChromeFrame(int tab_handle,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& target) {
+ bool private_message = false;
+ if (target.compare("*") != 0) {
+ if (is_privileged_) {
+ private_message = true;
+ } else {
+ if (!HaveSameOrigin(target, document_url_)) {
+ DLOG(WARNING) << "Dropping posted message since target doesn't match "
+ "the current document's origin. target=" << target;
+ return;
+ }
+ }
+ }
+
+ // Create a MessageEvent object that contains the message and origin
+ // as well as supporting other MessageEvent (see the HTML5 spec) properties.
+ // Then call the onmessage handler.
+ ScopedNpObject<NPObject> event;
+ bool ok = CreateMessageEvent(false, true, message, origin, event.Receive());
+ if (ok) {
+ // Don't call FireEvent here (or we'll have an event wrapped by an event).
+ DispatchEvent(event);
+
+ ScopedNpVariant result;
+ NPVariant params[2];
+ OBJECT_TO_NPVARIANT(event, params[0]);
+ bool invoke = false;
+ if (private_message) {
+ DCHECK(is_privileged_);
+ STRINGN_TO_NPVARIANT(target.c_str(), target.length(), params[1]);
+ invoke = InvokeDefault(onprivatemessage_handler_,
+ arraysize(params),
+ params,
+ &result);
+ } else {
+ invoke = InvokeDefault(onmessage_handler_, params[0], &result);
+ }
+ DLOG_IF(WARNING, !invoke) << "InvokeDefault failed";
+ } else {
+ NOTREACHED() << "CreateMessageEvent";
+ }
+}
+
+void ChromeFrameNPAPI::OnAutomationServerReady() {
+ Base::OnAutomationServerReady();
+
+ std::string proxy_settings;
+ bool has_prefs = pref_service_->Initialize(instance_,
+ automation_client_.get());
+ if (has_prefs && pref_service_->GetProxyValueJSONString(&proxy_settings)) {
+ automation_client_->SetProxySettings(proxy_settings);
+ }
+
+ if (!src_.empty()) {
+ if (!automation_client_->InitiateNavigation(src_)) {
+ DLOG(ERROR) << "Failed to navigate to: " << src_;
+ src_.clear();
+ }
+ }
+
+ SetReadyState(READYSTATE_COMPLETE);
+}
+
+void ChromeFrameNPAPI::OnAutomationServerLaunchFailed(
+ AutomationLaunchResult reason, const std::string& server_version) {
+ SetReadyState(READYSTATE_UNINITIALIZED);
+
+ if (reason == AUTOMATION_VERSION_MISMATCH) {
+ DisplayVersionMismatchWarning(m_hWnd, server_version);
+ }
+}
+
+bool ChromeFrameNPAPI::InvokeDefault(NPObject* object,
+ unsigned param_count,
+ const NPVariant* params,
+ NPVariant* result) {
+ if (!object)
+ return false;
+
+ bool ret = npapi::InvokeDefault(instance_, object, params, param_count,
+ result);
+ // InvokeDefault can return false in FF even though we do see the call
+ // go through. It's not clear to me what the circumstances are, so
+ // we log it as a warning while tracking it down.
+ DLOG_IF(WARNING, !ret) << "npapi::InvokeDefault failed";
+ return ret;
+}
+
+bool ChromeFrameNPAPI::InvokeDefault(NPObject* object, const std::string& param,
+ NPVariant* result) {
+ NPVariant arg;
+ STRINGN_TO_NPVARIANT(param.c_str(), param.length(), arg);
+ return InvokeDefault(object, arg, result);
+}
+
+bool ChromeFrameNPAPI::InvokeDefault(NPObject* object, const NPVariant& param,
+ NPVariant* result) {
+ return InvokeDefault(object, 1, &param, result);
+}
+
+bool ChromeFrameNPAPI::CreateEvent(const std::string& type, bool bubbles,
+ bool cancelable, NPObject** basic_event) {
+ DCHECK(basic_event);
+ NPObject* window = GetWindowObject();
+ if (!window) {
+ // Can fail if the browser is closing (seen in Opera).
+ return false;
+ }
+
+ const char* identifier_names[] = {
+ "document",
+ "createEvent",
+ "initEvent",
+ };
+
+ NPIdentifier identifiers[arraysize(identifier_names)];
+ npapi::GetStringIdentifiers(identifier_names, arraysize(identifier_names),
+ identifiers);
+
+ // Fetch the document object from the window.
+ ScopedNpVariant document;
+ bool ok = npapi::GetProperty(instance_, window, identifiers[0], &document);
+ if (!ok) {
+ // This could happen if the page is being unloaded.
+ DLOG(WARNING) << "Failed to fetch the document object";
+ return false;
+ }
+
+ bool success = false;
+ if (ok && NPVARIANT_IS_OBJECT(document)) {
+ // Call document.createEvent("Event") to create a basic event object.
+ NPVariant event_type;
+ STRINGN_TO_NPVARIANT("Event", sizeof("Event") - 1, event_type);
+ ScopedNpVariant result;
+ success = npapi::Invoke(instance_, NPVARIANT_TO_OBJECT(document),
+ identifiers[1], &event_type, 1, &result);
+ if (!NPVARIANT_IS_OBJECT(result)) {
+ DLOG(WARNING) << "Failed to invoke createEvent";
+ success = false;
+ } else {
+ NPVariant init_args[3];
+ STRINGN_TO_NPVARIANT(type.c_str(), type.length(), init_args[0]);
+ BOOLEAN_TO_NPVARIANT(bubbles, init_args[1]);
+ BOOLEAN_TO_NPVARIANT(cancelable, init_args[2]);
+
+ // Now initialize the event object by calling
+ // event.initEvent(type, bubbles, cancelable);
+ ScopedNpVariant init_results;
+ ok = npapi::Invoke(instance_, NPVARIANT_TO_OBJECT(result), identifiers[2],
+ init_args, arraysize(init_args), &init_results);
+ if (ok) {
+ success = true;
+ // Finally, pass the ownership to the caller.
+ *basic_event = NPVARIANT_TO_OBJECT(result);
+ VOID_TO_NPVARIANT(result); // Prevent the object from being released.
+ } else {
+ DLOG(ERROR) << "initEvent failed";
+ success = false;
+ }
+ }
+ }
+
+ return success;
+}
+
+bool ChromeFrameNPAPI::CreateMessageEvent(bool bubbles, bool cancelable,
+ const std::string& data,
+ const std::string& origin,
+ NPObject** message_event) {
+ DCHECK(message_event);
+ ScopedNpObject<NPObject> event;
+ bool ok = CreateEvent("message", false, true, event.Receive());
+ if (ok) {
+ typedef enum {
+ DATA,
+ ORIGIN,
+ LAST_EVENT_ID,
+ SOURCE,
+ MESSAGE_PORT,
+ IDENTIFIER_COUNT, // Must be last.
+ } StringIdentifiers;
+
+ static NPIdentifier identifiers[IDENTIFIER_COUNT] = {0};
+ if (!identifiers[0]) {
+ const NPUTF8* identifier_names[] = {
+ "data",
+ "origin",
+ "lastEventId",
+ "source",
+ "messagePort",
+ };
+ COMPILE_ASSERT(arraysize(identifier_names) == arraysize(identifiers),
+ mismatched_array_size);
+ npapi::GetStringIdentifiers(identifier_names, IDENTIFIER_COUNT,
+ identifiers);
+ }
+
+ NPVariant arg;
+ STRINGN_TO_NPVARIANT(data.c_str(), data.length(), arg);
+ npapi::SetProperty(instance_, event, identifiers[DATA], &arg);
+ STRINGN_TO_NPVARIANT(origin.c_str(), origin.length(), arg);
+ npapi::SetProperty(instance_, event, identifiers[ORIGIN], &arg);
+ STRINGN_TO_NPVARIANT("", 0, arg);
+ npapi::SetProperty(instance_, event, identifiers[LAST_EVENT_ID], &arg);
+ NULL_TO_NPVARIANT(arg);
+ npapi::SetProperty(instance_, event, identifiers[SOURCE], &arg);
+ npapi::SetProperty(instance_, event, identifiers[MESSAGE_PORT], &arg);
+ *message_event = event.Detach();
+ }
+
+ return ok;
+}
+
+
+void ChromeFrameNPAPI::DispatchEvent(NPObject* event) {
+ DCHECK(event != NULL);
+
+ ScopedNpObject<NPObject> embed;
+ npapi::GetValue(instance_, NPNVPluginElementNPObject, &embed);
+ if (embed != NULL) {
+ NPVariant param;
+ OBJECT_TO_NPVARIANT(event, param);
+ ScopedNpVariant result;
+ bool invoke = npapi::Invoke(instance_, embed,
+ npapi::GetStringIdentifier("dispatchEvent"), &param, 1, &result);
+ DLOG_IF(WARNING, !invoke) << "dispatchEvent failed";
+ } else {
+ NOTREACHED() << "NPNVPluginElementNPObject";
+ }
+}
+
+bool ChromeFrameNPAPI::ExecuteScript(const std::string& script,
+ NPVariant* result) {
+ NPObject* window = GetWindowObject();
+ if (!window) {
+ NOTREACHED();
+ return false;
+ }
+
+ NPString script_for_execution;
+ script_for_execution.UTF8Characters = script.c_str();
+ script_for_execution.UTF8Length = script.length();
+
+ return npapi::Evaluate(instance_, window, &script_for_execution, result);
+}
+
+NPObject* ChromeFrameNPAPI::JavascriptToNPObject(const std::string& script) {
+ // Convert the passed in script to an invocable NPObject
+ // To achieve this we save away the function in a dummy window property
+ // which is then read to get the script object representing the function.
+
+ std::string script_code =
+ "javascript:window.__cf_get_function_object =";
+
+ // If we are able to look up the name in the javascript namespace, then it
+ // means that the caller passed in a function name. Convert the function
+ // name to a NPObject we can invoke on.
+ if (IsValidJavascriptFunction(script)) {
+ script_code += script;
+ } else {
+ script_code += "new Function(\"";
+ script_code += script;
+ script_code += "\");";
+ }
+
+ NPVariant result;
+ if (!ExecuteScript(script_code, &result)) {
+ NOTREACHED();
+ return NULL;
+ }
+
+ DCHECK(result.type == NPVariantType_Object);
+ DCHECK(result.value.objectValue != NULL);
+ return result.value.objectValue;
+}
+
+bool ChromeFrameNPAPI::IsValidJavascriptFunction(const std::string& script) {
+ std::string script_code = "javascript:window['";
+ script_code += script;
+ script_code += "'];";
+
+ ScopedNpVariant result;
+ if (!ExecuteScript(script_code, &result)) {
+ NOTREACHED();
+ return NULL;
+ }
+
+ return result.type == NPVariantType_Object;
+}
+
+bool ChromeFrameNPAPI::NavigateToURL(const NPVariant* args, uint32_t arg_count,
+ NPVariant* result) {
+ // Note that 'result' might be NULL.
+ if (arg_count != 1 || !(NPVARIANT_IS_STRING(args[0]) ||
+ NPVARIANT_IS_NULL(args[0]))) {
+ NOTREACHED();
+ return false;
+ }
+
+ if (ready_state_ == READYSTATE_UNINITIALIZED) {
+ // Error(L"Chrome Frame failed to initialize.");
+ // TODO(tommi): call NPN_SetException
+ DLOG(WARNING) << "NavigateToURL called after failed initialization";
+ return false;
+ }
+
+ std::string url("about:blank");
+
+ if (!NPVARIANT_IS_NULL(args[0])) {
+ const NPString& str = args[0].value.stringValue;
+ if (str.UTF8Length) {
+ url.assign(std::string(str.UTF8Characters, str.UTF8Length));
+ }
+ }
+ DLOG(WARNING) << __FUNCTION__ << " " << url;
+ std::string full_url = ResolveURL(GetDocumentUrl(), url);
+ if (full_url.empty())
+ return false;
+
+ src_ = full_url;
+ // Navigate only if we completed initialization i.e. proxy is set etc.
+ if (ready_state_ == READYSTATE_COMPLETE) {
+ if (!automation_client_->InitiateNavigation(full_url)) {
+ // TODO(tommi): call NPN_SetException.
+ src_.clear();
+ return false;
+ }
+ }
+ return true;
+}
+
+bool ChromeFrameNPAPI::postMessage(NPObject* npobject, const NPVariant* args,
+ uint32_t arg_count, NPVariant* result) {
+ if (arg_count < 1 || arg_count > 2 || !NPVARIANT_IS_STRING(args[0])) {
+ NOTREACHED();
+ return false;
+ }
+
+ const NPString& str = args[0].value.stringValue;
+ std::string message(str.UTF8Characters, str.UTF8Length);
+ std::string target;
+ if (arg_count == 2 && NPVARIANT_IS_STRING(args[1])) {
+ const NPString& str = args[1].value.stringValue;
+ target.assign(str.UTF8Characters, str.UTF8Length);
+ if (target.compare("*") != 0) {
+ GURL resolved(target);
+ if (!resolved.is_valid()) {
+ npapi::SetException(npobject,
+ "Unable to parse the specified target URL.");
+ return false;
+ }
+ target = resolved.spec();
+ }
+ } else {
+ target = "*";
+ }
+
+ GURL url(GURL(document_url_).GetOrigin());
+ std::string origin(url.is_empty() ? "null" : url.spec());
+
+ automation_client_->ForwardMessageFromExternalHost(message, origin, target);
+
+ return true;
+}
+
+bool ChromeFrameNPAPI::postPrivateMessage(NPObject* npobject,
+ const NPVariant* args,
+ uint32_t arg_count,
+ NPVariant* result) {
+ if (!is_privileged_) {
+ DLOG(WARNING) << "postPrivateMessage invoked in non-privileged mode";
+ return false;
+ }
+
+ if (arg_count != 3 || !NPVARIANT_IS_STRING(args[0]) ||
+ !NPVARIANT_IS_STRING(args[1]) || !NPVARIANT_IS_STRING(args[2])) {
+ NOTREACHED();
+ return false;
+ }
+
+ const NPString& message_str = args[0].value.stringValue;
+ const NPString& origin_str = args[1].value.stringValue;
+ const NPString& target_str = args[2].value.stringValue;
+ std::string message(message_str.UTF8Characters, message_str.UTF8Length);
+ std::string origin(origin_str.UTF8Characters, origin_str.UTF8Length);
+ std::string target(target_str.UTF8Characters, target_str.UTF8Length);
+
+ automation_client_->ForwardMessageFromExternalHost(message, origin, target);
+
+ return true;
+}
+
+void ChromeFrameNPAPI::FireEvent(const std::string& event_type,
+ const std::string& data) {
+ NPVariant arg;
+ STRINGN_TO_NPVARIANT(data.c_str(), data.length(), arg);
+ FireEvent(event_type, arg);
+}
+
+void ChromeFrameNPAPI::FireEvent(const std::string& event_type,
+ const NPVariant& data) {
+ // Check that we're not bundling an event inside an event.
+ // Right now we're only expecting simple types for the data argument.
+ DCHECK(NPVARIANT_IS_OBJECT(data) == false);
+
+ ScopedNpObject<NPObject> ev;
+ CreateEvent(event_type, false, false, ev.Receive());
+ if (ev) {
+ // Add the 'data' member to the event.
+ bool set = npapi::SetProperty(instance_, ev,
+ npapi::GetStringIdentifier("data"), const_cast<NPVariant*>(&data));
+ DCHECK(set);
+ DispatchEvent(ev);
+ }
+}
+
+NpProxyService* ChromeFrameNPAPI::CreatePrefService() {
+ return new NpProxyService;
+}
+
+NPObject* ChromeFrameNPAPI::GetWindowObject() const {
+ if (!window_object_.get()) {
+ NPError ret = npapi::GetValue(instance_, NPNVWindowNPObject,
+ window_object_.Receive());
+ DLOG_IF(ERROR, ret != NPERR_NO_ERROR) << "NPNVWindowNPObject failed";
+ }
+ return window_object_;
+}
+
+bool ChromeFrameNPAPI::GetBrowserIncognitoMode() {
+ bool incognito_mode = false;
+
+ // Check disabled for Opera due to bug: http://b/issue?id=1815494
+ if (GetBrowserType() != BROWSER_OPERA) {
+ // Check whether host browser is in private mode;
+ NPBool private_mode = FALSE;
+ NPError err = npapi::GetValue(instance_,
+ NPNVprivateModeBool,
+ &private_mode);
+ if (err == NPERR_NO_ERROR && private_mode) {
+ incognito_mode = true;
+ }
+ } else {
+ DLOG(WARNING) << "Not checking for private mode in Opera";
+ }
+
+ return incognito_mode;
+}
+
+NPAPIUrlRequest* ChromeFrameNPAPI::ValidateRequest(
+ NPP instance, void* notify_data) {
+ ChromeFrameNPAPI* plugin_instance =
+ ChromeFrameNPAPI::ChromeFrameInstanceFromPluginInstance(instance);
+ if (plugin_instance) {
+ return plugin_instance->RequestFromNotifyData(notify_data);
+ }
+
+ return NULL;
+}
+
+NPAPIUrlRequest* ChromeFrameNPAPI::RequestFromNotifyData(
+ void* notify_data) const {
+ NPAPIUrlRequest* request = reinterpret_cast<NPAPIUrlRequest*>(notify_data);
+ DCHECK(request ? automation_client_->IsValidRequest(request) : 1);
+ return request;
+}
+
+bool ChromeFrameNPAPI::HandleContextMenuCommand(UINT cmd) {
+ if (cmd == IDC_ABOUT_CHROME_FRAME) {
+ // TODO: implement "About Chrome Frame"
+ }
+ return false;
+}
diff --git a/chrome_frame/chrome_frame_npapi.h b/chrome_frame/chrome_frame_npapi.h
new file mode 100644
index 0000000..005d72f
--- /dev/null
+++ b/chrome_frame/chrome_frame_npapi.h
@@ -0,0 +1,333 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_FRAME_NPAPI_H_
+#define CHROME_FRAME_CHROME_FRAME_NPAPI_H_
+
+#include <atlbase.h>
+#include <atlwin.h>
+#include <string>
+
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/chrome_frame_plugin.h"
+#include "chrome_frame/np_browser_functions.h"
+#include "chrome_frame/np_event_listener.h"
+#include "chrome_frame/np_proxy_service.h"
+#include "chrome_frame/npapi_url_request.h"
+
+class MessageLoop;
+
+// ChromeFrameNPAPI: Implementation of the NPAPI plugin, which is responsible
+// for hosting a chrome frame, i.e. an iframe like widget which hosts the the
+// chrome window. This object delegates to Chrome.exe (via the Chrome
+// IPC-based automation mechanism) for the actual rendering.
+class ChromeFrameNPAPI
+ : public CWindowImpl<ChromeFrameNPAPI>,
+ public ChromeFramePlugin<ChromeFrameNPAPI>,
+ public NpEventDelegate {
+ public:
+ typedef ChromeFramePlugin<ChromeFrameNPAPI> Base;
+
+ // NPObject structure which is exposed by us.
+ struct ChromeFrameNPObject : public NPObject {
+ NPP npp;
+ ChromeFrameNPAPI* chrome_frame_plugin_instance;
+ };
+
+ typedef enum {
+ PLUGIN_PROPERTY_VERSION,
+ PLUGIN_PROPERTY_SRC,
+ PLUGIN_PROPERTY_ONLOAD,
+ PLUGIN_PROPERTY_ONERROR,
+ PLUGIN_PROPERTY_ONMESSAGE,
+ PLUGIN_PROPERTY_READYSTATE,
+ PLUGIN_PROPERTY_ONPRIVATEMESSAGE,
+ PLUGIN_PROPERTY_USECHROMENETWORK,
+ PLUGIN_PROPERTY_COUNT // must be last
+ } PluginPropertyId;
+
+ static const int kWmSwitchFocusToChromeFrame = WM_APP + 0x100;
+
+ static NPClass plugin_class_;
+ static NPClass* PluginClass() {
+ return &plugin_class_;
+ }
+
+ ChromeFrameNPAPI();
+ ~ChromeFrameNPAPI();
+
+ bool Initialize(NPMIMEType mime_type, NPP instance, uint16 mode,
+ int16 argc, char* argn[], char* argv[]);
+ void Uninitialize();
+
+ bool SetWindow(NPWindow* window_info);
+ void UrlNotify(const char* url, NPReason reason, void* notify_data);
+ bool NewStream(NPMIMEType type, NPStream* stream, NPBool seekable,
+ uint16* stream_type);
+
+ void Print(NPPrint* print_info);
+
+ // NPObject functions, which ensure that the plugin object is scriptable.
+ static bool HasMethod(NPObject* obj, NPIdentifier name);
+ static bool Invoke(NPObject* header, NPIdentifier name,
+ const NPVariant* args, uint32_t arg_count,
+ NPVariant* result);
+ static NPObject* AllocateObject(NPP instance, NPClass* class_name);
+ static void DeallocateObject(NPObject* header);
+
+ // Called by the scripting environment when the native code is shutdown.
+ // Any attempt to message a NPObject instance after the invalidate callback
+ // has been called will result in undefined behavior, even if the native code
+ // is still retaining those NPObject instances.
+ static void Invalidate(NPObject* header);
+
+ // The following functions need to be implemented to ensure that FF3
+ // invokes methods on the plugin. If these methods are not implemented
+ // then invokes on the plugin NPObject from the script fail with a
+ // bad NPObject error.
+ static bool HasProperty(NPObject* obj, NPIdentifier name);
+ static bool GetProperty(NPObject* obj, NPIdentifier name, NPVariant *variant);
+ static bool SetProperty(NPObject* obj, NPIdentifier name,
+ const NPVariant *variant);
+
+ // Returns the ChromeFrameNPAPI object pointer from the NPP instance structure
+ // passed in by the browser.
+ static ChromeFrameNPAPI* ChromeFrameInstanceFromPluginInstance(NPP instance);
+
+ // Returns the ChromeFrameNPAPI object pointer from the NPObject structure
+ // which represents our plugin class.
+ static ChromeFrameNPAPI* ChromeFrameInstanceFromNPObject(void* object);
+
+ // Return a UrlRequest instance associated with the given instance and
+ // stream combination.
+ static NPAPIUrlRequest* ValidateRequest(NPP instance, void* notify_data);
+
+BEGIN_MSG_MAP(ChromeFrameNPAPI)
+ MESSAGE_HANDLER(WM_SETFOCUS, OnSetFocus)
+ CHAIN_MSG_MAP(Base)
+END_MSG_MAP()
+
+ LRESULT OnSetFocus(UINT message, WPARAM wparam, LPARAM lparam,
+ BOOL& handled); // NO_LINT
+
+ // Implementation of NpEventDelegate
+ virtual void OnEvent(const char* event_name);
+
+ void OnFocus();
+ void OnBlur();
+
+ // Implementation of SetProperty, public to allow unittesting.
+ bool SetProperty(NPIdentifier name, const NPVariant *variant);
+ // Implementation of GetProperty, public to allow unittesting.
+ bool GetProperty(NPIdentifier name, NPVariant *variant);
+
+ // Initialize string->identifier mapping, public to allow unittesting.
+ static void InitializeIdentifiers();
+
+ bool HandleContextMenuCommand(UINT cmd);
+ protected:
+ // Handler for accelerator messages passed on from the hosted chrome
+ // instance.
+ virtual void OnAcceleratorPressed(int tab_handle, const MSG& accel_message);
+ virtual void OnTabbedOut(int tab_handle, bool reverse);
+ virtual void OnOpenURL(int tab_handle, const GURL& url, int open_disposition);
+ virtual void OnLoad(int tab_handle, const GURL& url);
+ virtual void OnMessageFromChromeFrame(int tab_handle,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& target);
+ virtual void OnRequestStart(int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request);
+ virtual void OnRequestRead(int tab_handle, int request_id,
+ int bytes_to_read);
+ virtual void OnRequestEnd(int tab_handle, int request_id,
+ const URLRequestStatus& status);
+ virtual void OnSetCookieAsync(int tab_handle, const GURL& url,
+ const std::string& cookie);
+
+ // ChromeFrameDelegate overrides
+ virtual void OnLoadFailed(int error_code, const std::string& url);
+ virtual void OnAutomationServerReady();
+ virtual void OnAutomationServerLaunchFailed(
+ AutomationLaunchResult reason, const std::string& server_version);
+
+ private:
+ void SubscribeToFocusEvents();
+ void UnsubscribeFromFocusEvents();
+
+ // Equivalent of:
+ // event = window.document.createEvent("Event");
+ // event.initEvent(type, bubbles, cancelable);
+ // and then returns the event object.
+ bool CreateEvent(const std::string& type, bool bubbles, bool cancelable,
+ NPObject** basic_event);
+
+ // Creates and initializes an event object of type "message".
+ // Used for postMessage.
+ bool CreateMessageEvent(bool bubbles, bool cancelable,
+ const std::string& data, const std::string& origin,
+ NPObject** message_event);
+
+ // Calls chrome_frame.dispatchEvent to fire events to event listeners.
+ void DispatchEvent(NPObject* event);
+
+ // Returns a pointer to the <object> element in the page that
+ // hosts the plugin. Note that this is the parent element of the <embed>
+ // element. The <embed> element doesn't support some of the events that
+ // we require, so we use the object element for receiving events.
+ bool GetObjectElement(nsIDOMElement** element);
+
+ // Prototype for all methods that can be invoked from script.
+ typedef bool (ChromeFrameNPAPI::*PluginMethod)(NPObject* npobject,
+ const NPVariant* args,
+ uint32_t arg_count,
+ NPVariant* result);
+
+ // Implementations of scriptable methods.
+
+ bool NavigateToURL(const NPVariant* args, uint32_t arg_count,
+ NPVariant* result);
+
+ bool postMessage(NPObject* npobject, const NPVariant* args,
+ uint32_t arg_count, NPVariant* result);
+
+ // This method is only available when the control is in privileged mode.
+ bool postPrivateMessage(NPObject* npobject, const NPVariant* args,
+ uint32_t arg_count, NPVariant* result);
+
+ // Pointers to method implementations.
+ static PluginMethod plugin_methods_[];
+
+ // NPObject method ids exposed by the plugin.
+ static NPIdentifier plugin_method_identifiers_[];
+
+ // NPObject method names exposed by the plugin.
+ static const NPUTF8* plugin_method_identifier_names_[];
+
+ // NPObject property ids exposed by the plugin.
+ static NPIdentifier plugin_property_identifiers_[];
+
+ // NPObject property names exposed by the plugin.
+ static const NPUTF8*
+ plugin_property_identifier_names_[];
+
+ virtual void OnFinalMessage(HWND window);
+
+ // Helper function to invoke a function on a NPObject.
+ bool InvokeDefault(NPObject* object, const std::string& param,
+ NPVariant* result);
+
+ bool InvokeDefault(NPObject* object, const NPVariant& param,
+ NPVariant* result);
+
+ bool InvokeDefault(NPObject* object, unsigned param_count,
+ const NPVariant* params, NPVariant* result);
+
+ // Helper function to convert javascript code to a NPObject we can
+ // invoke on.
+ virtual NPObject* JavascriptToNPObject(const std::string& function_name);
+
+ // Helper function to execute a script.
+ // Returns S_OK on success.
+ bool ExecuteScript(const std::string& script, NPVariant* result);
+
+ // Returns true if the script passed in is a valid function in the DOM.
+ bool IsValidJavascriptFunction(const std::string& script);
+
+ // Converts the data parameter to an NPVariant and forwards the call to the
+ // other FireEvent method.
+ void FireEvent(const std::string& event_type, const std::string& data);
+
+ // Creates an event object, assigns the data parameter to a |data| property
+ // on the event object and then calls DispatchEvent to fire the event to
+ // listeners. event_type is the name of the event being fired.
+ void FireEvent(const std::string& event_type, const NPVariant& data);
+
+ // Returns a new prefs service. Virtual to allow overriding in unittests.
+ virtual NpProxyService* CreatePrefService();
+
+ // Returns our associated windows' location.
+ virtual std::string GetLocation();
+
+ // Returns true iff we're successfully able to query for the browser's
+ // incognito mode, and the browser returns true.
+ virtual bool GetBrowserIncognitoMode();
+
+ // Returns the window script object for the page.
+ // This function will cache the window object to avoid calling
+ // npapi::GetValue which can cause problems in Opera.
+ NPObject* GetWindowObject() const;
+
+ virtual void SetReadyState(READYSTATE new_state) {
+ ready_state_ = new_state;
+ NPVariant var;
+ INT32_TO_NPVARIANT(ready_state_, var);
+ FireEvent("readystatechanged", var);
+ }
+
+ // Host function to compile-time asserts over members of this class.
+ static void CompileAsserts();
+
+ // Get request from the stream notify data
+ NPAPIUrlRequest* RequestFromNotifyData(void* notify_data) const;
+
+ static LRESULT CALLBACK DropKillFocusHook(int code, WPARAM wparam,
+ LPARAM lparam); // NO_LINT
+
+ // The plugins opaque instance handle
+ NPP instance_;
+
+ // The plugin instantiation mode (NP_FULL or NP_EMBED)
+ int16 mode_;
+ // The plugins mime type.
+ std::string mime_type_;
+
+ // Set to true if we need a full page plugin.
+ bool force_full_page_plugin_;
+
+ scoped_refptr<NpProxyService> pref_service_;
+
+ // Used to receive focus and blur events from the object element
+ // that hosts the plugin.
+ scoped_refptr<NpEventListener> focus_listener_;
+
+ // In some cases the IPC channel proxy object is instantiated on the UI
+ // thread in FF. It then tries to use the IPC logger, which relies on
+ // the message loop being around. Declaring a dummy message loop
+ // is a hack to get around this. Eventually the automation code needs to
+ // be fixed to ensure that the channel proxy always gets created on a thread
+ // with a message loop.
+ static MessageLoop* message_loop_;
+ static int instance_count_;
+
+ // The following members contain the NPObject pointers representing the
+ // onload/onerror/onmessage handlers on the page.
+ ScopedNpObject<NPObject> onerror_handler_;
+ ScopedNpObject<NPObject> onmessage_handler_;
+ ScopedNpObject<NPObject> onprivatemessage_handler_;
+
+ // As a workaround for a problem in Opera we cache the window object.
+ // The problem stems from two things: window messages aren't always removed
+ // from the message queue and messages can be pumped inside GetValue.
+ // This can cause an infinite recursion of processing the same message
+ // repeatedly.
+ mutable ScopedNpObject<NPObject> window_object_;
+
+ // Note since 'onload' is a registered event name, the browser will
+ // automagically create a code block for the handling code and hook it
+ // up to the CF object via addEventListener.
+ // See this list of known event types:
+ // http://www.w3.org/TR/DOM-Level-3-Events/events.html#Event-types
+
+ READYSTATE ready_state_;
+
+
+ // Popups are enabled
+ bool enabled_popups_;
+
+ // The value of src property keeping the current URL.
+ std::string src_;
+};
+
+#endif // CHROME_FRAME_CHROME_FRAME_NPAPI_H_
diff --git a/chrome_frame/chrome_frame_npapi.rgs b/chrome_frame/chrome_frame_npapi.rgs
new file mode 100644
index 0000000..70e6ca1
--- /dev/null
+++ b/chrome_frame/chrome_frame_npapi.rgs
@@ -0,0 +1,13 @@
+HKLM {
+ NoRemove Software {
+ NoRemove MozillaPlugins {
+ ForceRemove '@google.com/ChromeFrame,version=1.0' {
+ val Path = s '%MODULE%'
+ val Description = s 'Google ChromeFrame'
+ val ProductName = s 'Google ChromeFrame'
+ val Vendor = s 'Google'
+ val Version = s '%VERSION%'
+ }
+ }
+ }
+}
diff --git a/chrome_frame/chrome_frame_npapi_entrypoints.cc b/chrome_frame/chrome_frame_npapi_entrypoints.cc
new file mode 100644
index 0000000..f1eec1c
--- /dev/null
+++ b/chrome_frame/chrome_frame_npapi_entrypoints.cc
@@ -0,0 +1,53 @@
+// Copyright (c) 2009 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 "chrome_frame/chrome_frame_npapi.h"
+
+#define NPAPI WINAPI
+
+// Plugin entry points.
+extern "C" {
+ NPError NPAPI NP_Initialize(NPNetscapeFuncs* browser_funcs);
+ NPError NPAPI NP_GetEntryPoints(NPPluginFuncs* plugin_funcs);
+ void NPAPI NP_Shutdown();
+}
+
+NPError NPAPI NP_Initialize(NPNetscapeFuncs* browser_funcs) {
+ DLOG(INFO) << __FUNCTION__;
+ _pAtlModule->Lock();
+ npapi::InitializeBrowserFunctions(browser_funcs);
+ return NPERR_NO_ERROR;
+}
+
+NPError NPAPI NP_GetEntryPoints(NPPluginFuncs* plugin_funcs) {
+ plugin_funcs->version = (NP_VERSION_MAJOR << 8) | NP_VERSION_MINOR;
+ plugin_funcs->size = sizeof(plugin_funcs);
+ plugin_funcs->newp = NPP_New;
+ plugin_funcs->destroy = NPP_Destroy;
+ plugin_funcs->setwindow = NPP_SetWindow;
+ plugin_funcs->newstream = NPP_NewStream;
+ plugin_funcs->destroystream = NPP_DestroyStream;
+ plugin_funcs->asfile = NULL;
+ plugin_funcs->writeready = NPP_WriteReady;
+ plugin_funcs->write = NPP_Write;
+ plugin_funcs->print = NPP_Print;
+ plugin_funcs->event = NULL;
+ plugin_funcs->urlnotify = NPP_URLNotify;
+ plugin_funcs->getvalue = NPP_GetValue;
+ plugin_funcs->setvalue = NPP_SetValue;
+ return NPERR_NO_ERROR;
+}
+
+void NPAPI NP_Shutdown() {
+ DLOG(INFO) << __FUNCTION__;
+
+ npapi::UninitializeBrowserFunctions();
+
+ _pAtlModule->Unlock(); // matches Lock() inside NP_Initialize
+
+ DLOG_IF(ERROR, _pAtlModule->GetLockCount() != 0)
+ << "Being shut down but still have " << _pAtlModule->GetLockCount()
+ << " living objects";
+}
+
diff --git a/chrome_frame/chrome_frame_npapi_unittest.cc b/chrome_frame/chrome_frame_npapi_unittest.cc
new file mode 100644
index 0000000..d2c9b4e
--- /dev/null
+++ b/chrome_frame/chrome_frame_npapi_unittest.cc
@@ -0,0 +1,551 @@
+// Copyright (c) 2009 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 "gtest/gtest.h"
+#include "gmock/gmock.h"
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/chrome_frame_npapi.h"
+#include "chrome_frame/ff_privilege_check.h"
+
+
+TEST(ChromeFrameNPAPI, DoesNotCrashOnConstruction) {
+ ChromeFrameNPAPI* api = new ChromeFrameNPAPI();
+ delete api;
+}
+
+
+// All mocks in the anonymous namespace.
+namespace {
+
+using ::testing::_;
+using ::testing::Eq;
+using ::testing::Invoke;
+using ::testing::Return;
+using ::testing::StrEq;
+
+// Make mocking privilege test easy.
+class MockPrivilegeTest {
+ public:
+ MockPrivilegeTest() {
+ CHECK(current_ == NULL);
+ current_ = this;
+ }
+ ~MockPrivilegeTest() {
+ CHECK(current_ == this);
+ current_ = NULL;
+ }
+
+ MOCK_METHOD1(IsFireFoxPrivilegedInvocation, bool(NPP));
+
+ static MockPrivilegeTest* current() { return current_; }
+
+ private:
+ static MockPrivilegeTest* current_;
+};
+
+MockPrivilegeTest* MockPrivilegeTest::current_ = NULL;
+
+const char* kMimeType = "application/chromeframe";
+// The default profile name is by default derived from the currently
+// running executable's name.
+const wchar_t* kDefaultProfileName = L"chrome_frame_unittests";
+
+
+class MockNPAPI: public ChromeFrameNPAPI {
+ public:
+ MockNPAPI() : mock_automation_client_(NULL) {}
+
+ MOCK_METHOD0(CreatePrefService, NpProxyService*());
+
+ MOCK_METHOD0(GetLocation, std::string());
+ MOCK_METHOD0(GetBrowserIncognitoMode, bool());
+
+ MOCK_METHOD1(JavascriptToNPObject, virtual NPObject*(const std::string&));
+
+ // Make public for test purposes
+ void OnAutomationServerReady() {
+ ChromeFrameNPAPI::OnAutomationServerReady();
+ }
+
+ // Neuter this (or it dchecks during testing).
+ void SetReadyState(READYSTATE new_state) {}
+
+ ChromeFrameAutomationClient* CreateAutomationClient() {
+ return mock_automation_client_;
+ }
+
+ ChromeFrameAutomationClient* mock_automation_client_;
+};
+
+class MockAutomationClient: public ChromeFrameAutomationClient {
+ public:
+ MOCK_METHOD6(Initialize, bool(ChromeFrameDelegate*, int, bool,
+ const std::wstring&, const std::wstring&,
+ bool));
+ MOCK_METHOD1(SetEnableExtensionAutomation, void(bool)); // NOLINT
+};
+
+class MockProxyService: public NpProxyService {
+ public:
+ MOCK_METHOD2(Initialize, bool(NPP instance, ChromeFrameAutomationClient*));
+};
+
+
+// Test fixture to allow testing the privileged NPAPI APIs
+class TestNPAPIPrivilegedApi: public ::testing::Test {
+ public:
+ virtual void SetUp() {
+ memset(&instance, 0, sizeof(instance));
+
+ // Gets owned & destroyed by mock_api (in the
+ // ChromeFramePlugin<T>::Uninitialize() function).
+ mock_automation = new MockAutomationClient;
+
+ mock_api.mock_automation_client_ = mock_automation;
+ mock_proxy = new MockProxyService;
+ mock_proxy->AddRef();
+ mock_proxy_holder.Attach(mock_proxy);
+ }
+
+ virtual void TearDown() {
+ }
+
+ void SetupPrivilegeTest(bool is_incognito,
+ bool expect_privilege_check,
+ bool is_privileged,
+ const std::wstring& profile_name,
+ const std::wstring& extra_args) {
+ EXPECT_CALL(mock_api, GetLocation())
+ .WillOnce(Return(std::string("http://www.google.com")));
+ EXPECT_CALL(mock_api, CreatePrefService())
+ .WillOnce(Return(mock_proxy));
+ EXPECT_CALL(mock_api, GetBrowserIncognitoMode())
+ .WillOnce(Return(is_incognito));
+
+ EXPECT_CALL(*mock_proxy, Initialize(_, _)).WillRepeatedly(Return(false));
+
+ EXPECT_CALL(*mock_automation,
+ Initialize(_, _, true, StrEq(profile_name), StrEq(extra_args), false))
+ .WillOnce(Return(true));
+
+ if (expect_privilege_check) {
+ EXPECT_CALL(mock_priv, IsFireFoxPrivilegedInvocation(_))
+ .WillOnce(Return(is_privileged));
+ } else {
+ EXPECT_CALL(mock_priv, IsFireFoxPrivilegedInvocation(_))
+ .Times(0); // Fail if privilege check invoked.
+ }
+ }
+
+ public:
+ MockNPAPI mock_api;
+ MockAutomationClient* mock_automation;
+ MockProxyService* mock_proxy;
+ ScopedNsPtr<nsISupports> mock_proxy_holder;
+ MockPrivilegeTest mock_priv;
+ NPP_t instance;
+};
+
+} // namespace
+
+// Stub for unittesting.
+bool IsFireFoxPrivilegedInvocation(NPP npp) {
+ MockPrivilegeTest* mock = MockPrivilegeTest::current();
+ if (!mock)
+ return false;
+
+ return mock->IsFireFoxPrivilegedInvocation(npp);
+}
+
+TEST_F(TestNPAPIPrivilegedApi, NoPrivilegeCheckWhenNoArguments) {
+ SetupPrivilegeTest(false, // Not incognito
+ false, // Fail if privilege check is invoked.
+ false,
+ kDefaultProfileName,
+ L""); // No extra args to initialize.
+
+ // No arguments, no privilege requested.
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ 0, 0, 0));
+}
+
+TEST_F(TestNPAPIPrivilegedApi, NoPrivilegeCheckWhenZeroArgument) {
+ SetupPrivilegeTest(false, // Not incognito
+ false, // Fail if privilege check is invoked.
+ false,
+ kDefaultProfileName,
+ L""); // No extra args to initialize.
+
+ // Privileged mode explicitly zero.
+ char* argn = "is_privileged";
+ char* argv = "0";
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ 1, &argn, &argv));
+}
+
+TEST_F(TestNPAPIPrivilegedApi, NotPrivilegedDoesNotAllowArgsOrProfile) {
+ SetupPrivilegeTest(false, // Not incognito.
+ true, // Fail unless privilege check is invoked.
+ false, // Not privileged.
+ kDefaultProfileName,
+ L""); // No extra arguments allowed.
+
+ char* argn[] = {
+ "privileged_mode",
+ "chrome_extra_arguments",
+ "chrome_profile_name",
+ };
+ char *argv[] = {
+ "1",
+ "foo",
+ "bar",
+ };
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ arraysize(argn), argn, argv));
+}
+
+TEST_F(TestNPAPIPrivilegedApi, PrivilegedAllowsArgsAndProfile) {
+ SetupPrivilegeTest(false, // Not incognito.
+ true, // Fail unless privilege check is invoked.
+ true, // Privileged mode.
+ L"custom_profile_name", // Custom profile expected.
+ L"-bar=far"); // Extra arguments expected
+
+ // With privileged mode we expect automation to be enabled.
+ EXPECT_CALL(*mock_automation, SetEnableExtensionAutomation(true))
+ .Times(1);
+
+ char* argn[] = {
+ "privileged_mode",
+ "chrome_extra_arguments",
+ "chrome_profile_name",
+ };
+ char *argv[] = {
+ "1",
+ "-bar=far",
+ "custom_profile_name",
+ };
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ arraysize(argn), argn, argv));
+
+ // Since we're mocking out ChromeFrameAutomationClient::Initialize, we need
+ // to tickle this explicitly.
+ mock_api.OnAutomationServerReady();
+}
+
+
+namespace {
+
+static const NPIdentifier kOnPrivateMessageId =
+ reinterpret_cast<NPIdentifier>(0x100);
+static const NPIdentifier kPostPrivateMessageId =
+ reinterpret_cast<NPIdentifier>(0x100);
+
+
+class MockNetscapeFuncs {
+ public:
+ MockNetscapeFuncs() {
+ CHECK(NULL == current_);
+ current_ = this;
+ }
+
+ ~MockNetscapeFuncs() {
+ CHECK(this == current_);
+ current_ = NULL;
+ }
+
+ MOCK_METHOD3(GetValue, NPError(NPP, NPNVariable, void *));
+ MOCK_METHOD3(GetStringIdentifiers, void(const NPUTF8 **,
+ int32_t,
+ NPIdentifier *)); // NOLINT
+ MOCK_METHOD1(RetainObject, NPObject*(NPObject*)); // NOLINT
+ MOCK_METHOD1(ReleaseObject, void(NPObject*)); // NOLINT
+
+
+ void GetPrivilegedStringIdentifiers(const NPUTF8 **names,
+ int32_t name_count,
+ NPIdentifier *identifiers) {
+ for (int32_t i = 0; i < name_count; ++i) {
+ if (0 == strcmp(names[i], "onprivatemessage")) {
+ identifiers[i] = kOnPrivateMessageId;
+ } else if (0 == strcmp(names[i], "postPrivateMessage")) {
+ identifiers[i] = kPostPrivateMessageId;
+ } else {
+ identifiers[i] = 0;
+ }
+ }
+ }
+
+ static const NPNetscapeFuncs* netscape_funcs() {
+ return &netscape_funcs_;
+ }
+
+ private:
+ static NPError MockGetValue(NPP instance,
+ NPNVariable variable,
+ void *ret_value) {
+ DCHECK(current_);
+ return current_->GetValue(instance, variable, ret_value);
+ }
+
+ static void MockGetStringIdentifiers(const NPUTF8 **names,
+ int32_t name_count,
+ NPIdentifier *identifiers) {
+ DCHECK(current_);
+ return current_->GetStringIdentifiers(names, name_count, identifiers);
+ }
+
+ static NPObject* MockRetainObject(NPObject* obj) {
+ DCHECK(current_);
+ return current_->RetainObject(obj);
+ }
+
+ static void MockReleaseObject(NPObject* obj) {
+ DCHECK(current_);
+ current_->ReleaseObject(obj);
+ }
+
+ static MockNetscapeFuncs* current_;
+ static NPNetscapeFuncs netscape_funcs_;
+};
+
+MockNetscapeFuncs* MockNetscapeFuncs::current_ = NULL;
+NPNetscapeFuncs MockNetscapeFuncs::netscape_funcs_ = {
+ 0, // size
+ 0, // version
+ NULL, // geturl
+ NULL, // posturl
+ NULL, // requestread
+ NULL, // newstream
+ NULL, // write
+ NULL, // destroystream
+ NULL, // status
+ NULL, // uagent
+ NULL, // memalloc
+ NULL, // memfree
+ NULL, // memflush
+ NULL, // reloadplugins
+ NULL, // getJavaEnv
+ NULL, // getJavaPeer
+ NULL, // geturlnotify
+ NULL, // posturlnotify
+ MockGetValue, // getvalue
+ NULL, // setvalue
+ NULL, // invalidaterect
+ NULL, // invalidateregion
+ NULL, // forceredraw
+ NULL, // getstringidentifier
+ MockGetStringIdentifiers, // getstringidentifiers
+ NULL, // getintidentifier
+ NULL, // identifierisstring
+ NULL, // utf8fromidentifier
+ NULL, // intfromidentifier
+ NULL, // createobject
+ MockRetainObject, // retainobject
+ MockReleaseObject, // releaseobject
+ NULL, // invoke
+ NULL, // invokeDefault
+ NULL, // evaluate
+ NULL, // getproperty
+ NULL, // setproperty
+ NULL, // removeproperty
+ NULL, // hasproperty
+ NULL, // hasmethod
+ NULL, // releasevariantvalue
+ NULL, // setexception
+ NULL, // pushpopupsenabledstate
+ NULL, // poppopupsenabledstate
+ NULL, // enumerate
+ NULL, // pluginthreadasynccall
+ NULL, // construct
+};
+
+NPObject* const kMockNPObject = reinterpret_cast<NPObject*>(0xCafeBabe);
+
+class TestNPAPIPrivilegedProperty: public TestNPAPIPrivilegedApi {
+ public:
+ virtual void SetUp() {
+ TestNPAPIPrivilegedApi::SetUp();
+ npapi::InitializeBrowserFunctions(
+ const_cast<NPNetscapeFuncs*>(mock_funcs.netscape_funcs()));
+
+ // Expect calls to release and retain objects.
+ EXPECT_CALL(mock_funcs, RetainObject(kMockNPObject))
+ .WillRepeatedly(Return(kMockNPObject));
+ EXPECT_CALL(mock_funcs, ReleaseObject(kMockNPObject))
+ .WillRepeatedly(Return());
+
+ // And we should expect SetEnableExtensionAutomation to be called
+ // for privileged tests.
+ EXPECT_CALL(*mock_automation, SetEnableExtensionAutomation(true))
+ .WillRepeatedly(Return());
+
+ // Initializes identifiers.
+ EXPECT_CALL(mock_funcs, GetStringIdentifiers(_, _, _))
+ .WillRepeatedly(
+ Invoke(&mock_funcs,
+ &MockNetscapeFuncs::GetPrivilegedStringIdentifiers));
+ MockNPAPI::InitializeIdentifiers();
+ }
+
+ virtual void TearDown() {
+ npapi::UninitializeBrowserFunctions();
+ TestNPAPIPrivilegedApi::TearDown();
+ }
+
+ public:
+ MockNetscapeFuncs mock_funcs;
+};
+
+
+} // namespace
+
+TEST_F(TestNPAPIPrivilegedProperty,
+ NonPrivilegedOnPrivateMessageInitializationFails) {
+ // Attempt setting onprivatemessage when not privileged.
+ SetupPrivilegeTest(false, // not incognito.
+ true, // expect privilege check.
+ false, // not privileged.
+ kDefaultProfileName,
+ L"");
+
+ char* on_private_message_str = "onprivatemessage()";
+ EXPECT_CALL(mock_api, JavascriptToNPObject(StrEq(on_private_message_str)))
+ .Times(0); // this should not be called.
+
+ char* argn[] = {
+ "privileged_mode",
+ "onprivatemessage",
+ };
+ char* argv[] = {
+ "1",
+ on_private_message_str,
+ };
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ arraysize(argn), argn, argv));
+ // Shouldn't be able to retrieve it.
+ NPVariant var;
+ VOID_TO_NPVARIANT(var);
+ EXPECT_FALSE(mock_api.GetProperty(kOnPrivateMessageId, &var));
+ EXPECT_TRUE(NPVARIANT_IS_VOID(var));
+
+ mock_api.Uninitialize();
+}
+
+TEST_F(TestNPAPIPrivilegedProperty,
+ PrivilegedOnPrivateMessageInitializationSucceeds) {
+ // Set onprivatemessage argument when privileged.
+ SetupPrivilegeTest(false, // not incognito.
+ true, // expect privilege check.
+ true, // privileged.
+ kDefaultProfileName,
+ L"");
+
+ char* on_private_message_str = "onprivatemessage()";
+ NPObject* on_private_object = kMockNPObject;
+ EXPECT_CALL(mock_api, JavascriptToNPObject(StrEq(on_private_message_str)))
+ .WillOnce(Return(on_private_object));
+
+ char* argn[] = {
+ "privileged_mode",
+ "onprivatemessage",
+ };
+ char* argv[] = {
+ "1",
+ on_private_message_str,
+ };
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ arraysize(argn), argn, argv));
+ // The property should have been set, verify that
+ // we can retrieve it and test it for correct value.
+ NPVariant var;
+ VOID_TO_NPVARIANT(var);
+ EXPECT_TRUE(mock_api.GetProperty(kOnPrivateMessageId, &var));
+ EXPECT_TRUE(NPVARIANT_IS_OBJECT(var));
+ EXPECT_EQ(kMockNPObject, NPVARIANT_TO_OBJECT(var));
+
+ mock_api.Uninitialize();
+}
+
+TEST_F(TestNPAPIPrivilegedProperty,
+ NonPrivilegedOnPrivateMessageAssignmentFails) {
+ // Assigning to onprivatemessage when not privileged should fail.
+ SetupPrivilegeTest(false, // not incognito.
+ true, // expect privilege check.
+ false, // not privileged.
+ kDefaultProfileName,
+ L"");
+
+ char* argn = "privileged_mode";
+ char* argv = "1";
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ 1, &argn, &argv));
+
+ NPVariant var = {};
+ OBJECT_TO_NPVARIANT(kMockNPObject, var);
+ // Setting should fail.
+ EXPECT_FALSE(mock_api.SetProperty(kOnPrivateMessageId, &var));
+
+ // And so should getting.
+ NULL_TO_NPVARIANT(var);
+ EXPECT_FALSE(mock_api.GetProperty(kOnPrivateMessageId, &var));
+
+ mock_api.Uninitialize();
+}
+
+TEST_F(TestNPAPIPrivilegedProperty,
+ PrivilegedOnPrivateMessageAssignmentSucceeds) {
+ // Assigning to onprivatemessage when privileged should succeed.
+ SetupPrivilegeTest(false, // not incognito.
+ true, // expect privilege check.
+ true, // privileged.
+ kDefaultProfileName,
+ L"");
+
+ char* argn = "privileged_mode";
+ char* argv = "1";
+ EXPECT_TRUE(mock_api.Initialize(const_cast<NPMIMEType>(kMimeType),
+ &instance,
+ NP_EMBED,
+ 1, &argn, &argv));
+
+ NPVariant var = {};
+ VOID_TO_NPVARIANT(var);
+ // Getting the property when NULL fails under current implementation.
+ // I shouldn't have thought this is correct behavior, e.g. I should
+ // have thought retrieving the NULL should succeed, but this is consistent
+ // with how other properties behave.
+ // TODO(robertshield): investigate and/or fix.
+ EXPECT_FALSE(mock_api.GetProperty(kOnPrivateMessageId, &var));
+ // EXPECT_TRUE(NPVARIANT_IS_OBJECT(var));
+ // EXPECT_EQ(NULL, NPVARIANT_TO_OBJECT(var));
+
+ // Setting the property should succeed.
+ OBJECT_TO_NPVARIANT(kMockNPObject, var);
+ EXPECT_TRUE(mock_api.SetProperty(kOnPrivateMessageId, &var));
+
+ // And fething it should return the value we just set.
+ VOID_TO_NPVARIANT(var);
+ EXPECT_TRUE(mock_api.GetProperty(kOnPrivateMessageId, &var));
+ EXPECT_TRUE(NPVARIANT_IS_OBJECT(var));
+ EXPECT_EQ(kMockNPObject, NPVARIANT_TO_OBJECT(var));
+
+ mock_api.Uninitialize();
+}
+
+// TODO(siggi): test invoking postPrivateMessage.
diff --git a/chrome_frame/chrome_frame_plugin.h b/chrome_frame/chrome_frame_plugin.h
new file mode 100644
index 0000000..b0814bb
--- /dev/null
+++ b/chrome_frame/chrome_frame_plugin.h
@@ -0,0 +1,214 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_FRAME_PLUGIN_H_
+#define CHROME_FRAME_CHROME_FRAME_PLUGIN_H_
+
+#include "base/win_util.h"
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/utils.h"
+
+#define IDC_ABOUT_CHROME_FRAME 40018
+
+// A class to implement common functionality for all types of
+// plugins: NPAPI. ActiveX and ActiveDoc
+template <typename T>
+class ChromeFramePlugin : public ChromeFrameDelegateImpl {
+ public:
+ ChromeFramePlugin()
+ : ignore_setfocus_(false),
+ is_privileged_(false) {
+ }
+ ~ChromeFramePlugin() {
+ Uninitialize();
+ }
+
+BEGIN_MSG_MAP(ChromeFrameActivex)
+ MESSAGE_HANDLER(WM_SETFOCUS, OnSetFocus)
+ MESSAGE_HANDLER(WM_SIZE, OnSize)
+ MESSAGE_HANDLER(WM_PARENTNOTIFY, OnParentNotify)
+END_MSG_MAP()
+
+ bool Initialize() {
+ DCHECK(!automation_client_.get());
+ automation_client_.reset(CreateAutomationClient());
+ if (!automation_client_.get()) {
+ NOTREACHED() << "new ChromeFrameAutomationClient";
+ return false;
+ }
+
+ return true;
+ }
+
+ void Uninitialize() {
+ if (automation_client_.get()) {
+ automation_client_->Uninitialize();
+ automation_client_.reset();
+ }
+ }
+
+ bool InitializeAutomation(const std::wstring& profile_name,
+ const std::wstring& extra_chrome_arguments,
+ bool incognito) {
+ // We don't want to do incognito when privileged, since we're
+ // running in browser chrome or some other privileged context.
+ bool incognito_mode = !is_privileged_ && incognito;
+ return automation_client_->Initialize(this, kCommandExecutionTimeout, true,
+ profile_name, extra_chrome_arguments,
+ incognito_mode);
+ }
+
+ // ChromeFrameDelegate implementation
+ virtual WindowType GetWindow() const {
+ return (static_cast<const T*>(this))->m_hWnd;
+ }
+
+ virtual void GetBounds(RECT* bounds) {
+ if (bounds) {
+ if (::IsWindow(GetWindow())) {
+ (static_cast<T*>(this))->GetClientRect(bounds);
+ }
+ }
+ }
+ virtual std::string GetDocumentUrl() {
+ return document_url_;
+ }
+ virtual void OnAutomationServerReady() {
+ // Issue the extension automation request if we're privileged to
+ // allow this control to handle extension requests from Chrome.
+ if (is_privileged_)
+ automation_client_->SetEnableExtensionAutomation(true);
+ }
+
+ virtual bool IsValid() const {
+ return automation_client_.get() != NULL;
+ }
+
+ protected:
+ virtual void OnNavigationFailed(int tab_handle, int error_code,
+ const GURL& gurl) {
+ OnLoadFailed(error_code, gurl.spec());
+ }
+
+ virtual void OnHandleContextMenu(int tab_handle, HANDLE menu_handle,
+ int x_pos, int y_pos, int align_flags) {
+ if (!menu_handle || !automation_client_.get()) {
+ NOTREACHED();
+ return;
+ }
+
+ // TrackPopupMenuEx call will fail on IE on Vista running
+ // in low integrity mode. We DO seem to be able to enumerate the menu
+ // though, so just clone it and show the copy:
+ HMENU copy = UtilCloneContextMenu(static_cast<HMENU>(menu_handle));
+ if (!copy)
+ return;
+
+ T* pThis = static_cast<T*>(this);
+ if (pThis->PreProcessContextMenu(copy)) {
+ UINT flags = align_flags | TPM_LEFTBUTTON | TPM_RETURNCMD | TPM_RECURSE;
+ UINT selected = TrackPopupMenuEx(copy, flags, x_pos, y_pos, GetWindow(),
+ NULL);
+ if (selected != 0 && !pThis->HandleContextMenuCommand(selected)) {
+ automation_client_->SendContextMenuCommandToChromeFrame(selected);
+ }
+ }
+
+ DestroyMenu(copy);
+ }
+
+ LRESULT OnSetFocus(UINT message, WPARAM wparam, LPARAM lparam,
+ BOOL& handled) { // NO_LINT
+ if (!ignore_setfocus_ && automation_client_ != NULL) {
+ TabProxy* tab = automation_client_->tab();
+ HWND chrome_window = automation_client_->tab_window();
+ if (tab && ::IsWindow(chrome_window)) {
+ DLOG(INFO) << "Setting initial focus";
+ tab->SetInitialFocus(win_util::IsShiftPressed());
+ }
+ }
+
+ return 0;
+ }
+
+ LRESULT OnSize(UINT message, WPARAM wparam, LPARAM lparam,
+ BOOL& handled) { // NO_LINT
+ handled = FALSE;
+ // When we get resized, we need to resize the external tab window too.
+ if (automation_client_.get())
+ automation_client_->Resize(LOWORD(lparam), HIWORD(lparam),
+ SWP_NOACTIVATE | SWP_NOZORDER);
+ return 0;
+ }
+
+ LRESULT OnParentNotify(UINT message, WPARAM wparam, LPARAM lparam,
+ BOOL& handled) { // NO_LINT
+ switch (LOWORD(wparam)) {
+ case WM_LBUTTONDOWN:
+ case WM_MBUTTONDOWN:
+ case WM_RBUTTONDOWN:
+ case WM_XBUTTONDOWN: {
+ // If we got activated via mouse click on the external tab,
+ // we need to update the state of this thread and tell the
+ // browser that we now have the focus.
+ HWND focus = ::GetFocus();
+ HWND plugin_window = GetWindow();
+ if (focus != plugin_window && !IsChild(plugin_window, focus)) {
+ ignore_setfocus_ = true;
+ SetFocus(plugin_window);
+ ignore_setfocus_ = false;
+ }
+ break;
+ }
+ }
+
+ return 0;
+ }
+
+ // Return true if context menu should be displayed. The menu could be
+ // modified as well (enable/disable commands, add/remove items).
+ // Override in most-derived class if needed.
+ bool PreProcessContextMenu(HMENU menu) {
+ // Add an "About" item.
+ // TODO: The string should be localized and menu should
+ // be modified in ExternalTabContainer:: once we go public.
+ AppendMenu(menu, MF_STRING, IDC_ABOUT_CHROME_FRAME,
+ L"About Chrome Frame...");
+ return true;
+ }
+
+ // Return true if menu command is processed, otherwise the command will be
+ // passed to Chrome for execution. Override in most-derived class if needed.
+ bool HandleContextMenuCommand(UINT cmd) {
+ return false;
+ }
+
+ // Allow overriding the type of automation client used, for unit tests.
+ virtual ChromeFrameAutomationClient* CreateAutomationClient() {
+ return new ChromeFrameAutomationClient;
+ }
+
+ protected:
+ // Our gateway to chrome land
+ scoped_ptr<ChromeFrameAutomationClient> automation_client_;
+
+ // Url of the containing document.
+ std::string document_url_;
+
+ // We set this flag when we're taking the focus ourselves
+ // and notifying the host browser that we're doing so.
+ // When the flag is not set, we transfer the focus to chrome.
+ bool ignore_setfocus_;
+
+ // The plugin is privileged if it is:
+ // * Invoked by a window running under the system principal in FireFox.
+ // * Being hosted by a custom host exposing the SID_ChromeFramePrivileged
+ // service.
+ //
+ // When privileged, additional interfaces are made available to the user.
+ bool is_privileged_;
+};
+
+#endif // CHROME_FRAME_CHROME_FRAME_PLUGIN_H_
+
diff --git a/chrome_frame/chrome_frame_unittest_main.cc b/chrome_frame/chrome_frame_unittest_main.cc
new file mode 100644
index 0000000..70157a4
--- /dev/null
+++ b/chrome_frame/chrome_frame_unittest_main.cc
@@ -0,0 +1,48 @@
+// Copyright 2009, Google Inc.
+// All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include "base/at_exit.h"
+#include "base/command_line.h"
+#include "gtest/gtest.h"
+
+class ObligatoryModule: public CAtlExeModuleT<ObligatoryModule> {
+};
+
+ObligatoryModule g_obligatory_atl_module;
+
+int main(int argc, char** argv) {
+ testing::InitGoogleTest(&argc, argv);
+
+ base::AtExitManager at_exit_manager;
+ CommandLine::Init(argc, argv);
+
+ RUN_ALL_TESTS();
+}
diff --git a/chrome_frame/chrome_launcher.cc b/chrome_frame/chrome_launcher.cc
new file mode 100644
index 0000000..7670aec
--- /dev/null
+++ b/chrome_frame/chrome_launcher.cc
@@ -0,0 +1,125 @@
+// Copyright (c) 2009 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 "chrome_frame/chrome_launcher.h"
+
+#include "base/base_switches.h"
+#include "base/command_line.h"
+#include "base/file_util.h"
+#include "base/logging.h"
+#include "base/path_service.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/common/chrome_switches.h"
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/crash_report.h"
+
+namespace chrome_launcher {
+
+const wchar_t kLauncherExeBaseName[] = L"chrome_launcher.exe";
+
+// These are the switches we will allow (along with their values) in the
+// safe-for-Low-Integrity version of the Chrome command line.
+const wchar_t* kAllowedSwitches[] = {
+ switches::kAutomationClientChannelID,
+ switches::kDisableMetrics,
+ switches::kNoFirstRun,
+ switches::kUserDataDir,
+ switches::kLoadExtension,
+};
+
+CommandLine* CreateLaunchCommandLine() {
+ // TODO(joi) As optimization, could launch Chrome directly when running at
+ // medium integrity. (Requires bringing in code to read SIDs, etc.)
+
+ // The launcher EXE will be in the same directory as the npchrome_tab DLL,
+ // so create a full path to it based on this assumption. Since our unit
+ // tests also use this function, and live in the directory above, we test
+ // existence of the file and try the path that includes the /servers/
+ // directory if needed.
+ FilePath module_path;
+ if (PathService::Get(base::FILE_MODULE, &module_path)) {
+ FilePath current_dir = module_path.DirName();
+ FilePath same_dir_path = current_dir.Append(kLauncherExeBaseName);
+ if (file_util::PathExists(same_dir_path)) {
+ return new CommandLine(same_dir_path.ToWStringHack());
+ } else {
+ FilePath servers_path =
+ current_dir.Append(L"servers").Append(kLauncherExeBaseName);
+ DCHECK(file_util::PathExists(servers_path)) <<
+ "What module is this? It's not in 'servers' or main output dir.";
+ return new CommandLine(servers_path.ToWStringHack());
+ }
+ } else {
+ NOTREACHED();
+ return NULL;
+ }
+}
+
+void SanitizeCommandLine(const CommandLine& original, CommandLine* sanitized) {
+ int num_sanitized_switches = 0;
+ for (int i = 0; i < arraysize(kAllowedSwitches); ++i) {
+ const wchar_t* current_switch = kAllowedSwitches[i];
+ if (original.HasSwitch(current_switch)) {
+ ++num_sanitized_switches;
+ std::wstring switch_value = original.GetSwitchValue(current_switch);
+ if (0 == switch_value.length()) {
+ sanitized->AppendSwitch(current_switch);
+ } else {
+ sanitized->AppendSwitchWithValue(current_switch, switch_value);
+ }
+ }
+ }
+ if (num_sanitized_switches != original.GetSwitchCount()) {
+ NOTREACHED();
+ LOG(ERROR) << "Original command line from Low Integrity had switches "
+ << "that are not on our whitelist.";
+ }
+}
+
+bool SanitizeAndLaunchChrome(const wchar_t* command_line) {
+ std::wstring command_line_with_program(L"dummy.exe ");
+ command_line_with_program += command_line;
+ CommandLine original(L"");
+ original.ParseFromString(command_line_with_program);
+ CommandLine sanitized(GetChromeExecutablePath());
+ SanitizeCommandLine(original, &sanitized);
+
+ return base::LaunchApp(sanitized.command_line_string(), false, false, NULL);
+}
+
+std::wstring GetChromeExecutablePath() {
+ std::wstring cur_path;
+ PathService::Get(base::DIR_MODULE, &cur_path);
+ file_util::AppendToPath(&cur_path, chrome::kBrowserProcessExecutableName);
+
+ // The installation model for Chrome places the DLLs in a versioned
+ // sub-folder one down from the Chrome executable. If we fail to find
+ // chrome.exe in the current path, try looking one up and launching that
+ // instead.
+ if (!file_util::PathExists(cur_path)) {
+ PathService::Get(base::DIR_MODULE, &cur_path);
+ file_util::UpOneDirectory(&cur_path);
+ file_util::AppendToPath(&cur_path, chrome::kBrowserProcessExecutableName);
+ }
+
+ return cur_path;
+}
+
+} // namespace chrome_launcher
+
+// Entrypoint that implements the logic of chrome_launcher.exe.
+int CALLBACK CfLaunchChrome() {
+ if (chrome_launcher::SanitizeAndLaunchChrome(::GetCommandLine())) {
+ return ERROR_SUCCESS;
+ } else {
+ return ERROR_OPEN_FAILED;
+ }
+}
+
+// Compile-time check to see that the type CfLaunchChromeProc is correct.
+#ifndef NODEBUG
+namespace {
+chrome_launcher::CfLaunchChromeProc cf_launch_chrome = CfLaunchChrome;
+} // namespace
+#endif // NODEBUG
diff --git a/chrome_frame/chrome_launcher.h b/chrome_frame/chrome_launcher.h
new file mode 100644
index 0000000..72b198a
--- /dev/null
+++ b/chrome_frame/chrome_launcher.h
@@ -0,0 +1,45 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_LAUNCHER_H_
+#define CHROME_FRAME_CHROME_LAUNCHER_H_
+
+#include <string>
+
+class CommandLine;
+
+namespace chrome_launcher {
+
+// The base name of the chrome_launcher.exe file.
+extern const wchar_t kLauncherExeBaseName[];
+
+// Creates a command line suitable for launching Chrome. You can add any
+// flags needed before launching.
+//
+// The command-line may use the Chrome executable directly, or use an in-between
+// process if needed for security/elevation purposes. You must delete the
+// returned command line.
+CommandLine* CreateLaunchCommandLine();
+
+// Fills in a new command line from the flags on this process's command line
+// that we are allowing Low Integrity to invoke.
+//
+// Logs a warning for any flags that were passed that are not allowed to be
+// invoked by Low Integrity.
+void SanitizeCommandLine(const CommandLine& original, CommandLine* sanitized);
+
+// Given a command-line without an initial program part, launch our associated
+// chrome.exe with a sanitized version of that command line. Returns true iff
+// successful.
+bool SanitizeAndLaunchChrome(const wchar_t* command_line);
+
+// Returns the full path to the Chrome executable.
+std::wstring GetChromeExecutablePath();
+
+// The type of the CfLaunchChrome entrypoint exported from this DLL.
+typedef int (__stdcall *CfLaunchChromeProc)();
+
+} // namespace chrome_launcher
+
+#endif // CHROME_FRAME_CHROME_LAUNCHER_H_
diff --git a/chrome_frame/chrome_launcher_main.cc b/chrome_frame/chrome_launcher_main.cc
new file mode 100644
index 0000000..5347f5a
--- /dev/null
+++ b/chrome_frame/chrome_launcher_main.cc
@@ -0,0 +1,79 @@
+// Copyright (c) 2009 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 <windows.h>
+
+#include "chrome_frame/chrome_launcher.h"
+
+// We want to keep this EXE tiny, so we avoid all dependencies and link to no
+// libraries, and we do not use the C runtime.
+//
+// To catch errors in debug builds, we define an extremely simple assert macro.
+#ifndef NDEBUG
+#define CLM_ASSERT(x) do { if (!(x)) { ::DebugBreak(); } } while (false)
+#else
+#define CLM_ASSERT(x)
+#endif // NDEBUG
+
+// In release builds, we skip the standard library completely to minimize
+// size. This is more work in debug builds, and unnecessary, hence the
+// different signatures.
+#ifndef NDEBUG
+int APIENTRY wWinMain(HINSTANCE, HINSTANCE, wchar_t*, int) {
+#else
+extern "C" void __cdecl WinMainCRTStartup() {
+#endif // NDEBUG
+ // This relies on the chrome_launcher.exe residing in the same directory
+ // as our DLL. We build a full path to avoid loading it from any other
+ // directory in the DLL search path.
+ //
+ // The code is a bit verbose because we can't use the standard library.
+ const wchar_t kBaseName[] = L"npchrome_tab.dll";
+ wchar_t file_path[MAX_PATH + (sizeof(kBaseName) / sizeof(kBaseName[0])) + 1];
+ file_path[0] = L'\0';
+ ::GetModuleFileName(::GetModuleHandle(NULL), file_path, MAX_PATH);
+
+ // Find index of last slash, and null-terminate the string after it.
+ //
+ // Proof for security purposes, since we can't use the safe string
+ // manipulation functions from the runtime:
+ // - File_path is always null-terminated, by us initially and by
+ // ::GetModuleFileName if it puts anything into the buffer.
+ // - If there is no slash in the path then it's a relative path, not an
+ // absolute one, and the code ends up creating a relative path to
+ // npchrome_tab.dll.
+ // - It's safe to use lstrcatW since we know the maximum length of both
+ // parts we are concatenating, and we know the buffer will fit them in
+ // the worst case.
+ int slash_index = lstrlenW(file_path);
+ // Invariant: 0 <= slash_index < MAX_PATH
+ CLM_ASSERT(slash_index > 0);
+ while (slash_index > 0 && file_path[slash_index] != L'\\')
+ --slash_index;
+ // Invariant: 0 <= slash_index < MAX_PATH and it is either the index of
+ // the last \ in the path, or 0.
+ if (slash_index != 0)
+ ++slash_index; // don't remove the last '\'
+ file_path[slash_index] = L'\0';
+
+ lstrcatW(file_path, kBaseName);
+
+ UINT exit_code = ERROR_FILE_NOT_FOUND;
+ HMODULE chrome_tab = ::LoadLibrary(file_path);
+ CLM_ASSERT(chrome_tab);
+ if (chrome_tab) {
+ chrome_launcher::CfLaunchChromeProc proc =
+ reinterpret_cast<chrome_launcher::CfLaunchChromeProc>(
+ ::GetProcAddress(chrome_tab, "CfLaunchChrome"));
+ CLM_ASSERT(proc);
+ if (proc) {
+ exit_code = proc();
+ } else {
+ exit_code = ERROR_INVALID_FUNCTION;
+ }
+
+ ::FreeLibrary(chrome_tab);
+ }
+ ::ExitProcess(exit_code);
+}
diff --git a/chrome_frame/chrome_launcher_unittest.cc b/chrome_frame/chrome_launcher_unittest.cc
new file mode 100644
index 0000000..1181f09
--- /dev/null
+++ b/chrome_frame/chrome_launcher_unittest.cc
@@ -0,0 +1,49 @@
+// Copyright (c) 2009 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 "base/command_line.h"
+#include "base/logging.h"
+#include "chrome/common/chrome_switches.h"
+#include "chrome_frame/chrome_launcher.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+namespace {
+
+// Utility class to disable logging. Required to disable DCHECKs that some
+// of our tests would otherwise trigger.
+class LogDisabler {
+ public:
+ LogDisabler() {
+ initial_log_level_ = logging::GetMinLogLevel();
+ logging::SetMinLogLevel(logging::LOG_FATAL + 1);
+ }
+
+ ~LogDisabler() {
+ logging::SetMinLogLevel(initial_log_level_);
+ }
+
+ private:
+ int initial_log_level_;
+};
+
+} // namespace
+
+TEST(ChromeLauncher, SanitizeCommandLine) {
+ CommandLine bad(L"dummy.exe");
+ bad.AppendSwitch(switches::kDisableMetrics); // in whitelist
+ bad.AppendSwitchWithValue(switches::kLoadExtension, L"foo"); // in whitelist
+ bad.AppendSwitch(L"no-such-switch"); // does not exist
+ bad.AppendSwitch(switches::kHomePage); // exists but not in whitelist
+
+ LogDisabler no_dchecks;
+
+ CommandLine sanitized(L"dumbo.exe");
+ chrome_launcher::SanitizeCommandLine(bad, &sanitized);
+ EXPECT_TRUE(sanitized.HasSwitch(switches::kDisableMetrics));
+ EXPECT_TRUE(sanitized.HasSwitch(switches::kLoadExtension));
+ EXPECT_FALSE(sanitized.HasSwitch(L"no-such-switch"));
+ EXPECT_FALSE(sanitized.HasSwitch(switches::kHomePage));
+
+ EXPECT_EQ(sanitized.GetSwitchValue(switches::kLoadExtension), L"foo");
+}
diff --git a/chrome_frame/chrome_protocol.cc b/chrome_frame/chrome_protocol.cc
new file mode 100644
index 0000000..87ca34e
--- /dev/null
+++ b/chrome_frame/chrome_protocol.cc
@@ -0,0 +1,85 @@
+// Copyright (c) 2009 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.
+
+// Implementation of ChromeProtocol
+#include "chrome_frame/chrome_protocol.h"
+
+#include "base/logging.h"
+
+static const wchar_t* kChromeMimeType = L"application/chromepage";
+
+// ChromeProtocol
+
+// Starts the class associated with the asynchronous pluggable protocol.
+STDMETHODIMP ChromeProtocol::Start(LPCWSTR url,
+ IInternetProtocolSink* prot_sink,
+ IInternetBindInfo* bind_info,
+ DWORD flags,
+ DWORD reserved) {
+ DLOG(INFO) << __FUNCTION__ << ": URL = " << url;
+ prot_sink->ReportProgress(BINDSTATUS_VERIFIEDMIMETYPEAVAILABLE,
+ kChromeMimeType);
+ prot_sink->ReportData(
+ BSCF_FIRSTDATANOTIFICATION | BSCF_LASTDATANOTIFICATION |
+ BSCF_DATAFULLYAVAILABLE,
+ 0,
+ 0);
+ return S_OK;
+}
+
+// Allows the pluggable protocol handler to continue processing data on the
+// apartment (or user interface) thread. This method is called in response
+// to a call to IInternetProtocolSink::Switch.
+STDMETHODIMP ChromeProtocol::Continue(PROTOCOLDATA* protocol_data) {
+ DLOG(INFO) << __FUNCTION__;
+ return S_OK;
+}
+
+// Aborts an operation in progress.
+STDMETHODIMP ChromeProtocol::Abort(HRESULT reason, DWORD options) {
+ DLOG(INFO) << __FUNCTION__;
+ return S_OK;
+}
+
+STDMETHODIMP ChromeProtocol::Terminate(DWORD options) {
+ DLOG(INFO) << __FUNCTION__;
+ return S_OK;
+}
+
+STDMETHODIMP ChromeProtocol::Suspend() {
+ return E_NOTIMPL;
+}
+STDMETHODIMP ChromeProtocol::Resume() {
+ return E_NOTIMPL;
+}
+
+// Reads data retrieved by the pluggable protocol handler.
+STDMETHODIMP ChromeProtocol::Read(void* buffer,
+ ULONG buffer_size_in_bytes,
+ ULONG* bytes_read) {
+ DLOG(INFO) << __FUNCTION__;
+ return S_FALSE;
+}
+
+// Moves the current seek offset.
+STDMETHODIMP ChromeProtocol::Seek(LARGE_INTEGER move_by,
+ DWORD origin,
+ ULARGE_INTEGER* new_position) {
+ DLOG(INFO) << __FUNCTION__;
+ return E_NOTIMPL;
+}
+
+// Locks the request so that IInternetProtocolRoot::Terminate ()
+// can be called and the remaining data can be read.
+STDMETHODIMP ChromeProtocol::LockRequest(DWORD options) {
+ DLOG(INFO) << __FUNCTION__;
+ return S_OK;
+}
+
+// Frees any resources associated with a lock. Called only if
+// IInternetProtocol::LockRequest () was called.
+STDMETHODIMP ChromeProtocol::UnlockRequest() {
+ DLOG(INFO) << __FUNCTION__;
+ return S_OK;
+}
diff --git a/chrome_frame/chrome_protocol.h b/chrome_frame/chrome_protocol.h
new file mode 100644
index 0000000..751162b
--- /dev/null
+++ b/chrome_frame/chrome_protocol.h
@@ -0,0 +1,65 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_CHROME_PROTOCOL_H_
+#define CHROME_FRAME_CHROME_PROTOCOL_H_
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include "chrome_frame/resource.h"
+#include "grit/chrome_frame_resources.h"
+
+// Include without path to make GYP build see it.
+#include "chrome_tab.h" // NOLINT
+
+// ChromeProtocol
+class ATL_NO_VTABLE ChromeProtocol
+ : public CComObjectRootEx<CComSingleThreadModel>,
+ public CComCoClass<ChromeProtocol, &CLSID_ChromeProtocol>,
+ public IInternetProtocol {
+ public:
+ ChromeProtocol() {
+ }
+ DECLARE_REGISTRY_RESOURCEID(IDR_CHROMEPROTOCOL)
+
+ BEGIN_COM_MAP(ChromeProtocol)
+ COM_INTERFACE_ENTRY(IInternetProtocol)
+ COM_INTERFACE_ENTRY(IInternetProtocolRoot)
+ END_COM_MAP()
+
+ DECLARE_PROTECT_FINAL_CONSTRUCT()
+
+ HRESULT FinalConstruct() {
+ return S_OK;
+ }
+ void FinalRelease() {
+ }
+
+ public:
+ // IInternetProtocolRoot
+ STDMETHOD(Start)(LPCWSTR url,
+ IInternetProtocolSink* prot_sink,
+ IInternetBindInfo* bind_info,
+ DWORD flags,
+ DWORD reserved);
+ STDMETHOD(Continue)(PROTOCOLDATA* protocol_data);
+ STDMETHOD(Abort)(HRESULT reason, DWORD options);
+ STDMETHOD(Terminate)(DWORD options);
+ STDMETHOD(Suspend)();
+ STDMETHOD(Resume)();
+
+ // IInternetProtocol based on IInternetProtocolRoot
+ STDMETHOD(Read)(void* buffer,
+ ULONG buffer_size_in_bytes,
+ ULONG* bytes_read);
+ STDMETHOD(Seek)(LARGE_INTEGER move_by,
+ DWORD origin,
+ ULARGE_INTEGER* new_position);
+ STDMETHOD(LockRequest)(DWORD options);
+ STDMETHOD(UnlockRequest)(void);
+};
+
+OBJECT_ENTRY_AUTO(__uuidof(ChromeProtocol), ChromeProtocol)
+
+#endif // CHROME_FRAME_CHROME_PROTOCOL_H_
diff --git a/chrome_frame/chrome_protocol.rgs b/chrome_frame/chrome_protocol.rgs
new file mode 100644
index 0000000..747d7b5
--- /dev/null
+++ b/chrome_frame/chrome_protocol.rgs
@@ -0,0 +1,39 @@
+HKLM {
+ NoRemove Software {
+ NoRemove Classes {
+ ChromeTab.ChromeProtocol.1 = s 'ChromeProtocol Class' {
+ CLSID = s '{9875BFAF-B04D-445E-8A69-BE36838CDE3E}'
+ }
+ ChromeTab.ChromeProtocol = s 'ChromeProtocol Class' {
+ CLSID = s '{9875BFAF-B04D-445E-8A69-BE36838CDE3E}'
+ CurVer = s 'ChromeTab.ChromeProtocol.1'
+ }
+ NoRemove CLSID {
+ ForceRemove {9875BFAF-B04D-445E-8A69-BE36838CDE3E} = s 'ChromeProtocol Class' {
+ ProgID = s 'ChromeTab.ChromeProtocol.1'
+ VersionIndependentProgID = s 'ChromeTab.ChromeProtocol'
+ ForceRemove 'Programmable'
+ InprocServer32 = s '%MODULE%' {
+ val ThreadingModel = s 'Apartment'
+ }
+ val AppID = s '%APPID%'
+ 'TypeLib' = s '{6F2664E1-FF6E-488A-BCD1-F4CA6001DFCC}'
+ }
+ }
+ }
+ }
+}
+
+HKLM {
+ NoRemove Software {
+ NoRemove Classes {
+ NoRemove Protocols {
+ NoRemove Handler {
+ NoRemove 'cf' {
+ val CLSID = s '{9875BFAF-B04D-445E-8A69-BE36838CDE3E}'
+ }
+ }
+ }
+ }
+ }
+}
diff --git a/chrome_frame/chrome_tab.cc b/chrome_frame/chrome_tab.cc
new file mode 100644
index 0000000..1b10ff6
--- /dev/null
+++ b/chrome_frame/chrome_tab.cc
@@ -0,0 +1,308 @@
+// Copyright (c) 2009 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.
+
+// chrome_tab.cc : Implementation of DLL Exports.
+#include "base/at_exit.h"
+#include "base/command_line.h"
+#include "base/file_util.h"
+#include "base/file_version_info.h"
+#include "base/logging.h"
+#include "base/path_service.h"
+#include "base/registry.h"
+#include "base/string_piece.h"
+#include "base/string_util.h"
+#include "base/sys_string_conversions.h"
+#include "chrome/common/chrome_constants.h"
+#include "grit/chrome_frame_resources.h"
+#include "chrome_frame/bho.h"
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/chrome_launcher.h"
+#include "chrome_frame/crash_report.h"
+#include "chrome_frame/resource.h"
+#include "chrome_frame/utils.h"
+
+// Include without path to make GYP build see it.
+#include "chrome_tab.h" // NOLINT
+
+static const wchar_t kBhoRegistryPath[] =
+ L"Software\\Microsoft\\Windows\\CurrentVersion\\Explorer"
+ L"\\Browser Helper Objects";
+
+const wchar_t kBhoNoLoadExplorerValue[] = L"NoExplorer";
+
+class ChromeTabModule
+ : public AtlPerUserModule<CAtlDllModuleT<ChromeTabModule> > {
+ public:
+ typedef AtlPerUserModule<CAtlDllModuleT<ChromeTabModule> > ParentClass;
+
+ DECLARE_LIBID(LIBID_ChromeTabLib)
+ DECLARE_REGISTRY_APPID_RESOURCEID(IDR_CHROMETAB,
+ "{FD9B1B31-F4D8-436A-8F4F-D3C2E36733D3}")
+
+ // Override to add our SYSTIME binary value to registry scripts.
+ // See chrome_frame_activex.rgs for usage.
+ virtual HRESULT AddCommonRGSReplacements(IRegistrarBase* registrar) throw() {
+ HRESULT hr = ParentClass::AddCommonRGSReplacements(registrar);
+
+ if (SUCCEEDED(hr)) {
+ SYSTEMTIME local_time;
+ ::GetSystemTime(&local_time);
+ std::string hex(HexEncode(&local_time, sizeof(local_time)));
+ base::StringPiece sp_hex(hex);
+ hr = registrar->AddReplacement(L"SYSTIME",
+ base::SysNativeMBToWide(sp_hex).c_str());
+ DCHECK(SUCCEEDED(hr));
+ }
+
+ if (SUCCEEDED(hr)) {
+ std::wstring app_path(
+ chrome_launcher::GetChromeExecutablePath());
+ app_path = file_util::GetDirectoryFromPath(app_path);
+ hr = registrar->AddReplacement(L"CHROME_APPPATH", app_path.c_str());
+ DCHECK(SUCCEEDED(hr));
+ }
+
+ if (SUCCEEDED(hr)) {
+ hr = registrar->AddReplacement(L"CHROME_APPNAME",
+ chrome::kBrowserProcessExecutableName);
+ DCHECK(SUCCEEDED(hr));
+
+ // Fill in VERSION from the VERSIONINFO stored in the DLL's resources.
+ scoped_ptr<FileVersionInfo> module_version_info(
+ FileVersionInfo::CreateFileVersionInfoForCurrentModule());
+ DCHECK(module_version_info != NULL);
+ std::wstring file_version(module_version_info->file_version());
+ hr = registrar->AddReplacement(L"VERSION", file_version.c_str());
+ DCHECK(SUCCEEDED(hr));
+ }
+
+ if (SUCCEEDED(hr)) {
+ // Add the directory of chrome_launcher.exe. This will be the same
+ // as the directory for the current DLL.
+ std::wstring module_dir;
+ FilePath module_path;
+ if (PathService::Get(base::FILE_MODULE, &module_path)) {
+ module_dir = module_path.DirName().ToWStringHack();
+ } else {
+ NOTREACHED();
+ }
+ hr = registrar->AddReplacement(L"CHROME_LAUNCHER_APPPATH",
+ module_dir.c_str());
+ DCHECK(SUCCEEDED(hr));
+ }
+
+ if (SUCCEEDED(hr)) {
+ // Add the filename of chrome_launcher.exe
+ hr = registrar->AddReplacement(L"CHROME_LAUNCHER_APPNAME",
+ chrome_launcher::kLauncherExeBaseName);
+ DCHECK(SUCCEEDED(hr));
+ }
+
+ return hr;
+ }
+};
+
+ChromeTabModule _AtlModule;
+
+base::AtExitManager* g_exit_manager = NULL;
+
+// DLL Entry Point
+extern "C" BOOL WINAPI DllMain(HINSTANCE instance,
+ DWORD reason,
+ LPVOID reserved) {
+ UNREFERENCED_PARAMETER(instance);
+ if (reason == DLL_PROCESS_ATTACH) {
+#ifndef NDEBUG
+ // Silence traces from the ATL registrar to reduce the log noise.
+ ATL::CTrace::s_trace.ChangeCategory(atlTraceRegistrar, 0,
+ ATLTRACESTATUS_DISABLED);
+#endif
+ InitializeCrashReporting(false, false);
+ g_exit_manager = new base::AtExitManager();
+ CommandLine::Init(0, NULL);
+ logging::InitLogging(NULL, logging::LOG_ONLY_TO_SYSTEM_DEBUG_LOG,
+ logging::LOCK_LOG_FILE, logging::DELETE_OLD_LOG_FILE);
+ } else if (reason == DLL_PROCESS_DETACH) {
+ g_patch_helper.UnpatchIfNeeded();
+ delete g_exit_manager;
+ g_exit_manager = NULL;
+ ShutdownCrashReporting();
+ }
+ return _AtlModule.DllMain(reason, reserved);
+}
+
+#ifdef _MANAGED
+#pragma managed(pop)
+#endif
+
+const wchar_t kPostPlatformUAKey[] =
+ L"Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\"
+ L"User Agent\\Post Platform";
+const wchar_t kClockUserAgent[] = L"chromeframe";
+
+// To delete the clock user agent, set value to NULL.
+HRESULT SetClockUserAgent(const wchar_t* value) {
+ HRESULT hr;
+ RegKey ua_key;
+ if (ua_key.Create(HKEY_CURRENT_USER, kPostPlatformUAKey, KEY_WRITE)) {
+ if (value) {
+ ua_key.WriteValue(kClockUserAgent, value);
+ } else {
+ ua_key.DeleteValue(kClockUserAgent);
+ }
+ hr = S_OK;
+ } else {
+ DLOG(ERROR) << __FUNCTION__ << ": " << kPostPlatformUAKey;
+ hr = E_UNEXPECTED;
+ }
+
+ return hr;
+}
+
+HRESULT RefreshElevationPolicy() {
+ const wchar_t kIEFrameDll[] = L"ieframe.dll";
+ const char kIERefreshPolicy[] = "IERefreshElevationPolicy";
+ HRESULT hr = E_NOTIMPL;
+ HMODULE ieframe_module = LoadLibrary(kIEFrameDll);
+ if (ieframe_module) {
+ typedef HRESULT (__stdcall *IERefreshPolicy)();
+ IERefreshPolicy ie_refresh_policy = reinterpret_cast<IERefreshPolicy>(
+ GetProcAddress(ieframe_module, kIERefreshPolicy));
+
+ if (ie_refresh_policy) {
+ hr = ie_refresh_policy();
+ } else {
+ hr = HRESULT_FROM_WIN32(GetLastError());
+ }
+
+ FreeLibrary(ieframe_module);
+ } else {
+ hr = HRESULT_FROM_WIN32(GetLastError());
+ }
+
+ return hr;
+}
+
+HRESULT RegisterChromeTabBHO() {
+ RegKey ie_bho_key;
+ if (!ie_bho_key.Create(HKEY_LOCAL_MACHINE, kBhoRegistryPath,
+ KEY_CREATE_SUB_KEY)) {
+ DLOG(WARNING) << "Failed to open registry key "
+ << kBhoRegistryPath
+ << " for write";
+ return E_FAIL;
+ }
+
+ wchar_t bho_class_id_as_string[MAX_PATH] = {0};
+ StringFromGUID2(CLSID_ChromeFrameBHO, bho_class_id_as_string,
+ arraysize(bho_class_id_as_string));
+
+ if (!ie_bho_key.CreateKey(bho_class_id_as_string, KEY_READ | KEY_WRITE)) {
+ DLOG(WARNING) << "Failed to create bho registry key under "
+ << kBhoRegistryPath
+ << " for write";
+ return E_FAIL;
+ }
+
+ ie_bho_key.WriteValue(kBhoNoLoadExplorerValue, 1);
+ DLOG(INFO) << "Registered ChromeTab BHO";
+
+ SetClockUserAgent(L"1");
+ RefreshElevationPolicy();
+ return S_OK;
+}
+
+HRESULT UnregisterChromeTabBHO() {
+ SetClockUserAgent(NULL);
+
+ RegKey ie_bho_key;
+ if (!ie_bho_key.Open(HKEY_LOCAL_MACHINE, kBhoRegistryPath,
+ KEY_READ | KEY_WRITE)) {
+ DLOG(WARNING) << "Failed to open registry key "
+ << kBhoRegistryPath
+ << " for write.";
+ return E_FAIL;
+ }
+
+ wchar_t bho_class_id_as_string[MAX_PATH] = {0};
+ StringFromGUID2(CLSID_ChromeFrameBHO, bho_class_id_as_string,
+ arraysize(bho_class_id_as_string));
+
+ if (!ie_bho_key.DeleteKey(bho_class_id_as_string)) {
+ DLOG(WARNING) << "Failed to delete bho registry key "
+ << bho_class_id_as_string
+ << " under "
+ << kBhoRegistryPath;
+ return E_FAIL;
+ }
+
+ DLOG(INFO) << "Unregistered ChromeTab BHO";
+ return S_OK;
+}
+
+// Used to determine whether the DLL can be unloaded by OLE
+STDAPI DllCanUnloadNow() {
+ return _AtlModule.DllCanUnloadNow();
+}
+
+// Returns a class factory to create an object of the requested type
+STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, LPVOID* ppv) {
+ if (g_patch_helper.state() == PatchHelper::UNKNOWN) {
+ g_patch_helper.InitializeAndPatchProtocolsIfNeeded();
+ UrlMkSetSessionOption(URLMON_OPTION_USERAGENT_REFRESH, NULL, 0, 0);
+ }
+
+ return _AtlModule.DllGetClassObject(rclsid, riid, ppv);
+}
+
+// DllRegisterServer - Adds entries to the system registry
+STDAPI DllRegisterServer() {
+ // registers object, typelib and all interfaces in typelib
+ HRESULT hr = _AtlModule.DllRegisterServer(TRUE);
+
+#ifdef GOOGLE_CHROME_BUILD
+ // Muck with the Omaha configuration so that we don't get updated by non-CF
+ // Google Chrome builds.
+ UtilUpdateOmahaConfig(true);
+#endif
+
+ if (SUCCEEDED(hr)) {
+ // Best effort attempt to register the BHO. At this point we silently
+ // ignore any errors during registration. There are some traces emitted
+ // to the debug log.
+ RegisterChromeTabBHO();
+ }
+
+ return hr;
+}
+
+// DllUnregisterServer - Removes entries from the system registry
+STDAPI DllUnregisterServer() {
+ HRESULT hr = _AtlModule.DllUnregisterServer(TRUE);
+
+#ifdef GOOGLE_CHROME_BUILD
+ // Undo any prior mucking with the Omaha config.
+ UtilUpdateOmahaConfig(false);
+#endif
+
+ if (SUCCEEDED(hr)) {
+ // Best effort attempt to unregister the BHO. At this point we silently
+ // ignore any errors during unregistration. There are some traces emitted
+ // to the debug log.
+ UnregisterChromeTabBHO();
+ }
+ return hr;
+}
+
+STDAPI RegisterNPAPIPlugin() {
+ HRESULT hr = _AtlModule.UpdateRegistryFromResourceS(IDR_CHROMEFRAME_NPAPI,
+ TRUE);
+ return hr;
+}
+
+STDAPI UnregisterNPAPIPlugin() {
+ HRESULT hr = _AtlModule.UpdateRegistryFromResourceS(IDR_CHROMEFRAME_NPAPI,
+ FALSE);
+ return hr;
+}
diff --git a/chrome_frame/chrome_tab.def b/chrome_frame/chrome_tab.def
new file mode 100644
index 0000000..ae2ddd2
--- /dev/null
+++ b/chrome_frame/chrome_tab.def
@@ -0,0 +1,15 @@
+; ChromeTab.def : Declares the module parameters.
+
+LIBRARY "npchrome_tab.dll"
+
+EXPORTS
+ DllCanUnloadNow PRIVATE
+ DllGetClassObject PRIVATE
+ DllRegisterServer PRIVATE
+ DllUnregisterServer PRIVATE
+ NP_Initialize PRIVATE
+ NP_GetEntryPoints PRIVATE
+ NP_Shutdown PRIVATE
+ CfLaunchChrome PRIVATE
+ RegisterNPAPIPlugin PRIVATE
+ UnregisterNPAPIPlugin PRIVATE
diff --git a/chrome_frame/chrome_tab.idl b/chrome_frame/chrome_tab.idl
new file mode 100644
index 0000000..6f8a2ac
--- /dev/null
+++ b/chrome_frame/chrome_tab.idl
@@ -0,0 +1,153 @@
+// Copyright (c) 2009 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.
+
+// This file will be processed by the MIDL tool to
+// produce the type library (chrome_tab.tlb) and marshalling code.
+
+#include "olectl.h"
+import "oaidl.idl";
+import "ocidl.idl";
+
+[
+ object,
+ uuid(B9F5EA20-C450-4f46-B70F-BFD3CA9A20C5),
+ dual,
+ nonextensible,
+ helpstring("IChromeFrame Interface"),
+ pointer_default(unique)
+]
+interface IChromeFrame : IDispatch {
+ [propget, id(1)]
+ HRESULT src([out, retval] BSTR* src);
+ [propput, id(1)]
+ HRESULT src([in] BSTR src);
+
+ [id(3)]
+ HRESULT postMessage([in] BSTR message, [in, optional] VARIANT target);
+
+ [id(4), propget]
+ HRESULT onload([out, retval] VARIANT* onload_handler);
+ [id(4), propput]
+ HRESULT onload([in] VARIANT onload_handler);
+
+ [propget, id(5)]
+ HRESULT onloaderror([out, retval] VARIANT* onerror_handler);
+ [propput, id(5)]
+ HRESULT onloaderror([in] VARIANT onerror_handler);
+
+ [propget, id(6)]
+ HRESULT onmessage([out, retval] VARIANT* onmessage_handler);
+ [propput, id(6)]
+ HRESULT onmessage([in] VARIANT onmessage_handler);
+
+ [propget, id(DISPID_READYSTATE)]
+ HRESULT readyState([out, retval] long* ready_state);
+
+ [id(7)]
+ HRESULT addEventListener([in] BSTR event_type, [in] IDispatch* listener,
+ [in, optional] VARIANT use_capture);
+
+ [id(8)]
+ HRESULT removeEventListener([in] BSTR event_type, [in] IDispatch* listener,
+ [in, optional] VARIANT use_capture);
+
+ [propget, id(9)]
+ HRESULT version([out, retval] BSTR* version);
+
+ [id(10), hidden]
+ // This method is available only when the control is in privileged mode.
+ HRESULT postPrivateMessage([in] BSTR message,
+ [in] BSTR origin,
+ [in] BSTR target);
+
+ [propget, id(11)]
+ HRESULT useChromeNetwork([out, retval] VARIANT_BOOL* pVal);
+ [propput, id(11)]
+ HRESULT useChromeNetwork([in] VARIANT_BOOL newVal);
+};
+
+[
+ object,
+ uuid(679E292F-DBAB-46b8-8693-03084CEF61BE),
+ oleautomation,
+ nonextensible,
+ hidden,
+]
+interface IChromeFramePrivileged: IUnknown {
+ // If the host returns false for wants_privileged, the control
+ // won't enable privileged mode.
+ HRESULT GetWantsPrivileged([out] boolean *wants_privileged);
+ // Extra arguments to supply to the Chrome instance. Returns S_FALSE when
+ // no extra arguments are needed. Always sets the output string to non-NULL.
+ HRESULT GetChromeExtraArguments([out] BSTR *args);
+ // The profile name we want to use.
+ HRESULT GetChromeProfileName([out] BSTR *profile_name);
+};
+
+// Expose this service to the ChromeFrame control to trigger privileged
+// mode. If the control is in privileged mode, it will forward messages
+// to the onmessage handler irrespective of origin.
+cpp_quote("#define SID_ChromeFramePrivileged __uuidof(IChromeFramePrivileged)")
+
+typedef enum {
+ CF_EVENT_DISPID_ONLOAD = 1,
+ CF_EVENT_DISPID_ONLOADERROR,
+ CF_EVENT_DISPID_ONMESSAGE,
+ CF_EVENT_DISPID_ONPRIVATEMESSAGE,
+ CF_EVENT_DISPID_ONREADYSTATECHANGED = DISPID_READYSTATECHANGE,
+} ChromeFrameEventDispId;
+
+[
+ uuid(6F2664E1-FF6E-488A-BCD1-F4CA6001DFCC),
+ version(1.0),
+ helpstring("ChromeTab 1.0 Type Library")
+]
+library ChromeTabLib {
+ importlib("stdole2.tlb");
+
+ [uuid(A96B8A02-DD11-4936-8C0F-B2520289FABB)]
+ dispinterface DIChromeFrameEvents {
+ properties:
+ // None.
+
+ methods:
+ [id(CF_EVENT_DISPID_ONLOAD)]
+ void onload();
+ [id(CF_EVENT_DISPID_ONLOADERROR)]
+ void onloaderror();
+ [id(CF_EVENT_DISPID_ONMESSAGE)]
+ void onmessage([in] IDispatch* event);
+ [id(CF_EVENT_DISPID_ONREADYSTATECHANGED)]
+ void onreadystatechanged();
+ [id(CF_EVENT_DISPID_ONPRIVATEMESSAGE)]
+ // This event is only fired when the control is in privileged mode.
+ void onprivatemessage([in] IDispatch* event, [in] BSTR target);
+ };
+
+ [uuid(BB1176EE-20DD-41DC-9D1E-AC1335C7BBB0)]
+ coclass HtmlFilter {
+ [default] interface IUnknown;
+ };
+
+ [uuid(9875BFAF-B04D-445E-8A69-BE36838CDE3E)]
+ coclass ChromeProtocol {
+ [default] interface IUnknown;
+ };
+
+ [uuid(3E1D0E7F-F5E3-44CC-AA6A-C0A637619AB8), control]
+ coclass ChromeActiveDocument {
+ [default] interface IChromeFrame;
+ };
+
+ [uuid(E0A900DF-9611-4446-86BD-4B1D47E7DB2A), control]
+ coclass ChromeFrame {
+ [default] interface IChromeFrame;
+ [default, source] dispinterface DIChromeFrameEvents;
+ };
+
+ [uuid(ECB3C477-1A0A-44bd-BB57-78F9EFE34FA7)]
+ coclass ChromeFrameBHO {
+ [default] interface IUnknown;
+ };
+};
diff --git a/chrome_frame/chrome_tab.rgs b/chrome_frame/chrome_tab.rgs
new file mode 100644
index 0000000..ab4008b
--- /dev/null
+++ b/chrome_frame/chrome_tab.rgs
@@ -0,0 +1,12 @@
+HKLM {
+ NoRemove Software {
+ NoRemove Classes {
+ NoRemove AppID {
+ '%APPID%' = s 'ChromeTab'
+ 'ChromeTab.DLL' {
+ val AppID = s '%APPID%'
+ }
+ }
+ }
+ }
+}
diff --git a/chrome_frame/chrome_tab_version.rc.version b/chrome_frame/chrome_tab_version.rc.version
new file mode 100644
index 0000000..84c02f4
--- /dev/null
+++ b/chrome_frame/chrome_tab_version.rc.version
@@ -0,0 +1,45 @@
+/////////////////////////////////////////////////////////////////////////////
+//
+// Version
+//
+
+VS_VERSION_INFO VERSIONINFO
+ FILEVERSION @MAJOR@,@MINOR@,@BUILD@,@PATCH@
+ PRODUCTVERSION @MAJOR@,@MINOR@,@BUILD@,@PATCH@
+ FILEFLAGSMASK 0x17L
+#ifdef _DEBUG
+ FILEFLAGS 0x1L
+#else
+ FILEFLAGS 0x0L
+#endif
+ FILEOS 0x4L
+ FILETYPE 0x2L
+ FILESUBTYPE 0x0L
+BEGIN
+ BLOCK "StringFileInfo"
+ BEGIN
+ // Note that Firefox 3.0 requires the charset to be 04e4 (multi-lingual).
+ BLOCK "040904e4"
+ BEGIN
+ VALUE "CompanyName", "@COMPANY_FULLNAME@"
+ VALUE "CompanyShortName", "@COMPANY_SHORTNAME@"
+ VALUE "ProductName", "Google Chrome Frame"
+ VALUE "ProductVersion", "@MAJOR@.@MINOR@.@BUILD@.@PATCH@"
+ VALUE "FileDescription", "Chrome Frame renders the Web of the future in the browsers of the past. It's like strapping a rocket engine to a minivan."
+ VALUE "FileVersion", "@MAJOR@.@MINOR@.@BUILD@.@PATCH@"
+ VALUE "InternalName", "Google Chrome Frame"
+ VALUE "LegalCopyright", "@COPYRIGHT@"
+ VALUE "MIMEType", "application/chromeframe"
+ VALUE "FileExtents", "chromeframe"
+ VALUE "FileOpenName", "chromeframe"
+ VALUE "OriginalFilename", "npchrome_tab.dll"
+ VALUE "LastChange", "@LASTCHANGE@"
+ VALUE "Official Build", "@OFFICIAL_BUILD@"
+ END
+ END
+ BLOCK "VarFileInfo"
+ BEGIN
+ // Note that Firefox 3.0 requires the charset to be 1252 (multi-lingual).
+ VALUE "Translation", 0x409, 1252
+ END
+END
diff --git a/chrome_frame/com_message_event.cc b/chrome_frame/com_message_event.cc
new file mode 100644
index 0000000..1e25b87
--- /dev/null
+++ b/chrome_frame/com_message_event.cc
@@ -0,0 +1,157 @@
+// Copyright (c) 2009 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 "chrome_frame/com_message_event.h"
+
+#include "base/logging.h"
+#include "base/string_util.h"
+
+ComMessageEvent::ComMessageEvent() {
+}
+
+ComMessageEvent::~ComMessageEvent() {
+}
+
+bool ComMessageEvent::Initialize(IOleContainer* container,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& event_type) {
+ DCHECK(container);
+ message_ = message;
+ origin_ = origin;
+ type_ = event_type;
+
+ // May remain NULL if container not IE
+ ScopedComPtr<IHTMLEventObj> basic_event;
+ ScopedComPtr<IHTMLDocument2> doc;
+
+ // Fetching doc may fail in non-IE containers.
+ container->QueryInterface(doc.Receive());
+ if (doc) {
+ ScopedComPtr<IHTMLDocument4> doc4;
+ doc4.QueryFrom(doc);
+ DCHECK(doc4); // supported by IE5.5 and higher
+ if (doc4) {
+ // IHTMLEventObj5 is only supported in IE8 and later, so we provide our
+ // own (minimalistic) implementation of it.
+ doc4->createEventObject(NULL, basic_event.Receive());
+ DCHECK(basic_event);
+ }
+ }
+
+ basic_event_ = basic_event;
+ return true;
+}
+
+STDMETHODIMP ComMessageEvent::GetTypeInfoCount(UINT* info) {
+ // Don't DCHECK as python scripts might still call this function
+ // inadvertently.
+ DLOG(WARNING) << "Not implemented: " << __FUNCTION__;
+ return E_NOTIMPL;
+}
+
+STDMETHODIMP ComMessageEvent::GetTypeInfo(UINT which_info, LCID lcid,
+ ITypeInfo** type_info) {
+ NOTREACHED();
+ return E_NOTIMPL;
+}
+
+STDMETHODIMP ComMessageEvent::GetIDsOfNames(REFIID iid, LPOLESTR* names,
+ UINT count_names, LCID lcid,
+ DISPID* dispids) {
+ HRESULT hr = S_OK;
+
+ // Note that since we're using LowerCaseEqualsASCII for string comparison,
+ // the second argument _must_ be all lower case. I.e. we cannot compare
+ // against L"messagePort" since it has a capital 'P'.
+ for (UINT i = 0; SUCCEEDED(hr) && i < count_names; ++i) {
+ const wchar_t* name_begin = names[i];
+ const wchar_t* name_end = name_begin + wcslen(name_begin);
+ if (LowerCaseEqualsASCII(name_begin, name_end, "data")) {
+ dispids[i] = DISPID_MESSAGE_EVENT_DATA;
+ } else if (LowerCaseEqualsASCII(name_begin, name_end, "origin")) {
+ dispids[i] = DISPID_MESSAGE_EVENT_ORIGIN;
+ } else if (LowerCaseEqualsASCII(name_begin, name_end, "lasteventid")) {
+ dispids[i] = DISPID_MESSAGE_EVENT_LAST_EVENT_ID;
+ } else if (LowerCaseEqualsASCII(name_begin, name_end, "source")) {
+ dispids[i] = DISPID_MESSAGE_EVENT_SOURCE;
+ } else if (LowerCaseEqualsASCII(name_begin, name_end, "messageport")) {
+ dispids[i] = DISPID_MESSAGE_EVENT_MESSAGE_PORT;
+ } else if (LowerCaseEqualsASCII(name_begin, name_end, "type")) {
+ dispids[i] = DISPID_MESSAGE_EVENT_TYPE;
+ } else {
+ if (basic_event_) {
+ hr = basic_event_->GetIDsOfNames(IID_IDispatch, &names[i], 1, lcid,
+ &dispids[i]);
+ } else {
+ hr = DISP_E_MEMBERNOTFOUND;
+ }
+
+ if (FAILED(hr)) {
+ DLOG(WARNING) << "member not found: " << names[i]
+ << StringPrintf(L"0x%08X", hr);
+ }
+ }
+ }
+ return hr;
+}
+
+STDMETHODIMP ComMessageEvent::Invoke(DISPID dispid, REFIID iid, LCID lcid,
+ WORD flags, DISPPARAMS* params,
+ VARIANT* result, EXCEPINFO* excepinfo,
+ UINT* arg_err) {
+ HRESULT hr = DISP_E_MEMBERNOTFOUND;
+ switch (dispid) {
+ case DISPID_MESSAGE_EVENT_DATA:
+ hr = GetStringProperty(flags, UTF8ToWide(message_).c_str(), result);
+ break;
+
+ case DISPID_MESSAGE_EVENT_ORIGIN:
+ hr = GetStringProperty(flags, UTF8ToWide(origin_).c_str(), result);
+ break;
+
+ case DISPID_MESSAGE_EVENT_TYPE:
+ hr = GetStringProperty(flags, UTF8ToWide(type_).c_str(), result);
+ break;
+
+ case DISPID_MESSAGE_EVENT_LAST_EVENT_ID:
+ hr = GetStringProperty(flags, L"", result);
+ break;
+
+ case DISPID_MESSAGE_EVENT_SOURCE:
+ case DISPID_MESSAGE_EVENT_MESSAGE_PORT:
+ if (flags & DISPATCH_PROPERTYGET) {
+ result->vt = VT_NULL;
+ hr = S_OK;
+ } else {
+ hr = DISP_E_TYPEMISMATCH;
+ }
+ break;
+
+ default:
+ if (basic_event_) {
+ hr = basic_event_->Invoke(dispid, iid, lcid, flags, params, result,
+ excepinfo, arg_err);
+ }
+ break;
+ }
+
+ return hr;
+}
+
+HRESULT ComMessageEvent::GetStringProperty(WORD flags, const wchar_t* value,
+ VARIANT* result) {
+ if (!result)
+ return E_INVALIDARG;
+
+ HRESULT hr;
+ if (flags & DISPATCH_PROPERTYGET) {
+ result->vt = VT_BSTR;
+ result->bstrVal = ::SysAllocString(value);
+ hr = S_OK;
+ } else {
+ hr = DISP_E_TYPEMISMATCH;
+ }
+ return hr;
+}
diff --git a/chrome_frame/com_message_event.h b/chrome_frame/com_message_event.h
new file mode 100644
index 0000000..86fec2d
--- /dev/null
+++ b/chrome_frame/com_message_event.h
@@ -0,0 +1,75 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_COM_MESSAGE_EVENT_H_
+#define CHROME_FRAME_COM_MESSAGE_EVENT_H_
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <mshtml.h> // IHTMLEventObj
+#include "base/basictypes.h"
+#include "base/scoped_comptr_win.h"
+
+// Implements a MessageEvent compliant event object by providing MessageEvent
+// specific properties itself and inherited properties from a browser provided
+// event implementation.
+// NOTE: The messagePort and source properties will always be NULL.
+// See the HTML 5 spec for further details on a MessageEvent object:
+// http://www.whatwg.org/specs/web-apps/current-work/multipage/comms.html#messageevent
+class ComMessageEvent
+ : public CComObjectRootEx<CComSingleThreadModel>,
+ public IDispatch {
+ public:
+ ComMessageEvent();
+ ~ComMessageEvent();
+
+BEGIN_COM_MAP(ComMessageEvent)
+ COM_INTERFACE_ENTRY(IDispatch)
+END_COM_MAP()
+
+ // The dispids we support. These are based on HTML5 and not IHTMLEventObj5
+ // (there are a couple of differences).
+ // http://dev.w3.org/html5/spec/Overview.html#messageevent
+ // vs http://msdn.microsoft.com/en-us/library/cc288548(VS.85).aspx
+ typedef enum {
+ DISPID_MESSAGE_EVENT_DATA = 201,
+ DISPID_MESSAGE_EVENT_ORIGIN,
+ DISPID_MESSAGE_EVENT_LAST_EVENT_ID,
+ DISPID_MESSAGE_EVENT_SOURCE,
+ DISPID_MESSAGE_EVENT_MESSAGE_PORT,
+ DISPID_MESSAGE_EVENT_TYPE
+ } MessageEventDispIds;
+
+ // Utility function for checking Invoke flags and assigning a BSTR to the
+ // result variable.
+ HRESULT GetStringProperty(WORD flags, const wchar_t* value, VARIANT* result);
+
+ STDMETHOD(GetTypeInfoCount)(UINT* info);
+ STDMETHOD(GetTypeInfo)(UINT which_info, LCID lcid, ITypeInfo** type_info);
+ STDMETHOD(GetIDsOfNames)(REFIID iid, LPOLESTR* names, UINT count_names,
+ LCID lcid, DISPID* dispids);
+ STDMETHOD(Invoke)(DISPID dispid, REFIID iid, LCID lcid, WORD flags,
+ DISPPARAMS* params, VARIANT* result, EXCEPINFO* excepinfo,
+ UINT* arg_err);
+
+ // Initializes this object. The container pointer is used to find the
+ // container's IHTMLEventObj implementation if available.
+ bool Initialize(IOleContainer* container, const std::string& message,
+ const std::string& origin, const std::string& event_type);
+
+ protected:
+ // HTML Event object to which we delegate property and method
+ // calls that we do not directly support. This may not be initialized
+ // if our container does not require the properties/methods exposed by
+ // the basic event object.
+ ScopedComPtr<IHTMLEventObj> basic_event_;
+ std::string message_;
+ std::string origin_;
+ std::string type_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(ComMessageEvent);
+};
+
+#endif // CHROME_FRAME_COM_MESSAGE_EVENT_H_
diff --git a/chrome_frame/com_type_info_holder.cc b/chrome_frame/com_type_info_holder.cc
new file mode 100644
index 0000000..a018f81
--- /dev/null
+++ b/chrome_frame/com_type_info_holder.cc
@@ -0,0 +1,123 @@
+// Copyright (c) 2009 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 "chrome_frame/com_type_info_holder.h"
+
+#include "base/lazy_instance.h"
+#include "base/logging.h"
+
+extern "C" IMAGE_DOS_HEADER __ImageBase;
+
+namespace com_util {
+
+base::LazyInstance<TypeInfoCache> type_info_cache(base::LINKER_INITIALIZED);
+
+// TypeInfoCache
+
+TypeInfoCache::~TypeInfoCache() {
+ CacheMap::iterator it = cache_.begin();
+ while (it != cache_.end()) {
+ delete it->second;
+ it++;
+ }
+}
+
+TypeInfoNameCache* TypeInfoCache::Lookup(const IID* iid) {
+ DCHECK(Singleton() == this);
+
+ TypeInfoNameCache* tih = NULL;
+
+ AutoLock lock(lock_);
+ CacheMap::iterator it = cache_.find(iid);
+ if (it == cache_.end()) {
+ tih = new TypeInfoNameCache();
+ HRESULT hr = tih ? tih->Initialize(*iid) : E_OUTOFMEMORY;
+ if (SUCCEEDED(hr)) {
+ cache_[iid] = tih;
+ } else {
+ NOTREACHED();
+ delete tih;
+ }
+ } else {
+ tih = it->second;
+ }
+
+ return tih;
+}
+
+TypeInfoCache* TypeInfoCache::Singleton() {
+ return type_info_cache.Pointer();
+}
+
+HRESULT TypeInfoNameCache::Initialize(const IID& iid) {
+ DCHECK(type_info_ == NULL);
+
+ wchar_t file_path[MAX_PATH];
+ DWORD path_len = ::GetModuleFileNameW(reinterpret_cast<HMODULE>(&__ImageBase),
+ file_path, arraysize(file_path));
+ if (path_len == 0 || path_len == MAX_PATH) {
+ NOTREACHED();
+ return E_UNEXPECTED;
+ }
+
+ ScopedComPtr<ITypeLib> type_lib;
+ HRESULT hr = LoadTypeLib(file_path, type_lib.Receive());
+ if (SUCCEEDED(hr)) {
+ hr = type_lib->GetTypeInfoOfGuid(iid, type_info_.Receive());
+ }
+
+ return hr;
+}
+
+// TypeInfoNameCache
+
+HRESULT TypeInfoNameCache::GetIDsOfNames(OLECHAR** names, uint32 count,
+ DISPID* dispids) {
+ DCHECK(type_info_ != NULL);
+
+ HRESULT hr = S_OK;
+ for (uint32 i = 0; i < count && SUCCEEDED(hr); ++i) {
+ NameToDispIdCache::HashType hash = NameToDispIdCache::Hash(names[i]);
+ if (!cache_.Lookup(hash, &dispids[i])) {
+ hr = type_info_->GetIDsOfNames(&names[i], 1, &dispids[i]);
+ if (SUCCEEDED(hr)) {
+ cache_.Add(hash, dispids[i]);
+ }
+ }
+ }
+
+ return hr;
+}
+
+HRESULT TypeInfoNameCache::Invoke(IDispatch* p, DISPID dispid, WORD flags,
+ DISPPARAMS* params, VARIANT* result,
+ EXCEPINFO* excepinfo, UINT* arg_err) {
+ DCHECK(type_info_);
+ HRESULT hr = type_info_->Invoke(p, dispid, flags, params, result, excepinfo,
+ arg_err);
+ DCHECK(hr != RPC_E_WRONG_THREAD);
+ return hr;
+}
+
+// NameToDispIdCache
+
+bool NameToDispIdCache::Lookup(HashType hash, DISPID* dispid) const {
+ AutoLock lock(lock_);
+ const DispidMap::const_iterator it = map_.find(hash);
+ bool found = (it != map_.end());
+ if (found)
+ *dispid = it->second;
+ return found;
+}
+
+void NameToDispIdCache::Add(HashType hash, DISPID dispid) {
+ AutoLock lock(lock_);
+ map_[hash] = dispid;
+}
+
+NameToDispIdCache::HashType NameToDispIdCache::Hash(const wchar_t* name) {
+ return LHashValOfName(LANG_NEUTRAL, name);
+}
+
+} // namespace com_util
diff --git a/chrome_frame/com_type_info_holder.h b/chrome_frame/com_type_info_holder.h
new file mode 100644
index 0000000..9b0e6cf
--- /dev/null
+++ b/chrome_frame/com_type_info_holder.h
@@ -0,0 +1,191 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_COM_TYPE_INFO_HOLDER_H_
+#define CHROME_FRAME_COM_TYPE_INFO_HOLDER_H_
+
+#include <map>
+#include <ocidl.h> // IProvideClassInfo2
+
+#include "base/lock.h"
+#include "base/scoped_comptr_win.h"
+
+#define NO_VTABLE __declspec(novtable)
+
+namespace com_util {
+
+// A map from a name hash (32 bit value) to a DISPID.
+// Used as a caching layer before looking the name up in a type lib.
+class NameToDispIdCache {
+ public:
+ typedef uint32 HashType;
+
+ bool Lookup(HashType hash, DISPID* dispid) const;
+ void Add(HashType hash, DISPID dispid);
+
+ // Hashes the name by calling LHashValOfName.
+ // The returned hash value is independent of the case of the characters
+ // in |name|.
+ static HashType Hash(const wchar_t* name);
+
+ protected:
+ typedef std::map<HashType, DISPID> DispidMap;
+ DispidMap map_;
+ mutable Lock lock_;
+};
+
+// Wraps an instance of ITypeInfo and builds+maintains a cache of names
+// to dispids. Also offers an Invoke method that simply forwards the call
+// to ITypeInfo::Invoke.
+class TypeInfoNameCache {
+ public:
+ // Loads the module's type library and fetches the ITypeInfo object for
+ // the specified interface ID.
+ HRESULT Initialize(const IID& iid);
+
+ // Fetches the id's of the given names. If there's a cache miss, the results
+ // are fetched from the underlying ITypeInfo and then cached.
+ HRESULT GetIDsOfNames(OLECHAR** names, uint32 count, DISPID* dispids);
+
+ // Calls ITypeInfo::Invoke.
+ HRESULT Invoke(IDispatch* p, DISPID dispid, WORD flags, DISPPARAMS* params,
+ VARIANT* result, EXCEPINFO* excepinfo, UINT* arg_err);
+
+ inline ITypeInfo* CopyTypeInfo() {
+ ITypeInfo* ti = type_info_.get();
+ if (ti)
+ ti->AddRef();
+ return ti;
+ }
+
+ protected:
+ ScopedComPtr<ITypeInfo> type_info_;
+ NameToDispIdCache cache_;
+};
+
+// The root class for type lib access.
+// This class has only one instance that should be accessed via the
+// Singleton method.
+class TypeInfoCache {
+ public:
+ TypeInfoCache() {
+ }
+
+ ~TypeInfoCache();
+
+ // Looks up a previously cached TypeInfoNameCache instance or creates and
+ // caches a new one.
+ TypeInfoNameCache* Lookup(const IID* iid);
+
+ // Call to get access to the singleton instance of TypeInfoCache.
+ static TypeInfoCache* Singleton();
+
+ protected:
+ typedef std::map<const IID*, TypeInfoNameCache*> CacheMap;
+ Lock lock_;
+ CacheMap cache_;
+};
+
+// Holds a pointer to the type info of a given COM interface.
+// The type info is loaded once on demand and after that cached.
+// NOTE: This class only supports loading the first typelib from the
+// current module.
+template <const IID& iid>
+class TypeInfoHolder {
+ public:
+ TypeInfoHolder() : type_info_(NULL) {
+ }
+
+ bool EnsureTI() {
+ if (!type_info_)
+ type_info_ = TypeInfoCache::Singleton()->Lookup(&iid);
+ return type_info_ != NULL;
+ }
+
+ HRESULT GetTypeInfo(UINT itinfo, LCID lcid, ITypeInfo** info) {
+ if (EnsureTI()) {
+ *info = type_info_->CopyTypeInfo();
+ return S_OK;
+ }
+
+ return E_UNEXPECTED;
+ }
+
+ HRESULT GetIDsOfNames(REFIID riid, OLECHAR** names, UINT count, LCID lcid,
+ DISPID* dispids) {
+ if (!EnsureTI())
+ return E_UNEXPECTED;
+ return type_info_->GetIDsOfNames(names, count, dispids);
+ }
+
+ HRESULT Invoke(IDispatch* p, DISPID dispid, REFIID riid, LCID lcid,
+ WORD flags, DISPPARAMS* params, VARIANT* result,
+ EXCEPINFO* excepinfo, UINT* arg_err) {
+ if (!EnsureTI())
+ return E_UNEXPECTED;
+
+ return type_info_->Invoke(p, dispid, flags, params, result, excepinfo,
+ arg_err);
+ }
+
+ protected:
+ TypeInfoNameCache* type_info_;
+};
+
+// Implements IDispatch part of T (where T is an IDispatch derived interface).
+// The class assumes that the type info of T is available in a typelib of the
+// current module.
+template <class T, const IID& iid = __uuidof(T)>
+class NO_VTABLE IDispatchImpl : public T {
+ public:
+ STDMETHOD(GetTypeInfoCount)(UINT* count) {
+ if (count == NULL)
+ return E_POINTER;
+ *count = 1;
+ return S_OK;
+ }
+
+ STDMETHOD(GetTypeInfo)(UINT itinfo, LCID lcid, ITypeInfo** pptinfo) {
+ return type_info_.GetTypeInfo(itinfo, lcid, pptinfo);
+ }
+
+ STDMETHOD(GetIDsOfNames)(REFIID riid, LPOLESTR* names, UINT count,
+ LCID lcid, DISPID* dispids) {
+ return type_info_.GetIDsOfNames(riid, names, count, lcid, dispids);
+ }
+ STDMETHOD(Invoke)(DISPID dispid, REFIID riid, LCID lcid, WORD flags,
+ DISPPARAMS* params, VARIANT* result, EXCEPINFO* excepinfo,
+ UINT* arg_err) {
+ return type_info_.Invoke(static_cast<IDispatch*>(this), dispid, riid, lcid,
+ flags, params, result, excepinfo, arg_err);
+ }
+
+ protected:
+ TypeInfoHolder<iid> type_info_;
+};
+
+// Simple implementation of IProvideClassInfo[2].
+template <const CLSID& class_id, const IID& source_iid>
+class NO_VTABLE IProvideClassInfo2Impl : public IProvideClassInfo2 {
+ public:
+ STDMETHOD(GetClassInfo)(ITypeInfo** pptinfo) {
+ return type_info_.GetTypeInfo(0, LANG_NEUTRAL, pptinfo);
+ }
+
+ STDMETHOD(GetGUID)(DWORD guid_kind, GUID* guid) {
+ if(guid == NULL || guid_kind != GUIDKIND_DEFAULT_SOURCE_DISP_IID)
+ return E_INVALIDARG;
+
+ *guid = source_iid;
+
+ return S_OK;
+ }
+
+ protected:
+ TypeInfoHolder<class_id> type_info_;
+};
+
+} // namespace com_util
+
+#endif // CHROME_FRAME_COM_TYPE_INFO_HOLDER_H_
diff --git a/chrome_frame/combine_libs.py b/chrome_frame/combine_libs.py
new file mode 100644
index 0000000..97d3de7
--- /dev/null
+++ b/chrome_frame/combine_libs.py
@@ -0,0 +1,114 @@
+# Copyright (c) 2009 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.
+
+# TODO(slightlyoff): move to using shared version of this script.
+
+'''This script makes it easy to combine libs and object files to a new lib,
+optionally removing some of the object files in the input libs by regular
+expression matching.
+For usage information, run the script with a --help argument.
+'''
+import optparse
+import os
+import re
+import subprocess
+import sys
+
+
+def Shell(*args):
+ '''Runs the program and args in args, returns the output from the program.'''
+ process = subprocess.Popen(args,
+ stdin = None,
+ stdout = subprocess.PIPE,
+ stderr = subprocess.STDOUT)
+ output = process.stdout.readlines()
+ process.wait()
+ retcode = process.returncode
+ if retcode != 0:
+ raise RuntimeError('%s exited with status %d' % (args[0], retcode))
+ return output
+
+
+def CollectRemovals(remove_re, inputs):
+ '''Returns a list of all object files in inputs that match remove_re.'''
+ removals = []
+ for input in inputs:
+ output = Shell('lib.exe', '/list', input)
+
+ for line in output:
+ line = line.rstrip()
+ if remove_re.search(line):
+ removals.append(line)
+
+ return removals
+
+
+def CombineLibraries(output, remove_re, inputs):
+ '''Combines all the libraries and objects in inputs, while removing any
+ object files that match remove_re.
+ '''
+ removals = []
+ if remove_re:
+ removals = CollectRemovals(remove_re, inputs)
+
+ print removals
+
+ args = ['lib.exe', '/out:%s' % output]
+ args += ['/remove:%s' % obj for obj in removals]
+ args += inputs
+ Shell(*args)
+
+
+USAGE = '''usage: %prog [options] <lib or obj>+
+
+Combines input libraries or objects into an output library, while removing
+any object file (in the input libraries) that matches a given regular
+expression.
+'''
+
+def GetOptionParser():
+ parser = optparse.OptionParser(USAGE)
+ parser.add_option('-o', '--output', dest = 'output',
+ help = 'write to this output library')
+ parser.add_option('-r', '--remove', dest = 'remove',
+ help = 'object files matching this regexp will be removed '
+ 'from the output library')
+ return parser
+
+
+def Main():
+ '''Main function for this script'''
+ parser = GetOptionParser()
+ (opt, args) = parser.parse_args()
+ output = opt.output
+ remove = opt.remove
+ if not output:
+ parser.error('You must specify an output file')
+
+ if not args:
+ parser.error('You must specify at least one object or library')
+
+ output = output.strip()
+ remove = remove.strip()
+
+ if remove:
+ try:
+ remove_re = re.compile(opt.remove)
+ except:
+ parser.error('%s is not a valid regular expression' % opt.remove)
+ else:
+ remove_re = None
+
+ if sys.platform != 'win32':
+ parser.error('this script only works on Windows for now')
+
+ # If this is set, we can't capture lib.exe's output.
+ if 'VS_UNICODE_OUTPUT' in os.environ:
+ del os.environ['VS_UNICODE_OUTPUT']
+
+ CombineLibraries(output, remove_re, args)
+
+
+if __name__ == '__main__':
+ Main()
diff --git a/chrome_frame/common/extra_defines.vsprops b/chrome_frame/common/extra_defines.vsprops
new file mode 100644
index 0000000..ca29c05
--- /dev/null
+++ b/chrome_frame/common/extra_defines.vsprops
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="Windows-1252"?>
+<VisualStudioPropertySheet
+ ProjectType="Visual C++"
+ Version="8.00"
+ Name="extra"
+ >
+ <Tool
+ Name="VCCLCompilerTool"
+ PreprocessorDefinitions="EXCLUDE_SKIA_DEPENDENCIES"
+ AdditionalIncludeDirectories=""
+ />
+</VisualStudioPropertySheet>
diff --git a/chrome_frame/crash_report.cc b/chrome_frame/crash_report.cc
new file mode 100644
index 0000000..5a67316
--- /dev/null
+++ b/chrome_frame/crash_report.cc
@@ -0,0 +1,144 @@
+// Copyright (c) 2009 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.
+
+// crash_report.cc : Implementation crash reporting.
+#include "chrome_frame/crash_report.h"
+
+#include "base/file_util.h"
+#include "base/logging.h"
+#include "base/win_util.h"
+#include "breakpad/src/client/windows/handler/exception_handler.h"
+#include "chrome/installer/util/google_update_settings.h"
+#include "chrome/installer/util/install_util.h"
+#include "chrome_frame/vectored_handler.h"
+#include "chrome_frame/vectored_handler-impl.h"
+
+namespace {
+// TODO(joshia): factor out common code with chrome used for crash reporting
+const wchar_t kGoogleUpdatePipeName[] = L"\\\\.\\pipe\\GoogleCrashServices\\";
+const wchar_t kChromePipeName[] = L"\\\\.\\pipe\\ChromeCrashServices";
+// Well known SID for the system principal.
+const wchar_t kSystemPrincipalSid[] = L"S-1-5-18";
+google_breakpad::ExceptionHandler* g_breakpad = NULL;
+
+// Returns the custom info structure based on the dll in parameter and the
+// process type.
+google_breakpad::CustomClientInfo* GetCustomInfo() {
+ // TODO(joshia): Grab these based on build.
+ static google_breakpad::CustomInfoEntry ver_entry(L"ver", L"0.1.0.0");
+ static google_breakpad::CustomInfoEntry prod_entry(L"prod", L"ChromeFrame");
+ static google_breakpad::CustomInfoEntry plat_entry(L"plat", L"Win32");
+ static google_breakpad::CustomInfoEntry type_entry(L"ptype", L"chrome_frame");
+ static google_breakpad::CustomInfoEntry entries[] = {
+ ver_entry, prod_entry, plat_entry, type_entry };
+ static google_breakpad::CustomClientInfo custom_info = {
+ entries, arraysize(entries) };
+ return &custom_info;
+}
+
+__declspec(naked)
+static EXCEPTION_REGISTRATION_RECORD* InternalRtlpGetExceptionList() {
+ __asm {
+ mov eax, fs:0
+ ret
+ }
+}
+} // end of namespace
+
+// Class which methods simply forwards to Win32 API and uses breakpad to write
+// a minidump. Used as template (external interface) of VectoredHandlerT<E>.
+class Win32VEHTraits : public VEHTraitsBase {
+ public:
+ static inline void* Register(PVECTORED_EXCEPTION_HANDLER func,
+ const void* module_start, const void* module_end) {
+ VEHTraitsBase::SetModule(module_start, module_end);
+ return ::AddVectoredExceptionHandler(1, func);
+ }
+
+ static inline ULONG Unregister(void* handle) {
+ return ::RemoveVectoredExceptionHandler(handle);
+ }
+
+ static inline bool WriteDump(EXCEPTION_POINTERS* p) {
+ return g_breakpad->WriteMinidumpForException(p);
+ }
+
+ static inline EXCEPTION_REGISTRATION_RECORD* RtlpGetExceptionList() {
+ return InternalRtlpGetExceptionList();
+ }
+
+ static inline WORD RtlCaptureStackBackTrace(DWORD FramesToSkip,
+ DWORD FramesToCapture, void** BackTrace, DWORD* BackTraceHash) {
+ return ::RtlCaptureStackBackTrace(FramesToSkip, FramesToCapture,
+ BackTrace, BackTraceHash);
+ }
+};
+
+extern "C" IMAGE_DOS_HEADER __ImageBase;
+bool InitializeCrashReporting(bool use_crash_service, bool full_dump) {
+ if (g_breakpad)
+ return true;
+
+ std::wstring pipe_name;
+ if (use_crash_service) {
+ // Crash reporting is done by crash_service.exe.
+ pipe_name = kChromePipeName;
+ } else {
+ // We want to use the Google Update crash reporting. We need to check if the
+ // user allows it first.
+ if (!GoogleUpdateSettings::GetCollectStatsConsent())
+ return true;
+
+ // Build the pipe name. It can be either:
+ // System-wide install: "NamedPipe\GoogleCrashServices\S-1-5-18"
+ // Per-user install: "NamedPipe\GoogleCrashServices\<user SID>"
+ wchar_t dll_path[MAX_PATH * 2] = {0};
+ GetModuleFileName(reinterpret_cast<HMODULE>(&__ImageBase), dll_path,
+ arraysize(dll_path));
+
+ std::wstring user_sid;
+ if (InstallUtil::IsPerUserInstall(dll_path)) {
+ if (!win_util::GetUserSidString(&user_sid)) {
+ return false;
+ }
+ } else {
+ user_sid = kSystemPrincipalSid;
+ }
+
+ pipe_name = kGoogleUpdatePipeName;
+ pipe_name += user_sid;
+ }
+
+ // Get the alternate dump directory. We use the temp path.
+ FilePath temp_directory;
+ if (!file_util::GetTempDir(&temp_directory) || temp_directory.empty()) {
+ return false;
+ }
+
+ MINIDUMP_TYPE dump_type = full_dump ? MiniDumpWithFullMemory : MiniDumpNormal;
+ g_breakpad = new google_breakpad::ExceptionHandler(
+ temp_directory.value(), NULL, NULL, NULL,
+ google_breakpad::ExceptionHandler::HANDLER_INVALID_PARAMETER |
+ google_breakpad::ExceptionHandler::HANDLER_PURECALL, dump_type,
+ pipe_name.c_str(), GetCustomInfo());
+
+ if (g_breakpad) {
+ // Find current module boundaries.
+ const void* start = &__ImageBase;
+ const char* s = reinterpret_cast<const char*>(start);
+ const IMAGE_NT_HEADERS32* nt = reinterpret_cast<const IMAGE_NT_HEADERS32*>
+ (s + __ImageBase.e_lfanew);
+ const void* end = s + nt->OptionalHeader.SizeOfImage;
+ VectoredHandler::Register(start, end);
+ }
+
+ return g_breakpad != NULL;
+}
+
+bool ShutdownCrashReporting() {
+ VectoredHandler::Unregister();
+ delete g_breakpad;
+ g_breakpad = NULL;
+ return true;
+}
diff --git a/chrome_frame/crash_report.h b/chrome_frame/crash_report.h
new file mode 100644
index 0000000..3758e33
--- /dev/null
+++ b/chrome_frame/crash_report.h
@@ -0,0 +1,13 @@
+// Copyright (c) 2009 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.
+
+// crash_report.h : Declarations for crash reporting.
+
+#ifndef CHROME_FRAME_CRASH_REPORT_H_
+#define CHROME_FRAME_CRASH_REPORT_H_
+
+bool InitializeCrashReporting(bool use_crash_service, bool full_dump);
+bool ShutdownCrashReporting();
+
+#endif // CHROME_FRAME_CRASH_REPORT_H_
diff --git a/chrome_frame/extra_system_apis.h b/chrome_frame/extra_system_apis.h
new file mode 100644
index 0000000..0fbe0ea
--- /dev/null
+++ b/chrome_frame/extra_system_apis.h
@@ -0,0 +1,37 @@
+// Copyright (c) 2009 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.
+
+// This header file contains declarations for system APIs and interfaces
+// that are either undocumented or are documented but not included in any
+// Platform SDK header files.
+#ifndef CHROME_FRAME_EXTRA_SYSTEM_APIS_H_
+#define CHROME_FRAME_EXTRA_SYSTEM_APIS_H_
+
+// This is an interface provided by the WebBrowser object. It allows us to
+// notify the browser of navigation events. MSDN documents this interface
+// (see http://msdn2.microsoft.com/en-us/library/aa752109(VS.85).aspx)
+// but this is not included in any Platform SDK header file.
+class __declspec(uuid("54A8F188-9EBD-4795-AD16-9B4945119636"))
+IWebBrowserEventsService : public IUnknown {
+ public:
+ STDMETHOD(FireBeforeNavigate2Event)(VARIANT_BOOL *cancel) = 0;
+ STDMETHOD(FireNavigateComplete2Event)(VOID) = 0;
+ STDMETHOD(FireDownloadBeginEvent)(VOID) = 0;
+ STDMETHOD(FireDownloadCompleteEvent)(VOID) = 0;
+ STDMETHOD(FireDocumentCompleteEvent)(VOID) = 0;
+};
+
+// This interface is used in conjunction with the IWebBrowserEventsService
+// interface. The web browser queries us for this interface when we invoke
+// one of the IWebBrowserEventsService methods. This interface supplies the
+// WebBrowser with a URL to use for the events. MSDN documents this interface
+// (see http://msdn2.microsoft.com/en-us/library/aa752103(VS.85).aspx)
+// but this is not included in any Platform SDK header file.
+class __declspec(uuid("{87CC5D04-EAFA-4833-9820-8F986530CC00}"))
+IWebBrowserEventsUrlService : public IUnknown {
+ public:
+ STDMETHOD(GetUrlForEvents)(BSTR *url) = 0;
+};
+
+#endif // CHROME_FRAME_EXTRA_SYSTEM_APIS_H_
diff --git a/chrome_frame/ff_30_privilege_check.cc b/chrome_frame/ff_30_privilege_check.cc
new file mode 100644
index 0000000..e2b6369
--- /dev/null
+++ b/chrome_frame/ff_30_privilege_check.cc
@@ -0,0 +1,99 @@
+// Copyright (c) 2009 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.
+
+// This file relies on the 1.9 version of the unfrozen interfaces
+// "nsIScriptSecurityManager" and "nsIScriptObjectPrincipal"
+// from gecko 1.9, which means that this implementation is specific to
+// FireFox 3.0 and any other browsers built from the same gecko version.
+// See [http://en.wikipedia.org/wiki/Gecko_(layout_engine]
+// It's a good bet that nsIScriptSecurityManager will change for gecko
+// 1.9.1 and FireFox 3.5, in which case we'll need another instance of this
+// code for the 3.5 version of FireFox.
+
+// Gecko headers need this on Windows.
+#define XP_WIN
+#include "chrome_frame/script_security_manager.h"
+#include "third_party/xulrunner-sdk/win/include/dom/nsIScriptObjectPrincipal.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsIServiceManager.h"
+
+// These are needed to work around typedef conflicts in chrome headers.
+#define _UINT32
+#define _INT32
+
+#include "chrome_frame/np_browser_functions.h"
+#include "chrome_frame/scoped_ns_ptr_win.h"
+#include "chrome_frame/ns_associate_iid_win.h"
+#include "base/logging.h"
+
+ASSOCIATE_IID(NS_ISERVICEMANAGER_IID_STR, nsIServiceManager);
+
+namespace {
+// Unfortunately no NS_ISCRIPTOBJECTPRINCIPAL_IID_STR
+// defined for this interface
+nsIID IID_nsIScriptObjectPrincipal = NS_ISCRIPTOBJECTPRINCIPAL_IID;
+} // namespace
+
+// Returns true iff we're being instantiated into a document
+// that has the system principal's privileges
+bool IsFireFoxPrivilegedInvocation(NPP instance) {
+ ScopedNsPtr<nsIServiceManager> service_manager;
+ NPError nperr = npapi::GetValue(instance, NPNVserviceManager,
+ service_manager.Receive());
+ if (nperr != NPERR_NO_ERROR || !service_manager.get())
+ return false;
+ DCHECK(service_manager);
+
+ // Get the document.
+ ScopedNsPtr<nsISupports> window;
+ nperr = npapi::GetValue(instance, NPNVDOMWindow, window.Receive());
+ if (nperr != NPERR_NO_ERROR || !window.get())
+ return false;
+ DCHECK(window);
+
+ // This interface allows us access to the window's principal.
+ ScopedNsPtr<nsIScriptObjectPrincipal, &IID_nsIScriptObjectPrincipal>
+ script_object_principal;
+ nsresult err = script_object_principal.QueryFrom(window);
+ if (NS_FAILED(err) || !script_object_principal.get())
+ return false;
+ DCHECK(script_object_principal);
+
+ // For regular HTML windows, this will be a principal encoding the
+ // document's origin. For browser XUL, this will be the all-powerful
+ // system principal.
+ nsIPrincipal* window_principal = script_object_principal->GetPrincipal();
+ DCHECK(window_principal);
+ if (!window_principal)
+ return false;
+
+ // Get the script security manager.
+ ScopedNsPtr<nsIScriptSecurityManager_FF35> security_manager_ff35;
+ PRBool is_system = PR_FALSE;
+
+ err = service_manager->GetServiceByContractID(
+ NS_SCRIPTSECURITYMANAGER_CONTRACTID,
+ nsIScriptSecurityManager_FF35::GetIID(),
+ reinterpret_cast<void**>(security_manager_ff35.Receive()));
+ if (NS_SUCCEEDED(err) && security_manager_ff35.get()) {
+ err = security_manager_ff35->IsSystemPrincipal(window_principal,
+ &is_system);
+ if (NS_FAILED(err))
+ is_system = PR_FALSE;
+ } else {
+ ScopedNsPtr<nsIScriptSecurityManager_FF30> security_manager_ff30;
+ err = service_manager->GetServiceByContractID(
+ NS_SCRIPTSECURITYMANAGER_CONTRACTID,
+ nsIScriptSecurityManager_FF30::GetIID(),
+ reinterpret_cast<void**>(security_manager_ff30.Receive()));
+ if (NS_SUCCEEDED(err) && security_manager_ff30.get()) {
+ err = security_manager_ff30->IsSystemPrincipal(window_principal,
+ &is_system);
+ }
+
+ if (NS_FAILED(err))
+ is_system = PR_FALSE;
+ }
+
+ return is_system == PR_TRUE;
+}
diff --git a/chrome_frame/ff_privilege_check.h b/chrome_frame/ff_privilege_check.h
new file mode 100644
index 0000000..a7b5c00
--- /dev/null
+++ b/chrome_frame/ff_privilege_check.h
@@ -0,0 +1,14 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_FF_PRIVILEGE_CHECK_H_
+#define CHROME_FRAME_FF_PRIVILEGE_CHECK_H_
+
+// Returns true iff we're being invoked in a privileged document
+// in FireFox 3.x.
+// An example privileged document is when the plugin is instantiated
+// in browser chrome by XUL.
+bool IsFireFoxPrivilegedInvocation(NPP instance);
+
+#endif // CHROME_FRAME_FF_PRIVILEGE_CHECK_H_
diff --git a/chrome_frame/find_dialog.cc b/chrome_frame/find_dialog.cc
new file mode 100644
index 0000000..15aaff7
--- /dev/null
+++ b/chrome_frame/find_dialog.cc
@@ -0,0 +1,107 @@
+// Copyright (c) 2009 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 "chrome_frame/find_dialog.h"
+
+#include <Richedit.h>
+
+#include "chrome_frame/chrome_frame_automation.h"
+
+const int kMaxFindChars = 1024;
+
+HHOOK CFFindDialog::msg_hook_ = NULL;
+
+CFFindDialog::CFFindDialog() {}
+
+void CFFindDialog::Init(ChromeFrameAutomationClient* automation_client) {
+ automation_client_ = automation_client;
+}
+
+LRESULT CFFindDialog::OnDestroy(UINT msg, WPARAM wparam, LPARAM lparam,
+ BOOL& handled) {
+ UninstallMessageHook();
+ return 0;
+}
+
+LRESULT CFFindDialog::OnFind(WORD wNotifyCode, WORD wID, HWND hWndCtl,
+ BOOL& bHandled) {
+ wchar_t buffer[kMaxFindChars + 1];
+ GetDlgItemText(IDC_FIND_TEXT, buffer, kMaxFindChars);
+ std::wstring find_text(buffer);
+
+ bool match_case = IsDlgButtonChecked(IDC_MATCH_CASE) == BST_CHECKED;
+ bool search_down = IsDlgButtonChecked(IDC_DIRECTION_DOWN) == BST_CHECKED;
+
+ automation_client_->FindInPage(find_text,
+ search_down ? FWD : BACK,
+ match_case ? CASE_SENSITIVE : IGNORE_CASE,
+ false);
+
+ return 0;
+}
+
+LRESULT CFFindDialog::OnCancel(WORD wNotifyCode, WORD wID, HWND hWndCtl,
+ BOOL& bHandled) {
+ DestroyWindow();
+ return 0;
+}
+
+LRESULT CFFindDialog::OnInitDialog(UINT msg, WPARAM wparam, LPARAM lparam,
+ BOOL& handled) {
+ // Init() must be called before Create() or DoModal()!
+ DCHECK(automation_client_);
+
+ InstallMessageHook();
+ SendDlgItemMessage(IDC_FIND_TEXT, EM_EXLIMITTEXT, 0, kMaxFindChars);
+ BOOL result = CheckRadioButton(IDC_DIRECTION_DOWN, IDC_DIRECTION_UP,
+ IDC_DIRECTION_DOWN);
+
+ HWND text_field = GetDlgItem(IDC_FIND_TEXT);
+ ::SetFocus(text_field);
+
+ return FALSE; // we set the focus ourselves.
+}
+
+LRESULT CALLBACK CFFindDialog::GetMsgProc(int code, WPARAM wparam,
+ LPARAM lparam) {
+ // Mostly borrowed from http://support.microsoft.com/kb/q187988/
+ // and http://www.codeproject.com/KB/atl/cdialogmessagehook.aspx.
+ LPMSG msg = reinterpret_cast<LPMSG>(lparam);
+ if (code >= 0 && wparam == PM_REMOVE &&
+ msg->message >= WM_KEYFIRST && msg->message <= WM_KEYLAST) {
+ HWND hwnd = GetActiveWindow();
+ if (::IsWindow(hwnd) && ::IsDialogMessage(hwnd, msg)) {
+ // The value returned from this hookproc is ignored, and it cannot
+ // be used to tell Windows the message has been handled. To avoid
+ // further processing, convert the message to WM_NULL before
+ // returning.
+ msg->hwnd = NULL;
+ msg->message = WM_NULL;
+ msg->lParam = 0L;
+ msg->wParam = 0;
+ }
+ }
+
+ // Passes the hook information to the next hook procedure in
+ // the current hook chain.
+ return ::CallNextHookEx(msg_hook_, code, wparam, lparam);
+}
+
+bool CFFindDialog::InstallMessageHook() {
+ // Make sure we only call this once.
+ DCHECK(msg_hook_ == NULL);
+ msg_hook_ = ::SetWindowsHookEx(WH_GETMESSAGE, &CFFindDialog::GetMsgProc,
+ _AtlBaseModule.m_hInst, GetCurrentThreadId());
+ DCHECK(msg_hook_ != NULL);
+ return msg_hook_ != NULL;
+}
+
+bool CFFindDialog::UninstallMessageHook() {
+ DCHECK(msg_hook_ != NULL);
+ BOOL result = ::UnhookWindowsHookEx(msg_hook_);
+ DCHECK(result);
+ msg_hook_ = NULL;
+
+ return result != FALSE;
+}
diff --git a/chrome_frame/find_dialog.h b/chrome_frame/find_dialog.h
new file mode 100644
index 0000000..9e7cafe
--- /dev/null
+++ b/chrome_frame/find_dialog.h
@@ -0,0 +1,54 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_FIND_DIALOG_H_
+#define CHROME_FRAME_FIND_DIALOG_H_
+
+#include <atlbase.h>
+#include <atlwin.h>
+
+#include "base/ref_counted.h"
+#include "resource.h"
+#include "grit/chrome_frame_resources.h"
+
+class ChromeFrameAutomationClient;
+
+class CFFindDialog : public CDialogImpl<CFFindDialog> {
+ public:
+ enum { IDD = IDD_FIND_DIALOG };
+
+ BEGIN_MSG_MAP(CFFindDialog)
+ MESSAGE_HANDLER(WM_DESTROY, OnDestroy)
+ MESSAGE_HANDLER(WM_INITDIALOG, OnInitDialog)
+ COMMAND_ID_HANDLER(IDOK, OnFind)
+ COMMAND_ID_HANDLER(IDCANCEL, OnCancel)
+ END_MSG_MAP()
+
+ CFFindDialog();
+ void Init(ChromeFrameAutomationClient* automation_client);
+
+ LRESULT OnDestroy(UINT msg, WPARAM wparam,
+ LPARAM lparam, BOOL& handled); // NOLINT
+ LRESULT OnFind(WORD wNotifyCode, WORD wID,
+ HWND hWndCtl, BOOL& bHandled); // NOLINT
+ LRESULT OnCancel(WORD wNotifyCode, WORD wID,
+ HWND hWndCtl, BOOL& bHandled); // NOLINT
+ LRESULT OnInitDialog(UINT msg, WPARAM wparam,
+ LPARAM lparam, BOOL& handled); // NOLINT
+
+ private:
+
+ // Since the message loop we expect to run in isn't going to be nicely
+ // calling IsDialogMessage(), we need to hook the wnd proc and call it
+ // ourselves. See http://support.microsoft.com/kb/q187988/
+ bool InstallMessageHook();
+ bool UninstallMessageHook();
+ static LRESULT CALLBACK GetMsgProc(int code, WPARAM wparam, LPARAM lparam);
+ static HHOOK msg_hook_;
+
+ // We don't own these, and they must exist at least as long as we do.
+ ChromeFrameAutomationClient* automation_client_;
+};
+
+#endif // CHROME_FRAME_FIND_DIALOG_H_
diff --git a/chrome_frame/frame.html b/chrome_frame/frame.html
new file mode 100644
index 0000000..d63d3ac
--- /dev/null
+++ b/chrome_frame/frame.html
@@ -0,0 +1,20 @@
+<html>
+<!-- TODO(slightlyoff): Move to tests directory? -->
+<head>
+<title>Script test</title>
+<script>
+function OnLoad() {
+ var host = window.externalHost;
+ host.onmessage = OnHostMessage;
+ host.ForwardMessageToExternalHost("Hello from ChromeFrame");
+}
+
+function OnHostMessage(text) {
+ window.alert("In ChromeFrame: \r\n Message from host: " + text);
+}
+</script>
+</head>
+<body onload="OnLoad();">
+Test script
+</body>
+</html>
diff --git a/chrome_frame/frame_w_controls.html b/chrome_frame/frame_w_controls.html
new file mode 100644
index 0000000..d80642f
--- /dev/null
+++ b/chrome_frame/frame_w_controls.html
@@ -0,0 +1,29 @@
+<html>
+<!-- TODO(slightlyoff): Move to tests directory? -->
+<head>
+<title>Script test</title>
+<script>
+function msg(txt) {
+ window.document.getElementById("my_text").innerText = txt;
+}
+
+function OnLoad() {
+ var host = window.externalHost;
+ host.ForwardMessageToExternalHost("OnChromeFrameMessage",
+ "Hello from ChromeFrame");
+}
+
+function OnHostMessage(text) {
+ msg("In ChromeFrame: \r\n Message from host: " + text);
+}
+</script>
+</head>
+<body onload="OnLoad();">
+Here's an edit field: <input type="text"><br>
+Here's another: <input type="text"><br>
+<p>
+Message:<br>
+<pre><div id="my_text"></div></pre>
+</p>
+</body>
+</html>
diff --git a/chrome_frame/function_stub.h b/chrome_frame/function_stub.h
new file mode 100644
index 0000000..55f0c5e
--- /dev/null
+++ b/chrome_frame/function_stub.h
@@ -0,0 +1,241 @@
+// Copyright (c) 2009 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.
+
+
+#ifndef CHROME_FRAME_FUNCTION_STUB_H_
+#define CHROME_FRAME_FUNCTION_STUB_H_
+
+#include <windows.h>
+#include "base/logging.h"
+
+// IMPORTANT: The struct below must be byte aligned.
+#pragma pack(push)
+#pragma pack(1)
+
+#ifndef _M_IX86
+#error Only x86 supported right now.
+#endif
+
+extern "C" IMAGE_DOS_HEADER __ImageBase;
+
+// This struct is assembly code + signature. The purpose of the struct is to be
+// able to hook an existing function with our own and store information such
+// as the original function pointer with the code stub. Typically this is used
+// for patching entries of a vtable or e.g. a globally registered wndproc
+// for a class as opposed to a window.
+// When unhooking, you can just call the BypassStub() function and leave the
+// stub in memory. This unhooks your function while leaving the (potential)
+// chain of patches intact.
+//
+// @note: This class is meant for __stdcall calling convention and
+// it uses eax as a temporary variable. The struct can
+// be improved in the future to save eax before the
+// operation and then restore it.
+//
+// For instance if the function prototype is:
+//
+// @code
+// LRESULT WndProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam);
+// @endcode
+//
+// and we would like to add one static argument to make it, say:
+//
+// @code
+// LRESULT MyNewWndProc(WNDPROC original, HWND hwnd, UINT msg,
+// WPARAM wparam, LPARAM lparam);
+// @endcode
+//
+// That can be achieved by wrapping the function up with a FunctionStub:
+//
+// @code
+// FunctionStub* stub = FunctionStub::Create(original_wndproc, MyNewWndProc);
+// SetClassLongPtr(wnd, GCLP_WNDPROC, stub->code());
+// @endcode
+struct FunctionStub {
+ private:
+ typedef enum AsmConstants {
+ POP_EAX = 0x58,
+ PUSH = 0x68,
+ PUSH_EAX = 0x50,
+ JUMP_RELATIVE = 0xE9
+ };
+
+ FunctionStub(uintptr_t extra_argument, void* dest)
+ : signature_(reinterpret_cast<HMODULE>(&__ImageBase)) {
+ Opcodes::Hook& hook = code_.hook_;
+ hook.pop_return_addr_ = POP_EAX;
+ hook.push_ = PUSH;
+ hook.arg_ = extra_argument;
+ hook.push_return_addr_ = PUSH_EAX;
+ hook.jump_to_ = JUMP_RELATIVE;
+
+ // Calculate the target jump to the destination function.
+ hook.target_ = CalculateRelativeJump(dest, &hook.jump_to_);
+
+ // Allow the process to execute this struct as code.
+ DWORD old_protect = 0;
+ // Allow reads too since we want read-only member variable access at
+ // all times.
+ ::VirtualProtect(this, sizeof(FunctionStub), PAGE_EXECUTE_READ,
+ &old_protect);
+ }
+
+ ~FunctionStub() {
+ // No more execution rights.
+ DWORD old_protect = 0;
+ ::VirtualProtect(this, sizeof(FunctionStub), PAGE_READWRITE, &old_protect);
+ }
+
+ // Calculates the target value for a relative jump.
+ // The function assumes that the size of the opcode is 1 byte.
+ inline uintptr_t CalculateRelativeJump(const void* absolute_to,
+ const void* absolute_from) const {
+ return (reinterpret_cast<uintptr_t>(absolute_to) -
+ reinterpret_cast<uintptr_t>(absolute_from)) -
+ (sizeof(uint8) + sizeof(uintptr_t));
+ }
+
+ // Does the opposite of what CalculateRelativeJump does, which is to
+ // calculate back the absolute address that previously was relative to
+ // some other address.
+ inline uintptr_t CalculateAbsoluteAddress(const void* relative_to,
+ uintptr_t relative_address) const {
+ return relative_address + sizeof(uint8) + sizeof(uintptr_t) +
+ reinterpret_cast<uintptr_t>(relative_to);
+ }
+
+ // Used to identify function stubs that belong to this module.
+ HMODULE signature_;
+
+ // IMPORTANT: Do not change the order of member variables
+ union Opcodes {
+ // Use this struct when the stub forwards the call to our hook function
+ // providing an extra argument.
+ struct Hook {
+ uint8 pop_return_addr_; // pop eax
+ uint8 push_; // push arg ; push...
+ uintptr_t arg_; // ; extra argument
+ uint8 push_return_addr_; // push eax ; push the return address
+ uint8 jump_to_; // jmp ; jump...
+ uintptr_t target_; // ; to the hook function
+ } hook_;
+ // When the stub is bypassed, we jump directly to a given target,
+ // usually the originally hooked function.
+ struct Bypassed {
+ uint8 jump_to_; // jmp to
+ uintptr_t target_; // relative target.
+ } bypassed_;
+ };
+
+ Opcodes code_;
+
+ public:
+ // Neutralizes this stub and converts it to a direct jump to a new target.
+ void BypassStub(void* new_target) {
+ DWORD old_protect = 0;
+ // Temporarily allow us to write to member variables
+ ::VirtualProtect(this, sizeof(FunctionStub), PAGE_EXECUTE_READWRITE,
+ &old_protect);
+
+ // Now, just change the first 5 bytes to jump directly to the new target.
+ Opcodes::Bypassed& bypassed = code_.bypassed_;
+ bypassed.jump_to_ = JUMP_RELATIVE;
+ bypassed.target_ = CalculateRelativeJump(new_target, &bypassed.jump_to_);
+
+ // Restore the previous protection flags.
+ ::VirtualProtect(this, sizeof(FunctionStub), old_protect, &old_protect);
+
+ // Flush the instruction cache just in case.
+ ::FlushInstructionCache(::GetCurrentProcess(), this, sizeof(FunctionStub));
+ }
+
+ // @returns the argument supplied in the call to @ref Create
+ inline uintptr_t argument() const {
+ DCHECK(code_.hook_.pop_return_addr_ == POP_EAX) << "stub has been disabled";
+ return code_.hook_.arg_;
+ }
+
+ inline bool is_bypassed() const {
+ return code_.bypassed_.jump_to_ == JUMP_RELATIVE;
+ }
+
+ inline uintptr_t absolute_target() const {
+ DCHECK(code_.hook_.pop_return_addr_ == POP_EAX) << "stub has been disabled";
+ return CalculateAbsoluteAddress(
+ reinterpret_cast<const void*>(&code_.hook_.jump_to_),
+ code_.hook_.target_);
+ }
+
+ // Returns true if the stub is valid and enabled.
+ // Don't call this method after bypassing the stub.
+ inline bool is_valid() const {
+ return signature_ == reinterpret_cast<HMODULE>(&__ImageBase) &&
+ code_.hook_.pop_return_addr_ == POP_EAX;
+ }
+
+ inline PROC code() const {
+ return reinterpret_cast<PROC>(const_cast<Opcodes*>(&code_));
+ }
+
+ // Use to create a new function stub as shown above.
+ //
+ // @param extra_argument The static argument to pass to the function.
+ // @param dest Target function to which the stub applies.
+ // @returns NULL if an error occurs, otherwise a pointer to the
+ // function stub.
+ //
+ // TODO(tommi): Change this so that VirtualAlloc isn't called for
+ // every stub. Instead we should re-use each allocation for
+ // multiple stubs. In practice I'm guessing that there would
+ // only be one allocation per process, since each allocation actually
+ // allocates at least one page of memory (4K). Size of FunctionStub
+ // is 12 bytes, so one page could house 341 function stubs.
+ // When stubs are created frequently, VirtualAlloc could fail
+ // and last error is ERROR_NOT_ENOUGH_MEMORY (8).
+ static FunctionStub* Create(uintptr_t extra_argument, void* dest) {
+ DCHECK(dest);
+
+ // Use VirtualAlloc to get memory for the stub. This gives us a
+ // whole page that we don't share with anyone else.
+ // Initially the memory must be READWRITE to allow for construction
+ // PAGE_EXECUTE is set in constructor.
+ FunctionStub* ret = reinterpret_cast<FunctionStub*>(VirtualAlloc(NULL,
+ sizeof(FunctionStub), MEM_RESERVE | MEM_COMMIT, PAGE_READWRITE));
+
+ if (!ret) {
+ NOTREACHED();
+ } else {
+ // Construct
+ ret->FunctionStub::FunctionStub(extra_argument, dest);
+ }
+
+ return ret;
+ }
+
+ static FunctionStub* FromCode(void* address) {
+ Opcodes* code = reinterpret_cast<Opcodes*>(address);
+ if (code->hook_.pop_return_addr_ == POP_EAX) {
+ FunctionStub* stub = reinterpret_cast<FunctionStub*>(
+ reinterpret_cast<uint8*>(address) - sizeof(HMODULE));
+ if (stub->is_valid())
+ return stub;
+ }
+
+ return NULL;
+ }
+
+ // Deallocates a FunctionStub. The stub must not be in use on any thread!
+ static bool Destroy(FunctionStub* stub) {
+ DCHECK(stub != NULL);
+ FunctionStub* to_free = reinterpret_cast<FunctionStub*>(stub);
+ to_free->FunctionStub::~FunctionStub();
+ BOOL success = VirtualFree(to_free, sizeof(FunctionStub), MEM_DECOMMIT);
+ DCHECK(success) << "VirtualFree";
+ return success != FALSE;
+ }
+};
+
+#pragma pack(pop)
+
+#endif // CHROME_FRAME_FUNCTION_STUB_H_
diff --git a/chrome_frame/host.html b/chrome_frame/host.html
new file mode 100644
index 0000000..a3da73c
--- /dev/null
+++ b/chrome_frame/host.html
@@ -0,0 +1,47 @@
+<HTML>
+<!-- TODO(slightlyoff): Move to tests directory? -->
+<HEAD>
+<TITLE> Chrome Frame Test </TITLE>
+<SCRIPT type="text/javascript">
+function GetChromeFrame() {
+ var chromeFrame = window.document.ChromeFrame
+ return chromeFrame;
+}
+
+function OnChromeFrameMessage(text) {
+ window.alert("In host: \r\nMessage from ChromeFrame: " + text);
+
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.PostMessageToFrame("Hello from host");
+ return "OK";
+}
+
+function OnNavigate() {
+ var url = document.getElementById('inputurl');
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.src = url.value;
+}
+
+function onLoad() {
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.onmessage = OnChromeFrameMessage;
+}
+
+</SCRIPT>
+</HEAD>
+<BODY onload="onLoad();">
+Chrome Frame Test activex
+<br><br>
+<input id="inputurl" type="text" name="URL">
+<input type="submit" value="Navigate" onClick="OnNavigate();">
+<center>
+<OBJECT ID="ChromeFrame" WIDTH=500 HEIGHT=500 CODEBASE="http://www.google.com"
+ CLASSID="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <PARAM NAME="src" VALUE="http://www.google.com">
+ <embed ID="ChromeFramePlugin" WIDTH=500 HEIGHT=500 NAME="ChromeFrame"
+ SRC="http://www.google.com" TYPE="application/chromeframe">
+ </embed>
+</OBJECT>
+</center>
+</BODY>
+</HTML>
diff --git a/chrome_frame/host_w_controls.html b/chrome_frame/host_w_controls.html
new file mode 100644
index 0000000..0e44fc5
--- /dev/null
+++ b/chrome_frame/host_w_controls.html
@@ -0,0 +1,97 @@
+<HTML>
+<!-- TODO(slightlyoff): Move to tests directory? -->
+<HEAD>
+<TITLE> Chrome Frame Test </TITLE>
+<SCRIPT type="text/javascript">
+function msg(txt) {
+ try {
+ document.getElementById("my_text").innerHTML = txt;
+ } catch(e) {
+ alert("error");
+ }
+}
+
+function GetChromeFrame() {
+ var chromeFrame = window.document.ChromeFrame
+ return chromeFrame;
+}
+
+function OnChromeFrameMessage(text) {
+ msg("In host: \r\nMessage from ChromeFrame: " + text);
+
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.PostMessageToFrame("OnHostMessage", "Hello from host");
+ return "OK";
+}
+
+function OnNavigate() {
+ var url = document.getElementById('inputurl');
+ GetChromeFrame().src = url.value;
+}
+
+function OnFocus() {
+ msg("OnFocus");
+}
+
+window.onload = function() {
+ var chromeFrame = GetChromeFrame();
+ var url = location.href;
+ url = url.substr(0, url.lastIndexOf('/') + 1) + "frame_w_controls.html";
+ chromeFrame.src = url;
+
+ try {
+ var cf = document.getElementById('ChromeFrame');
+ cf.addEventListener("focus", OnFocus, true);
+ cf.addEventListener("blur", function() { msg('blur'); }, true);
+ msg("ready");
+ } catch(e) {
+ alert("error");
+ }
+}
+
+function setFocusToCf() {
+ var cf = document.getElementById('ChromeFrame');
+ cf.focus();
+ // alert(cf.hasFocus());
+ return true;
+}
+
+</SCRIPT>
+<style>
+/* CSS magic to avoid the focus rect */
+object:focus {
+ outline: 0;
+}
+</style>
+<!--
+object:focus { outline: none; }
+:focus { outline: none }
+a:focus { outline: 1px dotted invert }
+-->
+</HEAD>
+<BODY>
+Chrome Frame Test activex
+<br><br>
+<input id="inputurl" type="text" name="URL">
+<input type="submit" value="Navigate" onClick="OnNavigate();">
+<center>
+<OBJECT ID="ChromeFrame" tabindex="0"
+ WIDTH="500"
+ HEIGHT="300"
+ CODEBASE="http://www.google.com"
+ CLASSID="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <!-- <PARAM NAME="BackColor" VALUE="100"> -->
+ <!-- <PARAM NAME="src" VALUE="file:///z:/code/debug/test.html"> -->
+ <embed ID="ChromeFramePlugin" WIDTH=500 HEIGHT=300 NAME="ChromeFrame"
+ SRC="http://www.google.com" TYPE="application/chromeframe">
+ </embed>
+</OBJECT>
+<p>To test the focus: <input id="fake_edit" type="text" name="fake"></p>
+<p><button onclick="return setFocusToCf();">SetFocusToCF</button></p>
+<p>
+Message:<br>
+<pre><p id="my_text"></p></pre>
+</p>
+</center>
+</BODY>
+</HTML>
diff --git a/chrome_frame/html_utils.cc b/chrome_frame/html_utils.cc
new file mode 100644
index 0000000..60fdd7e
--- /dev/null
+++ b/chrome_frame/html_utils.cc
@@ -0,0 +1,281 @@
+// Copyright (c) 2009 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 "chrome_frame/html_utils.h"
+
+#include "base/string_util.h"
+#include "base/string_tokenizer.h"
+
+const wchar_t* kQuotes = L"\"'";
+
+HTMLScanner::StringRange::StringRange() {
+}
+
+HTMLScanner::StringRange::StringRange(StrPos start, StrPos end)
+ : start_(start), end_(end) {
+}
+
+bool HTMLScanner::StringRange::LowerCaseEqualsASCII(const char* other) const {
+ return ::LowerCaseEqualsASCII(start_, end_, other);
+}
+
+bool HTMLScanner::StringRange::Equals(const wchar_t* other) const {
+ int ret = wcsncmp(&start_[0], other, end_ - start_);
+ if (ret == 0)
+ ret = (other[end_ - start_] == L'\0') ? 0 : -1;
+ return ret == 0;
+}
+
+std::wstring HTMLScanner::StringRange::Copy() const {
+ return std::wstring(start_, end_);
+}
+
+bool HTMLScanner::StringRange::GetTagName(std::wstring* tag_name) const {
+ if (*start_ != L'<') {
+ LOG(ERROR) << "Badly formatted tag found";
+ return false;
+ }
+
+ StrPos name_start = start_;
+ name_start++;
+ while (name_start < end_ && IsWhitespace(*name_start))
+ name_start++;
+
+ if (name_start >= end_) {
+ // We seem to have a degenerate tag (i.e. < >). Return false here.
+ return false;
+ }
+
+ StrPos name_end = name_start + 1;
+ while (name_end < end_ && !IsWhitespace(*name_end))
+ name_end++;
+
+ if (name_end > end_) {
+ // This looks like an improperly formatted tab ('<foo'). Return false here.
+ return false;
+ }
+
+ tag_name->assign(name_start, name_end);
+ return true;
+}
+
+
+bool HTMLScanner::StringRange::GetTagAttribute(const wchar_t* attribute_name,
+ StringRange* attribute_value) const {
+ if (NULL == attribute_name || NULL == attribute_value) {
+ NOTREACHED();
+ return false;
+ }
+
+ // Use this so we can use the convenience method LowerCaseEqualsASCII()
+ // from string_util.h.
+ std::string search_name_ascii(WideToASCII(attribute_name));
+
+ WStringTokenizer tokenizer(start_, end_, L" =/");
+ tokenizer.set_options(WStringTokenizer::RETURN_DELIMS);
+
+ // Set up the quote chars so that we get quoted attribute values as single
+ // tokens.
+ tokenizer.set_quote_chars(L"\"'");
+
+ const bool PARSE_STATE_NAME = true;
+ const bool PARSE_STATE_VALUE = false;
+ bool parse_state = PARSE_STATE_NAME;
+
+ // Used to skip the first token, which is the tag name.
+ bool first_token_skipped = false;
+
+ // This is set during a loop iteration in which an '=' sign was spotted.
+ // It is used to filter out degenerate tags such as:
+ // <meta foo==bar>
+ bool last_token_was_delim = false;
+
+ // Set this if the attribute name has been found that we might then
+ // pick up the value in the next loop iteration.
+ bool attribute_name_found = false;
+
+ while (tokenizer.GetNext()) {
+ // If we have a whitespace delimiter, just keep going. Cases of this should
+ // be reduced by the CollapseWhitespace call. If we have an '=' character,
+ // we update our state and reiterate.
+ if (tokenizer.token_is_delim()) {
+ if (*tokenizer.token_begin() == L'=') {
+ if (last_token_was_delim) {
+ // Looks like we have a badly formed tag, just stop parsing now.
+ return false;
+ }
+ parse_state = !parse_state;
+ last_token_was_delim = true;
+ }
+ continue;
+ }
+
+ last_token_was_delim = false;
+
+ // The first non-delimiter token is the tag name, which we don't want.
+ if (!first_token_skipped) {
+ first_token_skipped = true;
+ continue;
+ }
+
+ if (PARSE_STATE_NAME == parse_state) {
+ // We have a tag name, check to see if it matches our target name:
+ if (::LowerCaseEqualsASCII(tokenizer.token_begin(), tokenizer.token_end(),
+ search_name_ascii.c_str())) {
+ attribute_name_found = true;
+ continue;
+ }
+ } else if (PARSE_STATE_VALUE == parse_state && attribute_name_found) {
+ attribute_value->start_ = tokenizer.token_begin();
+ attribute_value->end_ = tokenizer.token_end();
+
+ // Unquote the attribute value if need be.
+ attribute_value->UnQuote();
+
+ return true;
+ } else if (PARSE_STATE_VALUE == parse_state) {
+ // If we haven't found the attribute name we want yet, ignore this token
+ // and go back to looking for our name.
+ parse_state = PARSE_STATE_NAME;
+ }
+ }
+
+ return false;
+}
+
+bool HTMLScanner::StringRange::UnQuote() {
+ if (start_ + 2 > end_) {
+ // String's too short to be quoted, bail.
+ return false;
+ }
+
+ if ((*start_ == L'\'' && *(end_ - 1) == L'\'') ||
+ (*start_ == L'"' && *(end_ - 1) == L'"')) {
+ start_ = start_ + 1;
+ end_ = end_ - 1;
+ return true;
+ }
+
+ return false;
+}
+
+HTMLScanner::HTMLScanner(const wchar_t* html_string)
+ : html_string_(CollapseWhitespace(html_string, true)),
+ quotes_(kQuotes) {
+}
+
+void HTMLScanner::GetTagsByName(const wchar_t* name, StringRangeList* tag_list,
+ const wchar_t* stop_tag) {
+ DCHECK(NULL != name);
+ DCHECK(NULL != tag_list);
+ DCHECK(NULL != stop_tag);
+
+ StringRange remaining_html(html_string_.begin(), html_string_.end());
+
+ std::wstring search_name(name);
+ TrimWhitespace(search_name, TRIM_ALL, &search_name);
+
+ // Use this so we can use the convenience method LowerCaseEqualsASCII()
+ // from string_util.h.
+ std::string search_name_ascii(WideToASCII(search_name));
+ std::string stop_tag_ascii(WideToASCII(stop_tag));
+
+ StringRange current_tag;
+ std::wstring current_name;
+ while (NextTag(&remaining_html, &current_tag)) {
+ if (current_tag.GetTagName(&current_name)) {
+ if (LowerCaseEqualsASCII(current_name, search_name_ascii.c_str())) {
+ tag_list->push_back(current_tag);
+ } else if (LowerCaseEqualsASCII(current_name, stop_tag_ascii.c_str())) {
+ // We hit the stop tag so it's time to go home.
+ break;
+ }
+ }
+ }
+}
+
+struct ScanState {
+ bool in_quote;
+ bool in_escape;
+ wchar_t quote_char;
+ ScanState() : in_quote(false), in_escape(false) {}
+};
+
+bool HTMLScanner::IsQuote(wchar_t c) {
+ return quotes_.find(c) != std::wstring::npos;
+}
+
+bool HTMLScanner::IsHTMLCommentClose(StringRange* html_string, StrPos pos) {
+ if (pos < html_string->end_ && pos > html_string->start_ + 2 &&
+ *pos == L'>') {
+ return *(pos-1) == L'-' && *(pos-2) == L'-';
+ }
+ return false;
+}
+
+bool HTMLScanner::NextTag(StringRange* html_string, StringRange* tag) {
+ DCHECK(NULL != html_string);
+ DCHECK(NULL != tag);
+
+ tag->start_ = html_string->start_;
+ while (tag->start_ < html_string->end_ && *tag->start_ != L'<') {
+ tag->start_++;
+ }
+
+ // we went past the end of the string.
+ if (tag->start_ >= html_string->end_) {
+ return false;
+ }
+
+ tag->end_ = tag->start_ + 1;
+
+ // Get the tag name to see if we are in an HTML comment. If we are, then
+ // don't consider quotes. This should work for example:
+ // <!-- foo ' --> <meta foo='bar'>
+ std::wstring tag_name;
+ StringRange start_range(tag->start_, html_string->end_);
+ start_range.GetTagName(&tag_name);
+ if (StartsWith(tag_name, L"!--", true)) {
+ // We're inside a comment tag, keep going until we get out of it.
+ while (tag->end_ < html_string->end_ &&
+ !IsHTMLCommentClose(html_string, tag->end_)) {
+ tag->end_++;
+ }
+ } else {
+ // Properly handle quoted strings within non-comment tags by maintaining
+ // some state while scanning. Specifically, we have to maintain state on
+ // whether we are inside a string, what the string terminating character
+ // will be and whether we are inside an escape sequence.
+ ScanState state;
+ while (tag->end_ < html_string->end_) {
+ if (state.in_quote) {
+ if (state.in_escape) {
+ state.in_escape = false;
+ } else if (*tag->end_ == '\\') {
+ state.in_escape = true;
+ } else if (*tag->end_ == state.quote_char) {
+ state.in_quote = false;
+ }
+ } else {
+ state.in_quote = IsQuote(state.quote_char = *tag->end_);
+ }
+
+ if (!state.in_quote && *tag->end_ == L'>') {
+ break;
+ }
+ tag->end_++;
+ }
+ }
+
+ // We hit the end_ but found no matching tag closure. Consider this an
+ // incomplete tag and do not report it.
+ if (tag->end_ >= html_string->end_)
+ return false;
+
+ // Modify html_string to point to just beyond the end_ of the current tag.
+ html_string->start_ = tag->end_ + 1;
+
+ return true;
+}
+
diff --git a/chrome_frame/html_utils.h b/chrome_frame/html_utils.h
new file mode 100644
index 0000000..c670b00
--- /dev/null
+++ b/chrome_frame/html_utils.h
@@ -0,0 +1,120 @@
+// Copyright (c) 2009 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.
+
+// This file defines utility functions for working with html.
+
+#ifndef CHROME_FRAME_HTML_UTILS_H_
+#define CHROME_FRAME_HTML_UTILS_H_
+
+#include <string>
+#include <vector>
+
+#include "base/basictypes.h"
+#include "testing/gtest/include/gtest/gtest_prod.h"
+
+// Forward declarations
+class HtmlUtilUnittest;
+
+//
+// Class designed to take a string of HTML and extract from it named
+// attribute values from named tags.
+//
+// Caveat: this class currently doesn't handle multi-word UTF-16 encoded
+// characters. Doesn't handle implies that any data following such a
+// character could possibly be misinterpreted.
+//
+class HTMLScanner {
+ public:
+ typedef std::wstring::const_iterator StrPos;
+
+ // Structure maintaining const_iterators into html_string_.
+ class StringRange {
+ friend class HTMLScanner;
+ public:
+ StringRange();
+ StringRange(StrPos start, StrPos end);
+
+ bool LowerCaseEqualsASCII(const char* other) const;
+ bool Equals(const wchar_t* other) const;
+
+ // Copies the data described by StringRange into destination.
+ std::wstring Copy() const;
+
+ // If this StringRange represents a tag, this method extracts the name of
+ // the tag and sticks it in tag_name.
+ // Returns true if the tag name was successfully extracted.
+ // Returns false if this string doesn't look like a valid tag.
+ bool GetTagName(std::wstring* tag_name) const;
+
+ // From a given string range, uses a string tokenizer to extract the value
+ // of the named attribute if a simple scan finds that the attribute name is
+ // present.
+ //
+ // Returns true if the named attribute can be located and it has a value
+ // which has been placed in attribute_value.
+ //
+ // Note that the attribute value is unquoted here as well, so that
+ // GetTagAttribute(*<foo bar="baz">*, L"bar", *out_value*) will stick
+ // 'bar' in out_value and not '"bar"'.
+ //
+ // Returns false if the named attribute is not present in the tag or if it
+ // did not have a value.
+ //
+ bool GetTagAttribute(const wchar_t* attribute_name,
+ StringRange* attribute_value) const;
+
+ // Unquotes a StringRange by removing a matching pair of either ' or "
+ // characters from the beginning and end of the string if present.
+ // Returns true if string was modified, false otherwise.
+ bool UnQuote();
+ private:
+ StrPos start_;
+ StrPos end_;
+ };
+
+ typedef std::vector<StringRange> StringRangeList;
+
+ // html_string must be a null-terminated string containing the HTML
+ // to be scanned.
+ explicit HTMLScanner(const wchar_t* html_string);
+
+ // Returns the set of ranges denoting HTML tags that match the given name.
+ // If stop_tag_name is given, then as soon as a tag with this name is
+ // encountered this method will return.
+ void GetTagsByName(const wchar_t* name, StringRangeList* tag_list,
+ const wchar_t* stop_tag_name);
+
+ private:
+ friend class HtmlUtilUnittest;
+ FRIEND_TEST(HtmlUtilUnittest, BasicTest);
+
+ // Given html_string which represents the remaining html range, this method
+ // returns the next tag in tag and advances html_string to one character after
+ // the end of tag. This method is intended to be called repeatedly to extract
+ // all of the tags in sequence.
+ //
+ // Returns true if another tag was found and 'tag' was populated with a valid
+ // range.
+ // Returns false if we have reached the end of the html data.
+ bool NextTag(StringRange* html_string, StringRange* tag);
+
+ // Returns true if c can be found in quotes_, false otherwise
+ bool IsQuote(wchar_t c);
+
+ // Returns true if pos refers to the last character in an HTML comment in a
+ // string described by html_string, false otherwise.
+ // For example with html_string describing <!-- foo> -->, pos must refer to
+ // the last > for this method to return true.
+ bool IsHTMLCommentClose(StringRange* html_string, StrPos pos);
+
+ // We store a (CollapsedWhitespace'd) copy of the html data.
+ const std::wstring html_string_;
+
+ // Store the string of quote characters to avoid repeated construction.
+ const std::wstring quotes_;
+
+ DISALLOW_COPY_AND_ASSIGN(HTMLScanner);
+};
+
+#endif // CHROME_FRAME_HTML_UTILS_H_
diff --git a/chrome_frame/icu_stubs.cc b/chrome_frame/icu_stubs.cc
new file mode 100644
index 0000000..2f3abcf
--- /dev/null
+++ b/chrome_frame/icu_stubs.cc
@@ -0,0 +1,169 @@
+// Copyright (c) 2009 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 "base/logging.h"
+#include "base/string_util.h"
+#include "googleurl/src/url_canon.h"
+
+#include <windows.h>
+
+////////////////////////////////////////////////////////////////////////////////
+// Avoid dependency on string_util_icu.cc (which pulls in icu).
+
+std::string WideToAnsiDirect(const wchar_t* wide, size_t wide_len) {
+ std::string ret;
+ char* write = WriteInto(&ret, wide_len + 1);
+ for (size_t i = 0; i < wide_len; ++i) {
+ // We can only convert characters below 0x80 directly from wide to ansi.
+ DCHECK(wide[i] <= 127) << "can't convert";
+ write[i] = static_cast<char>(wide[i]);
+ }
+
+ write[wide_len] = '\0';
+
+ return ret;
+}
+
+bool WideToUTF8(const wchar_t* wide, size_t wide_len, std::string* utf8) {
+ DCHECK(utf8);
+
+ // Add a cutoff. If it's all ASCII, convert it directly
+ size_t i;
+ for (i = 0; i < wide_len; ++i) {
+ if (wide[i] > 127)
+ break;
+ }
+
+ // If we made it to the end without breaking, then it's all ANSI, so do a
+ // quick convert
+ if (i == wide_len) {
+ *utf8 = WideToAnsiDirect(wide, wide_len);
+ return true;
+ }
+
+ // Figure out how long the string is
+ int size = WideCharToMultiByte(CP_UTF8, 0, wide, wide_len + 1, NULL, 0, NULL,
+ NULL);
+
+ if (size > 0) {
+ WideCharToMultiByte(CP_UTF8, 0, wide, wide_len + 1, WriteInto(utf8, size),
+ size, NULL, NULL);
+ }
+
+ return (size > 0);
+}
+
+std::string WideToUTF8(const std::wstring& wide) {
+ std::string ret;
+ if (!wide.empty()) {
+ // Ignore the success flag of this call, it will do the best it can for
+ // invalid input, which is what we want here.
+ WideToUTF8(wide.data(), wide.length(), &ret);
+ }
+ return ret;
+}
+
+bool UTF8ToWide(const char* src, size_t src_len, std::wstring* output) {
+ DCHECK(output);
+
+ if (src_len == 0) {
+ output->clear();
+ return true;
+ }
+
+ int wide_chars = MultiByteToWideChar(CP_UTF8, 0, src, src_len, NULL, 0);
+ if (!wide_chars) {
+ NOTREACHED();
+ return false;
+ }
+
+ wide_chars++; // make room for L'\0'
+ // Note that WriteInto will fill the string with '\0', so in the case
+ // where the input string is not \0 terminated, we will still be ensured
+ // that the output string will be.
+ if (!MultiByteToWideChar(CP_UTF8, 0, src, src_len,
+ WriteInto(output, wide_chars), wide_chars)) {
+ NOTREACHED();
+ output->clear();
+ return false;
+ }
+
+ return true;
+}
+
+std::wstring UTF8ToWide(const base::StringPiece& utf8) {
+ std::wstring ret;
+ if (!utf8.empty())
+ UTF8ToWide(utf8.data(), utf8.length(), &ret);
+ return ret;
+}
+
+#ifdef WCHAR_T_IS_UTF16
+string16 UTF8ToUTF16(const std::string& utf8) {
+ std::wstring ret;
+ if (!utf8.empty())
+ UTF8ToWide(utf8.data(), utf8.length(), &ret);
+ return ret;
+}
+#else
+#error Need WCHAR_T_IS_UTF16
+#endif
+
+////////////////////////////////////////////////////////////////////////////////
+// Replace ICU dependent functions in googleurl.
+/*#define __UTF_H__
+#include "third_party/icu38/public/common/unicode/utf16.h"
+#define U_IS_SURROGATE(c) (((c)&0xfffff800)==0xd800)
+extern const char16 kUnicodeReplacementCharacter;*/
+
+namespace url_canon {
+
+bool IDNToASCII(const char16* src, int src_len, CanonOutputW* output) {
+ // We should only hit this when the user attempts to navigate
+ // CF to an invalid URL.
+ DLOG(WARNING) << __FUNCTION__ << " not implemented";
+ return false;
+}
+
+bool ReadUTFChar(const char* str, int* begin, int length,
+ unsigned* code_point_out) {
+ // We should only hit this when the user attempts to navigate
+ // CF to an invalid URL.
+ DLOG(WARNING) << __FUNCTION__ << " not implemented";
+
+ // TODO(tommi): consider if we can use something like
+ // http://bjoern.hoehrmann.de/utf-8/decoder/dfa/
+ return false;
+}
+
+bool ReadUTFChar(const char16* str, int* begin, int length,
+ unsigned* code_point) {
+/*
+ if (U16_IS_SURROGATE(str[*begin])) {
+ if (!U16_IS_SURROGATE_LEAD(str[*begin]) || *begin + 1 >= length ||
+ !U16_IS_TRAIL(str[*begin + 1])) {
+ // Invalid surrogate pair.
+ *code_point = kUnicodeReplacementCharacter;
+ return false;
+ } else {
+ // Valid surrogate pair.
+ *code_point = U16_GET_SUPPLEMENTARY(str[*begin], str[*begin + 1]);
+ (*begin)++;
+ }
+ } else {
+ // Not a surrogate, just one 16-bit word.
+ *code_point = str[*begin];
+ }
+
+ if (U_IS_UNICODE_CHAR(*code_point))
+ return true;
+
+ // Invalid code point.
+ *code_point = kUnicodeReplacementCharacter;
+ return false;*/
+ CHECK(false);
+ return false;
+}
+
+} // namespace url_canon
diff --git a/chrome_frame/ie8_types.h b/chrome_frame/ie8_types.h
new file mode 100644
index 0000000..c683471
--- /dev/null
+++ b/chrome_frame/ie8_types.h
@@ -0,0 +1,27 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_IE8_TYPES_H_
+#define CHROME_FRAME_IE8_TYPES_H_
+
+#include <urlmon.h>
+
+#ifndef __IInternetBindInfoEx_INTERFACE_DEFINED__
+#define __IInternetBindInfoEx_INTERFACE_DEFINED__
+
+#define IID_IInternetBindInfoEx (__uuidof(IInternetBindInfoEx))
+
+MIDL_INTERFACE("A3E015B7-A82C-4DCD-A150-569AEEED36AB")
+IInternetBindInfoEx : public IInternetBindInfo {
+ public:
+ virtual HRESULT STDMETHODCALLTYPE GetBindInfoEx(
+ DWORD *grfBINDF,
+ BINDINFO *pbindinfo,
+ DWORD *grfBINDF2,
+ DWORD *pdwReserved) = 0;
+};
+
+#endif __IInternetBindInfoEx_INTERFACE_DEFINED__
+
+#endif // CHROME_FRAME_IE8_TYPES_H_
diff --git a/chrome_frame/iids.cc b/chrome_frame/iids.cc
new file mode 100644
index 0000000..681250a
--- /dev/null
+++ b/chrome_frame/iids.cc
@@ -0,0 +1,10 @@
+// Copyright (c) 2009 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.
+
+// Pull in IIDs, CLSIDs etc from our .idl file.
+#include "chrome_tab.h"
+
+extern "C" {
+#include "chrome_tab_i.c"
+}
diff --git a/chrome_frame/in_place_menu.h b/chrome_frame/in_place_menu.h
new file mode 100644
index 0000000..9742896
--- /dev/null
+++ b/chrome_frame/in_place_menu.h
@@ -0,0 +1,231 @@
+// Copyright 2008, Google Inc.
+// All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+#ifndef CHROME_FRAME_IN_PLACE_MENU_H_
+#define CHROME_FRAME_IN_PLACE_MENU_H_
+
+// in_place_menu.h : menu merging implementation
+//
+// This file is a modified version of the menu.h file, which is
+// part of the ActiveDoc MSDN sample. The modifications are largely
+// conversions to Google coding guidelines. Below is the original header
+// from the file.
+
+// This is a part of the Active Template Library.
+// Copyright (c) Microsoft Corporation. All rights reserved.
+//
+// This source code is only intended as a supplement to the
+// Active Template Library Reference and related
+// electronic documentation provided with the library.
+// See these sources for detailed information regarding the
+// Active Template Library product.
+
+#include "base/logging.h"
+#include "base/scoped_comptr_win.h"
+
+template <class T>
+class InPlaceMenu {
+ public:
+ InPlaceMenu() : shared_menu_(NULL), ole_menu_(NULL), our_menu_(NULL) {
+ }
+
+ ~InPlaceMenu() {
+ InPlaceMenuDestroy();
+ }
+
+ HRESULT InPlaceMenuCreate(LPCWSTR menu_name) {
+ // We might already have an in-place menu set, because we set menus
+ // IOleDocumentView::UIActivate as well as in
+ // IOleInPlaceActiveObject::OnDocWindowActivate. If we have already
+ // done our work, just return silently
+ if (ole_menu_ || shared_menu_)
+ return S_OK;
+
+ ScopedComPtr<IOleInPlaceFrame> in_place_frame;
+ GetInPlaceFrame(in_place_frame.Receive());
+ // We have no IOleInPlaceFrame, no menu merging possible
+ if (!in_place_frame) {
+ NOTREACHED();
+ return E_FAIL;
+ }
+ // Create a blank menu and ask the container to add
+ // its menus into the OLEMENUGROUPWIDTHS structure
+ shared_menu_ = ::CreateMenu();
+ OLEMENUGROUPWIDTHS mgw = {0};
+ HRESULT hr = in_place_frame->InsertMenus(shared_menu_, &mgw);
+ if (FAILED(hr)) {
+ ::DestroyMenu(shared_menu_);
+ shared_menu_ = NULL;
+ return hr;
+ }
+ // Insert our menus
+ our_menu_ = LoadMenu(_AtlBaseModule.GetResourceInstance(),menu_name);
+ MergeMenus(shared_menu_, our_menu_, &mgw.width[0], 1);
+ // Send the menu to the client
+ ole_menu_ = (HMENU)OleCreateMenuDescriptor(shared_menu_, &mgw);
+ T* t = static_cast<T*>(this);
+ in_place_frame->SetMenu(shared_menu_, ole_menu_, t->m_hWnd);
+ return S_OK;
+ }
+
+ HRESULT InPlaceMenuDestroy() {
+ ScopedComPtr<IOleInPlaceFrame> in_place_frame;
+ GetInPlaceFrame(in_place_frame.Receive());
+ if (in_place_frame) {
+ in_place_frame->RemoveMenus(shared_menu_);
+ in_place_frame->SetMenu(NULL, NULL, NULL);
+ }
+ if (ole_menu_) {
+ OleDestroyMenuDescriptor(ole_menu_);
+ ole_menu_ = NULL;
+ }
+ if (shared_menu_) {
+ UnmergeMenus(shared_menu_, our_menu_);
+ DestroyMenu(shared_menu_);
+ shared_menu_ = NULL;
+ }
+ if (our_menu_) {
+ DestroyMenu(our_menu_);
+ our_menu_ = NULL;
+ }
+ return S_OK;
+ }
+
+ void MergeMenus(HMENU shared_menu, HMENU source_menu, LONG* menu_widths,
+ unsigned int width_index) {
+ // Copy the popups from the source menu
+ // Insert at appropriate spot depending on width_index
+ DCHECK(width_index == 0 || width_index == 1);
+ int position = 0;
+ if (width_index == 1)
+ position = (int)menu_widths[0];
+ int group_width = 0;
+ int menu_items = GetMenuItemCount(source_menu);
+ for (int index = 0; index < menu_items; index++) {
+ // Get the HMENU of the popup
+ HMENU popup_menu = ::GetSubMenu(source_menu, index);
+ // Separators move us to next group
+ UINT state = GetMenuState(source_menu, index, MF_BYPOSITION);
+ if (!popup_menu && (state & MF_SEPARATOR)) {
+ // Servers should not touch past 5
+ DCHECK(width_index <= 5);
+ menu_widths[width_index] = group_width;
+ group_width = 0;
+ if (width_index < 5)
+ position += static_cast<int>(menu_widths[width_index+1]);
+ width_index += 2;
+ } else {
+ // Get the menu item text
+ TCHAR item_text[256] = {0};
+ int text_length = GetMenuString(source_menu, index, item_text,
+ ARRAYSIZE(item_text), MF_BYPOSITION);
+ // Popups are handled differently than normal menu items
+ if (popup_menu) {
+ if (::GetMenuItemCount(popup_menu) != 0) {
+ // Strip the HIBYTE because it contains a count of items
+ state = LOBYTE(state) | MF_POPUP; // Must be popup
+ // Non-empty popup -- add it to the shared menu bar
+ InsertMenu(shared_menu, position, state|MF_BYPOSITION,
+ reinterpret_cast<UINT_PTR>(popup_menu), item_text);
+ ++position;
+ ++group_width;
+ }
+ } else if (text_length > 0) {
+ // only non-empty items are added
+ DCHECK(item_text[0] != 0);
+ // here the state does not contain a count in the HIBYTE
+ InsertMenu(shared_menu, position, state|MF_BYPOSITION,
+ GetMenuItemID(source_menu, index), item_text);
+ ++position;
+ ++group_width;
+ }
+ }
+ }
+ }
+
+ void UnmergeMenus(HMENU shared_menu, HMENU source_menu) {
+ int our_item_count = GetMenuItemCount(source_menu);
+ int shared_item_count = GetMenuItemCount(shared_menu);
+
+ for (int index = shared_item_count - 1; index >= 0; index--) {
+ // Check the popup menus
+ HMENU popup_menu = ::GetSubMenu(shared_menu, index);
+ if (popup_menu) {
+ // If it is one of ours, remove it from the shared menu
+ for (int sub_index = 0; sub_index < our_item_count; sub_index++) {
+ if (::GetSubMenu(source_menu, sub_index) == popup_menu) {
+ // Remove the menu from hMenuShared
+ RemoveMenu(shared_menu, index, MF_BYPOSITION);
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ protected:
+ HRESULT GetInPlaceFrame(IOleInPlaceFrame** in_place_frame) {
+ if (!in_place_frame) {
+ NOTREACHED();
+ return E_POINTER;
+ }
+ T* t = static_cast<T*>(this);
+ HRESULT hr = E_FAIL;
+ if (!t->in_place_frame_) {
+ // We weren't given an IOleInPlaceFrame pointer, so
+ // we'll have to get it ourselves.
+ if (t->m_spInPlaceSite) {
+ t->frame_info_.cb = sizeof(OLEINPLACEFRAMEINFO);
+ ScopedComPtr<IOleInPlaceUIWindow> in_place_ui_window;
+ RECT position_rect = {0};
+ RECT clip_rect = {0};
+ hr = t->m_spInPlaceSite->GetWindowContext(in_place_frame,
+ in_place_ui_window.Receive(),
+ &position_rect, &clip_rect,
+ &t->frame_info_);
+ }
+ } else {
+ *in_place_frame = t->in_place_frame_;
+ (*in_place_frame)->AddRef();
+ hr = S_OK;
+ }
+ return hr;
+ }
+
+ protected:
+ // The OLE menu descriptor created by the OleCreateMenuDescriptor
+ HMENU ole_menu_;
+ // The shared menu that we pass to IOleInPlaceFrame::SetMenu
+ HMENU shared_menu_;
+ // Our menu resource that we want to insert
+ HMENU our_menu_;
+};
+
+#endif // CHROME_FRAME_IN_PLACE_MENU_H_
+
diff --git a/chrome_frame/installer_util/test.txt b/chrome_frame/installer_util/test.txt
new file mode 100644
index 0000000..30d74d2
--- /dev/null
+++ b/chrome_frame/installer_util/test.txt
@@ -0,0 +1 @@
+test \ No newline at end of file
diff --git a/chrome_frame/np_browser_functions.cc b/chrome_frame/np_browser_functions.cc
new file mode 100644
index 0000000..9cc6f77
--- /dev/null
+++ b/chrome_frame/np_browser_functions.cc
@@ -0,0 +1,512 @@
+// Copyright (c) 2009 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 "chrome_frame/np_browser_functions.h"
+
+#include "base/logging.h"
+
+namespace npapi {
+
+// global function pointers (within this namespace) for the NPN functions.
+union NpVersion {
+ struct {
+ uint8 major;
+ uint8 minor;
+ } v;
+ uint16 version;
+};
+
+NpVersion g_version = {0};
+
+NPN_GetURLProcPtr g_geturl = NULL;
+NPN_PostURLProcPtr g_posturl = NULL;
+NPN_RequestReadProcPtr g_requestread = NULL;
+NPN_NewStreamProcPtr g_newstream = NULL;
+NPN_WriteProcPtr g_write = NULL;
+NPN_DestroyStreamProcPtr g_destroystream = NULL;
+NPN_StatusProcPtr g_status = NULL;
+NPN_UserAgentProcPtr g_useragent = NULL;
+NPN_MemAllocProcPtr g_memalloc = NULL;
+NPN_MemFreeProcPtr g_memfree = NULL;
+NPN_MemFlushProcPtr g_memflush = NULL;
+NPN_ReloadPluginsProcPtr g_reloadplugins = NULL;
+NPN_GetJavaEnvProcPtr g_getJavaEnv = NULL;
+NPN_GetJavaPeerProcPtr g_getJavaPeer = NULL;
+NPN_GetURLNotifyProcPtr g_geturlnotify = NULL;
+NPN_PostURLNotifyProcPtr g_posturlnotify = NULL;
+NPN_GetValueProcPtr g_getvalue = NULL;
+NPN_SetValueProcPtr g_setvalue = NULL;
+NPN_InvalidateRectProcPtr g_invalidaterect = NULL;
+NPN_InvalidateRegionProcPtr g_invalidateregion = NULL;
+NPN_ForceRedrawProcPtr g_forceredraw = NULL;
+NPN_GetStringIdentifierProcPtr g_getstringidentifier = NULL;
+NPN_GetStringIdentifiersProcPtr g_getstringidentifiers = NULL;
+NPN_GetIntIdentifierProcPtr g_getintidentifier = NULL;
+NPN_IdentifierIsStringProcPtr g_identifierisstring = NULL;
+NPN_UTF8FromIdentifierProcPtr g_utf8fromidentifier = NULL;
+NPN_IntFromIdentifierProcPtr g_intfromidentifier = NULL;
+NPN_CreateObjectProcPtr g_createobject = NULL;
+NPN_RetainObjectProcPtr g_retainobject = NULL;
+NPN_ReleaseObjectProcPtr g_releaseobject = NULL;
+NPN_InvokeProcPtr g_invoke = NULL;
+NPN_InvokeDefaultProcPtr g_invoke_default = NULL;
+NPN_EvaluateProcPtr g_evaluate = NULL;
+NPN_GetPropertyProcPtr g_getproperty = NULL;
+NPN_SetPropertyProcPtr g_setproperty = NULL;
+NPN_RemovePropertyProcPtr g_removeproperty = NULL;
+NPN_HasPropertyProcPtr g_hasproperty = NULL;
+NPN_HasMethodProcPtr g_hasmethod = NULL;
+NPN_ReleaseVariantValueProcPtr g_releasevariantvalue = NULL;
+NPN_SetExceptionProcPtr g_setexception = NULL;
+NPN_PushPopupsEnabledStateProcPtr g_pushpopupsenabledstate = NULL;
+NPN_PopPopupsEnabledStateProcPtr g_poppopupsenabledstate = NULL;
+NPN_EnumerateProcPtr g_enumerate = NULL;
+NPN_PluginThreadAsyncCallProcPtr g_pluginthreadasynccall = NULL;
+NPN_ConstructProcPtr g_construct = NULL;
+NPN_GetValueForURLProcPtr g_getvalueforurl = NULL;
+NPN_SetValueForURLProcPtr g_setvalueforurl = NULL;
+NPN_GetAuthenticationInfoProcPtr g_getauthenticationinfo = NULL;
+
+// Must be called prior to calling any of the browser functions below.
+void InitializeBrowserFunctions(NPNetscapeFuncs* functions) {
+ CHECK(functions);
+ DCHECK(g_geturl == NULL || g_geturl == functions->geturl);
+
+ g_version.version = functions->version;
+
+ g_geturl = functions->geturl;
+ g_posturl = functions->posturl;
+ g_requestread = functions->requestread;
+ g_newstream = functions->newstream;
+ g_write = functions->write;
+ g_destroystream = functions->destroystream;
+ g_status = functions->status;
+ g_useragent = functions->uagent;
+ g_memalloc = functions->memalloc;
+ g_memfree = functions->memfree;
+ g_memflush = functions->memflush;
+ g_reloadplugins = functions->reloadplugins;
+ g_getJavaEnv = functions->getJavaEnv;
+ g_getJavaPeer = functions->getJavaPeer;
+ g_geturlnotify = functions->geturlnotify;
+ g_posturlnotify = functions->posturlnotify;
+ g_getvalue = functions->getvalue;
+ g_setvalue = functions->setvalue;
+ g_invalidaterect = functions->invalidaterect;
+ g_invalidateregion = functions->invalidateregion;
+ g_forceredraw = functions->forceredraw;
+ g_getstringidentifier = functions->getstringidentifier;
+ g_getstringidentifiers = functions->getstringidentifiers;
+ g_getintidentifier = functions->getintidentifier;
+ g_identifierisstring = functions->identifierisstring;
+ g_utf8fromidentifier = functions->utf8fromidentifier;
+ g_intfromidentifier = functions->intfromidentifier;
+ g_createobject = functions->createobject;
+ g_retainobject = functions->retainobject;
+ g_releaseobject = functions->releaseobject;
+ g_invoke = functions->invoke;
+ g_invoke_default = functions->invokeDefault;
+ g_evaluate = functions->evaluate;
+ g_getproperty = functions->getproperty;
+ g_setproperty = functions->setproperty;
+ g_removeproperty = functions->removeproperty;
+ g_hasproperty = functions->hasproperty;
+ g_hasmethod = functions->hasmethod;
+ g_releasevariantvalue = functions->releasevariantvalue;
+ g_setexception = functions->setexception;
+ g_pushpopupsenabledstate = functions->pushpopupsenabledstate;
+ g_poppopupsenabledstate = functions->poppopupsenabledstate;
+ g_enumerate = functions->enumerate;
+ g_pluginthreadasynccall = functions->pluginthreadasynccall;
+ g_construct = functions->construct;
+
+ if (g_version.v.minor >= NPVERS_HAS_URL_AND_AUTH_INFO) {
+ g_getvalueforurl = functions->getvalueforurl;
+ g_setvalueforurl = functions->setvalueforurl;
+ g_getauthenticationinfo = functions->getauthenticationinfo;
+ }
+}
+
+void UninitializeBrowserFunctions() {
+ g_version.version = 0;
+
+ // We skip doing this in the official build as it doesn't serve much purpose
+// during shutdown. The reason for it being here in the other types of builds
+// is to spot potential browser bugs whereby the browser leaves living objects
+// in our DLL after shutdown has been called. In theory those objects could
+// trigger a call to the browser functions after shutdown has been called
+// and for non official builds we want that to simply crash.
+// For official builds we leave the function pointers around since they
+// continue to valid.
+ g_geturl = NULL;
+ g_posturl = NULL;
+ g_requestread = NULL;
+ g_newstream = NULL;
+ g_write = NULL;
+ g_destroystream = NULL;
+ g_status = NULL;
+ g_useragent = NULL;
+ g_memalloc = NULL;
+ g_memfree = NULL;
+ g_memflush = NULL;
+ g_reloadplugins = NULL;
+ g_getJavaEnv = NULL;
+ g_getJavaPeer = NULL;
+ g_geturlnotify = NULL;
+ g_posturlnotify = NULL;
+ g_getvalue = NULL;
+ g_setvalue = NULL;
+ g_invalidaterect = NULL;
+ g_invalidateregion = NULL;
+ g_forceredraw = NULL;
+ g_getstringidentifier = NULL;
+ g_getstringidentifiers = NULL;
+ g_getintidentifier = NULL;
+ g_identifierisstring = NULL;
+ g_utf8fromidentifier = NULL;
+ g_intfromidentifier = NULL;
+ g_createobject = NULL;
+ g_retainobject = NULL;
+ g_releaseobject = NULL;
+ g_invoke = NULL;
+ g_invoke_default = NULL;
+ g_evaluate = NULL;
+ g_getproperty = NULL;
+ g_setproperty = NULL;
+ g_removeproperty = NULL;
+ g_hasproperty = NULL;
+ g_hasmethod = NULL;
+ g_releasevariantvalue = NULL;
+ g_setexception = NULL;
+ g_pushpopupsenabledstate = NULL;
+ g_poppopupsenabledstate = NULL;
+ g_enumerate = NULL;
+ g_pluginthreadasynccall = NULL;
+ g_construct = NULL;
+ g_getvalueforurl = NULL;
+ g_setvalueforurl = NULL;
+ g_getauthenticationinfo = NULL;
+}
+
+bool IsInitialized() {
+ // We only check one function for convenience.
+ return g_getvalue != NULL;
+}
+
+// Function stubs for functions that the host browser implements.
+uint8 VersionMinor() {
+ return g_version.v.minor;
+}
+
+uint8 VersionMajor() {
+ return g_version.v.major;
+}
+
+NPError GetURL(NPP instance, const char* URL, const char* window) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_geturl(instance, URL, window);
+}
+
+NPError PostURL(NPP instance, const char* URL, const char* window, uint32 len,
+ const char* buf, NPBool file) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_posturl(instance, URL, window, len, buf, file);
+}
+
+NPError RequestRead(NPStream* stream, NPByteRange* rangeList) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_requestread(stream, rangeList);
+}
+
+NPError NewStream(NPP instance, NPMIMEType type, const char* window,
+ NPStream** stream) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_newstream(instance, type, window, stream);
+}
+
+int32 Write(NPP instance, NPStream* stream, int32 len, void* buffer) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_write(instance, stream, len, buffer);
+}
+
+NPError DestroyStream(NPP instance, NPStream* stream, NPReason reason) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_destroystream(instance, stream, reason);
+}
+
+void Status(NPP instance, const char* message) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_status(instance, message);
+}
+
+const char* UserAgent(NPP instance) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_useragent(instance);
+}
+
+void* MemAlloc(uint32 size) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_memalloc(size);
+}
+
+void MemFree(void* ptr) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_memfree(ptr);
+}
+
+uint32 MemFlush(uint32 size) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_memflush(size);
+}
+
+void ReloadPlugins(NPBool reloadPages) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_reloadplugins(reloadPages);
+}
+
+void* GetJavaEnv() {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_getJavaEnv();
+}
+
+void* GetJavaPeer(NPP instance) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_getJavaPeer(instance);
+}
+
+NPError GetURLNotify(NPP instance, const char* URL, const char* window,
+ void* notifyData) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_geturlnotify(instance, URL, window, notifyData);
+}
+
+NPError PostURLNotify(NPP instance, const char* URL, const char* window,
+ uint32 len, const char* buf, NPBool file,
+ void* notifyData) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_posturlnotify(instance, URL, window, len, buf, file, notifyData);
+}
+
+NPError GetValue(NPP instance, NPNVariable variable, void* ret_value) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_getvalue(instance, variable, ret_value);
+}
+
+NPError SetValue(NPP instance, NPPVariable variable, void* value) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_setvalue(instance, variable, value);
+}
+
+void InvalidateRect(NPP instance, NPRect* rect) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_invalidaterect(instance, rect);
+}
+
+void InvalidateRegion(NPP instance, NPRegion region) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_invalidateregion(instance, region);
+}
+
+void ForceRedraw(NPP instance) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_forceredraw(instance);
+}
+
+void ReleaseVariantValue(NPVariant* variant) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_releasevariantvalue(variant);
+}
+
+NPIdentifier GetStringIdentifier(const NPUTF8* name) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_getstringidentifier(name);
+}
+
+void GetStringIdentifiers(const NPUTF8** names, int32_t nameCount,
+ NPIdentifier* identifiers) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_getstringidentifiers(names, nameCount, identifiers);
+}
+
+NPIdentifier GetIntIdentifier(int32_t intid) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_getintidentifier(intid);
+}
+
+int32_t IntFromIdentifier(NPIdentifier identifier) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_intfromidentifier(identifier);
+}
+
+bool IdentifierIsString(NPIdentifier identifier) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_identifierisstring(identifier);
+
+}
+
+NPUTF8* UTF8FromIdentifier(NPIdentifier identifier) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_utf8fromidentifier(identifier);
+
+}
+
+NPObject* CreateObject(NPP instance, NPClass* aClass) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_createobject(instance, aClass);
+
+}
+
+NPObject* RetainObject(NPObject* obj) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_retainobject(obj);
+
+}
+
+void ReleaseObject(NPObject* obj) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_releaseobject(obj);
+
+}
+
+bool Invoke(NPP npp, NPObject* obj, NPIdentifier methodName,
+ const NPVariant* args, unsigned argCount, NPVariant* result) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_invoke(npp, obj, methodName, args, argCount, result);
+}
+
+bool InvokeDefault(NPP npp, NPObject* obj, const NPVariant* args,
+ unsigned argCount, NPVariant* result) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_invoke_default(npp, obj, args, argCount, result);
+}
+
+bool Evaluate(NPP npp, NPObject* obj, NPString* script, NPVariant* result) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_evaluate(npp, obj, script, result);
+}
+
+bool GetProperty(NPP npp, NPObject* obj, NPIdentifier propertyName,
+ NPVariant* result) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_getproperty(npp, obj, propertyName, result);
+}
+
+bool SetProperty(NPP npp, NPObject* obj, NPIdentifier propertyName,
+ const NPVariant* value) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_setproperty(npp, obj, propertyName, value);
+}
+
+bool HasProperty(NPP npp, NPObject* npobj, NPIdentifier propertyName) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_hasproperty(npp, npobj, propertyName);
+}
+
+bool HasMethod(NPP npp, NPObject* npobj, NPIdentifier methodName) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_hasmethod(npp, npobj, methodName);
+}
+
+bool RemoveProperty(NPP npp, NPObject* obj, NPIdentifier propertyName) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_removeproperty(npp, obj, propertyName);
+}
+
+void SetException(NPObject* obj, const NPUTF8* message) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_setexception(obj, message);
+}
+
+void PushPopupsEnabledState(NPP npp, NPBool enabled) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_pushpopupsenabledstate(npp, enabled);
+}
+
+void PopPopupsEnabledState(NPP npp) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_poppopupsenabledstate(npp);
+}
+
+bool Enumerate(NPP npp, NPObject* obj, NPIdentifier** identifier,
+ uint32_t* count) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_enumerate(npp, obj, identifier, count);
+}
+
+void PluginThreadAsyncCall(NPP instance, void (*func)(void*), void* userData) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_pluginthreadasynccall(instance, func, userData);
+}
+
+bool Construct(NPP npp, NPObject* obj, const NPVariant* args, uint32_t argCount,
+ NPVariant* result) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ return g_construct(npp, obj, args, argCount, result);
+}
+
+NPError GetValueForURL(NPP instance, NPNURLVariable variable, const char* url,
+ char** value, uint32* len) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ DCHECK(npapi::VersionMinor() >= NPVERS_HAS_URL_AND_AUTH_INFO);
+ if (!g_getvalueforurl) {
+ NOTREACHED();
+ return NPERR_INCOMPATIBLE_VERSION_ERROR;
+ }
+
+ return g_getvalueforurl(instance, variable, url, value, len);
+}
+
+NPError SetValueForURL(NPP instance, NPNURLVariable variable, const char* url,
+ const char* value, uint32 len) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ DCHECK(npapi::VersionMinor() >= NPVERS_HAS_URL_AND_AUTH_INFO);
+ if (g_setvalueforurl) {
+ NOTREACHED();
+ return NPERR_INCOMPATIBLE_VERSION_ERROR;
+ }
+
+ return g_setvalueforurl(instance, variable, url, value, len);
+}
+
+NPError GetAuthenticationInfo(NPP instance, const char* protocol,
+ const char* host, int32 port, const char* scheme,
+ const char *realm, char** username, uint32* ulen,
+ char** password, uint32* plen) {
+ DCHECK(IsInitialized()) << __FUNCTION__;
+ DCHECK(npapi::VersionMinor() >= NPVERS_HAS_URL_AND_AUTH_INFO);
+ if (g_getauthenticationinfo) {
+ NOTREACHED();
+ return NPERR_INCOMPATIBLE_VERSION_ERROR;
+ }
+
+ return g_getauthenticationinfo(instance, protocol, host, port, scheme,
+ realm, username, ulen, password, plen);
+}
+
+std::string StringFromIdentifier(NPIdentifier identifier) {
+ std::string ret;
+ NPUTF8* utf8 = UTF8FromIdentifier(identifier);
+ if (utf8) {
+ ret = utf8;
+ MemFree(utf8);
+ }
+ return ret;
+}
+
+} // namespace npapi
+
+void AllocateStringVariant(const std::string& str, NPVariant* var) {
+ DCHECK(var);
+
+ int len = str.length();
+ NPUTF8* buffer = reinterpret_cast<NPUTF8*>(npapi::MemAlloc(len + 1));
+ if (buffer) {
+ buffer[len] = '\0';
+ memcpy(buffer, str.c_str(), len);
+ STRINGN_TO_NPVARIANT(buffer, len, *var);
+ } else {
+ NULL_TO_NPVARIANT(*var);
+ }
+}
+
diff --git a/chrome_frame/np_browser_functions.h b/chrome_frame/np_browser_functions.h
new file mode 100644
index 0000000..ec29345
--- /dev/null
+++ b/chrome_frame/np_browser_functions.h
@@ -0,0 +1,246 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_NP_BROWSER_FUNCTIONS_H_
+#define CHROME_FRAME_NP_BROWSER_FUNCTIONS_H_
+
+#include "base/logging.h"
+#include "third_party/WebKit/WebCore/bridge/npapi.h"
+#include "third_party/WebKit/WebCore/plugins/npfunctions.h"
+
+namespace npapi {
+
+// Must be called prior to calling any of the browser functions below.
+void InitializeBrowserFunctions(NPNetscapeFuncs* functions);
+void UninitializeBrowserFunctions();
+
+// Returns true iff InitializeBrowserFunctions has been called successully.
+bool IsInitialized();
+
+// Function stubs for functions that the host browser implements.
+
+uint8 VersionMinor();
+uint8 VersionMajor();
+
+NPError GetURL(NPP instance, const char* URL, const char* window);
+
+NPError PostURL(NPP instance, const char* URL, const char* window, uint32 len,
+ const char* buf, NPBool file);
+
+NPError RequestRead(NPStream* stream, NPByteRange* rangeList);
+
+NPError NewStream(NPP instance, NPMIMEType type, const char* window,
+ NPStream** stream);
+
+int32 Write(NPP instance, NPStream* stream, int32 len, void* buffer);
+
+NPError DestroyStream(NPP instance, NPStream* stream, NPReason reason);
+
+void Status(NPP instance, const char* message);
+
+const char* UserAgent(NPP instance);
+
+void* MemAlloc(uint32 size);
+
+void MemFree(void* ptr);
+
+uint32 MemFlush(uint32 size);
+
+void ReloadPlugins(NPBool reloadPages);
+
+void* GetJavaEnv();
+
+void* GetJavaPeer(NPP instance);
+
+NPError GetURLNotify(NPP instance, const char* URL, const char* window,
+ void* notifyData);
+
+NPError PostURLNotify(NPP instance, const char* URL, const char* window,
+ uint32 len, const char* buf, NPBool file,
+ void* notifyData);
+
+NPError GetValue(NPP instance, NPNVariable variable, void* ret_value);
+
+NPError SetValue(NPP instance, NPPVariable variable, void* value);
+
+void InvalidateRect(NPP instance, NPRect* rect);
+
+void InvalidateRegion(NPP instance, NPRegion region);
+
+void ForceRedraw(NPP instance);
+
+void ReleaseVariantValue(NPVariant* variant);
+
+NPIdentifier GetStringIdentifier(const NPUTF8* name);
+
+void GetStringIdentifiers(const NPUTF8** names, int32_t nameCount,
+ NPIdentifier* identifiers);
+
+NPIdentifier GetIntIdentifier(int32_t intid);
+
+int32_t IntFromIdentifier(NPIdentifier identifier);
+
+bool IdentifierIsString(NPIdentifier identifier);
+
+NPUTF8* UTF8FromIdentifier(NPIdentifier identifier);
+
+NPObject* CreateObject(NPP, NPClass* aClass);
+
+NPObject* RetainObject(NPObject* obj);
+
+void ReleaseObject(NPObject* obj);
+
+bool Invoke(NPP npp, NPObject* obj, NPIdentifier methodName,
+ const NPVariant* args, unsigned argCount, NPVariant* result);
+
+bool InvokeDefault(NPP npp, NPObject* obj, const NPVariant* args,
+ unsigned argCount, NPVariant* result);
+
+bool Evaluate(NPP npp, NPObject* obj, NPString* script, NPVariant* result);
+
+bool GetProperty(NPP npp, NPObject* obj, NPIdentifier propertyName,
+ NPVariant* result);
+
+bool SetProperty(NPP npp, NPObject* obj, NPIdentifier propertyName,
+ const NPVariant* value);
+
+bool HasProperty(NPP npp, NPObject* npobj, NPIdentifier propertyName);
+
+bool HasMethod(NPP npp, NPObject* npobj, NPIdentifier methodName);
+
+bool RemoveProperty(NPP npp, NPObject* obj, NPIdentifier propertyName);
+
+void SetException(NPObject* obj, const NPUTF8* message);
+
+void PushPopupsEnabledState(NPP npp, NPBool enabled);
+
+void PopPopupsEnabledState(NPP npp);
+
+bool Enumerate(NPP npp, NPObject* obj, NPIdentifier** identifier,
+ uint32_t* count);
+
+void PluginThreadAsyncCall(NPP instance, void (*func)(void*), void* userData);
+
+bool Construct(NPP npp, NPObject* obj, const NPVariant* args, uint32_t argCount,
+ NPVariant* result);
+
+NPError GetValueForURL(NPP instance, NPNURLVariable variable, const char* url,
+ char** value, uint32* len);
+NPError SetValueForURL(NPP instance, NPNURLVariable variable, const char* url,
+ const char* value, uint32 len);
+NPError GetAuthenticationInfo(NPP instance, const char* protocol,
+ const char* host, int32 port, const char* scheme,
+ const char *realm, char** username, uint32* ulen,
+ char** password, uint32* plen);
+uint32 ScheduleTimer(NPP instance, uint32 interval, NPBool repeat,
+ void (*timerFunc)(NPP npp, uint32 timerID));
+void UnscheduleTimer(NPP instance, uint32 timerID);
+NPError PopUpContextMenu(NPP instance, NPMenu* menu);
+NPBool ConvertPoint(NPP instance, double sourceX, double sourceY,
+ NPCoordinateSpace sourceSpace, double *destX,
+ double *destY, NPCoordinateSpace destSpace);
+
+// Helper routine that wraps UTF8FromIdentifier to convert a string identifier
+// to an STL string. It's not super efficient since it could possibly do two
+// heap allocations (STL string has a stack based buffer for smaller strings).
+// For debugging purposes it is useful.
+std::string StringFromIdentifier(NPIdentifier identifier);
+
+} // namespace npapi
+
+// Simple helper class for freeing NPVariants at the end of a scope.
+class ScopedNpVariant : public NPVariant {
+ public:
+ ScopedNpVariant() {
+ VOID_TO_NPVARIANT(*this);
+ }
+
+ ~ScopedNpVariant() {
+ Free();
+ }
+
+ void Free() {
+ npapi::ReleaseVariantValue(this);
+ VOID_TO_NPVARIANT(*this);
+ }
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(ScopedNpVariant);
+};
+
+// Simple helper class for freeing NPObjects at the end of a scope.
+template <typename NpoType = NPObject>
+class ScopedNpObject {
+ public:
+ ScopedNpObject() : npo_(NULL) {
+ }
+
+ explicit ScopedNpObject(NpoType* npo) : npo_(npo) {
+ }
+
+ ~ScopedNpObject() {
+ Free();
+ }
+
+ NpoType* get() const {
+ return npo_;
+ }
+
+ operator NpoType*() const {
+ return npo_;
+ }
+
+ NpoType* operator->() const {
+ return npo_;
+ }
+
+ ScopedNpObject<NpoType>& operator=(NpoType* npo) {
+ if (npo != npo_) {
+ DCHECK(npo_ == NULL);
+ npapi::RetainObject(npo);
+ npo_ = npo;
+ }
+ return *this;
+ }
+
+ void Free() {
+ if (npo_) {
+ npapi::ReleaseObject(npo_);
+ npo_ = NULL;
+ }
+ }
+
+ NpoType** Receive() {
+ DCHECK(npo_ == NULL) << "Object leak. Pointer must be NULL";
+ return &npo_;
+ }
+
+ NpoType* Detach() {
+ NpoType* p = npo_;
+ npo_ = NULL;
+ return p;
+ }
+
+ void Attach(NpoType* p) {
+ DCHECK(npo_ == NULL);
+ npo_ = p;
+ }
+
+ NpoType* Copy() const {
+ if (npo_ != NULL)
+ npapi::RetainObject(npo_);
+ return npo_;
+ }
+
+ private:
+ NpoType* npo_;
+ DISALLOW_COPY_AND_ASSIGN(ScopedNpObject);
+};
+
+// Allocates a new NPUTF8 string and assigns it to the variant.
+// If memory allocation fails, the variant type will be set to NULL.
+// The memory allocation is done via the npapi browser functions.
+void AllocateStringVariant(const std::string& str, NPVariant* var);
+
+#endif // CHROME_FRAME_NP_BROWSER_FUNCTIONS_H_
diff --git a/chrome_frame/np_event_listener.cc b/chrome_frame/np_event_listener.cc
new file mode 100644
index 0000000..ff5479a
--- /dev/null
+++ b/chrome_frame/np_event_listener.cc
@@ -0,0 +1,367 @@
+// Copyright (c) 2009 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 "chrome_frame/np_event_listener.h"
+
+#include "base/string_util.h"
+
+#include "third_party/xulrunner-sdk/win/include/string/nsEmbedString.h"
+#include "third_party/xulrunner-sdk/win/include/dom/nsIDOMElement.h"
+#include "third_party/xulrunner-sdk/win/include/dom/nsIDOMEventTarget.h"
+#include "third_party/xulrunner-sdk/win/include/dom/nsIDOMEvent.h"
+
+#include "chrome_frame/scoped_ns_ptr_win.h"
+#include "chrome_frame/ns_associate_iid_win.h"
+
+ASSOCIATE_IID(NS_IDOMELEMENT_IID_STR, nsIDOMElement);
+ASSOCIATE_IID(NS_IDOMNODE_IID_STR, nsIDOMNode);
+ASSOCIATE_IID(NS_IDOMEVENTTARGET_IID_STR, nsIDOMEventTarget);
+ASSOCIATE_IID(NS_IDOMEVENTLISTENER_IID_STR, nsIDOMEventListener);
+
+DomEventListener::DomEventListener(NpEventDelegate* delegate)
+ : NpEventListenerBase<DomEventListener>(delegate) {
+}
+
+DomEventListener::~DomEventListener() {
+}
+
+// We implement QueryInterface etc ourselves in order to avoid
+// extra dependencies brought on by the NS_IMPL_* macros.
+NS_IMETHODIMP DomEventListener::QueryInterface(REFNSIID iid, void** ptr) {
+ DCHECK(thread_id_ == ::GetCurrentThreadId());
+ nsresult res = NS_NOINTERFACE;
+
+ if (memcmp(&iid, &__uuidof(nsIDOMEventListener), sizeof(nsIID)) == 0 ||
+ memcmp(&iid, &__uuidof(nsISupports), sizeof(nsIID)) == 0) {
+ *ptr = static_cast<nsIDOMEventListener*>(this);
+ AddRef();
+ res = NS_OK;
+ }
+
+ return res;
+}
+
+NS_IMETHODIMP DomEventListener::HandleEvent(nsIDOMEvent *event) {
+ DCHECK(thread_id_ == ::GetCurrentThreadId());
+ DCHECK(event);
+
+ nsEmbedString tag;
+ event->GetType(tag);
+ delegate_->OnEvent(WideToUTF8(tag.get()).c_str());
+
+ return NS_OK;
+}
+
+bool DomEventListener::Subscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count) {
+ DCHECK(event_names);
+ DCHECK(event_name_count > 0);
+
+ ScopedNsPtr<nsIDOMElement> element;
+ bool ret = GetObjectElement(instance, element.Receive());
+ if (ret) {
+ ScopedNsPtr<nsIDOMEventTarget> target;
+ target.QueryFrom(element);
+ if (target) {
+ for (int i = 0; i < event_name_count && ret; ++i) {
+ nsEmbedString name(ASCIIToWide(event_names[i]).c_str());
+ // See NPObjectEventListener::Subscribe (below) for a note on why
+ // we set the useCapture parameter to PR_FALSE.
+ nsresult res = target->AddEventListener(name, this, PR_FALSE);
+ DCHECK(res == NS_OK) << "AddEventListener: " << event_names[i];
+ ret = NS_SUCCEEDED(res);
+ }
+ } else {
+ DLOG(ERROR) << "failed to get nsIDOMEventTarget";
+ ret = false;
+ }
+ }
+
+ return ret;
+}
+
+bool DomEventListener::Unsubscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count) {
+ DCHECK(event_names);
+ DCHECK(event_name_count > 0);
+
+ ScopedNsPtr<nsIDOMElement> element;
+ bool ret = GetObjectElement(instance, element.Receive());
+ if (ret) {
+ ScopedNsPtr<nsIDOMEventTarget> target;
+ target.QueryFrom(element);
+ if (target) {
+ for (int i = 0; i < event_name_count && ret; ++i) {
+ nsEmbedString name(ASCIIToWide(event_names[i]).c_str());
+ nsresult res = target->RemoveEventListener(name, this, PR_FALSE);
+ DCHECK(res == NS_OK) << "RemoveEventListener: " << event_names[i];
+ ret = NS_SUCCEEDED(res) && ret;
+ }
+ } else {
+ DLOG(ERROR) << "failed to get nsIDOMEventTarget";
+ ret = false;
+ }
+ }
+
+ return ret;
+}
+
+bool DomEventListener::GetObjectElement(NPP instance, nsIDOMElement** element) {
+ DCHECK(element);
+
+ ScopedNsPtr<nsIDOMElement> elem;
+ // Fetching the dom element works in Firefox, but is not implemented
+ // in webkit.
+ npapi::GetValue(instance, NPNVDOMElement, elem.Receive());
+ if (!elem.get()) {
+ DLOG(INFO) << "Failed to get NPNVDOMElement";
+ return false;
+ }
+
+ nsEmbedString tag;
+ nsresult res = elem->GetTagName(tag);
+ if (NS_SUCCEEDED(res)) {
+ if (LowerCaseEqualsASCII(tag.get(), tag.get() + tag.Length(), "embed")) {
+ ScopedNsPtr<nsIDOMNode> parent;
+ elem->GetParentNode(parent.Receive());
+ if (parent) {
+ elem.Release();
+ res = parent.QueryInterface(elem.Receive());
+ DCHECK(NS_SUCCEEDED(res));
+ }
+ }
+ } else {
+ NOTREACHED() << " GetTagName";
+ }
+
+ *element = elem.Detach();
+
+ return *element != NULL;
+}
+
+///////////////////////////////////
+// NPObjectEventListener
+
+NPObjectEventListener::NPObjectEventListener(NpEventDelegate* delegate)
+ : NpEventListenerBase<NPObjectEventListener>(delegate) {
+}
+
+NPObjectEventListener::~NPObjectEventListener() {
+ DLOG_IF(ERROR, npo_.get() == NULL);
+}
+
+NPObject* NPObjectEventListener::GetObjectElement(NPP instance) {
+ NPObject* object = NULL;
+ // We can't trust the return value from getvalue.
+ // In Opera, the return value can be false even though the correct
+ // object is returned.
+ npapi::GetValue(instance, NPNVPluginElementNPObject, &object);
+
+ if (object) {
+ NPIdentifier* ids = GetCachedStringIds();
+ NPVariant var;
+ if (npapi::GetProperty(instance, object, ids[TAG_NAME], &var)) {
+ DCHECK(NPVARIANT_IS_STRING(var));
+ const NPString& np_tag = NPVARIANT_TO_STRING(var);
+ std::string tag(np_tag.UTF8Characters, np_tag.UTF8Length);
+ npapi::ReleaseVariantValue(&var);
+
+ if (lstrcmpiA(tag.c_str(), "embed") == 0) {
+ // We've got the <embed> element but we really want
+ // the <object> element.
+ if (npapi::GetProperty(instance, object, ids[PARENT_ELEMENT], &var)) {
+ DCHECK(NPVARIANT_IS_OBJECT(var));
+ npapi::ReleaseObject(object);
+ object = NPVARIANT_TO_OBJECT(var);
+ }
+ } else {
+ DLOG(INFO) << __FUNCTION__ << " got " << tag;
+ }
+ } else {
+ DLOG(INFO) << __FUNCTION__ << " failed to get the element's tag";
+ }
+ } else {
+ DLOG(WARNING) << __FUNCTION__ << " failed to get NPNVPluginElementNPObject";
+ }
+
+ return object;
+}
+
+// Implementation of NpEventListener
+bool NPObjectEventListener::Subscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count) {
+ DCHECK(event_names);
+ DCHECK(event_name_count > 0);
+ DCHECK(npo_.get() == NULL);
+
+ ScopedNpObject<> plugin_element(GetObjectElement(instance));
+ if (!plugin_element.get())
+ return false;
+
+ // This object seems to be getting leaked :-(
+ bool ret = false;
+ npo_.Attach(reinterpret_cast<Npo*>(
+ npapi::CreateObject(instance, PluginClass())));
+
+ if (!npo_.get()) {
+ NOTREACHED() << "createobject";
+ } else {
+ npo_->Initialize(this);
+ ret = true;
+
+ NPIdentifier* ids = GetCachedStringIds();
+
+ NPVariant args[3];
+ OBJECT_TO_NPVARIANT(npo_, args[1]);
+ // We don't want to set 'capture' (last parameter) to true.
+ // If we do, then in Opera, we'll simply not get callbacks unless
+ // the target <object> or <embed> element we're syncing with has its
+ // on[event] property assigned to some function handler. weird.
+ // Ideally though we'd like to set capture to true since we'd like to
+ // only be triggered for this particular object (and not for bubbling
+ // events, but alas it's not meant to be.
+ BOOLEAN_TO_NPVARIANT(false, args[2]);
+ for (int i = 0; i < event_name_count; ++i) {
+ ScopedNpVariant result;
+ STRINGZ_TO_NPVARIANT(event_names[i], args[0]);
+ ret = npapi::Invoke(instance, plugin_element, ids[ADD_EVENT_LISTENER],
+ args, arraysize(args), &result) && ret;
+ if (!ret) {
+ DLOG(WARNING) << __FUNCTION__ << " invoke failed for "
+ << event_names[i];
+ break;
+ }
+ }
+ }
+
+ return ret;
+}
+
+bool NPObjectEventListener::Unsubscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count) {
+ DCHECK(event_names);
+ DCHECK(event_name_count > 0);
+ DCHECK(npo_.get() != NULL);
+
+ ScopedNpObject<> plugin_element(GetObjectElement(instance));
+ if (!plugin_element.get())
+ return false;
+
+ NPIdentifier* ids = GetCachedStringIds();
+
+ NPVariant args[3];
+ OBJECT_TO_NPVARIANT(npo_, args[1]);
+ BOOLEAN_TO_NPVARIANT(false, args[2]);
+ for (int i = 0; i < event_name_count; ++i) {
+ // TODO(tommi): look into why chrome isn't releasing the reference
+ // count here. As it stands the reference count doesn't go down
+ // and as a result, the NPO gets leaked.
+ ScopedNpVariant result;
+ STRINGZ_TO_NPVARIANT(event_names[i], args[0]);
+ bool ret = npapi::Invoke(instance, plugin_element,
+ ids[REMOVE_EVENT_LISTENER], args, arraysize(args), &result);
+ DLOG_IF(ERROR, !ret) << __FUNCTION__ << " invoke: " << ret;
+ }
+
+ npo_.Free();
+
+ return true;
+}
+
+void NPObjectEventListener::HandleEvent(Npo* npo, NPObject* event) {
+ DCHECK(npo);
+ DCHECK(event);
+
+ NPIdentifier* ids = GetCachedStringIds();
+ ScopedNpVariant result;
+ bool ret = npapi::GetProperty(npo->npp(), event, ids[TYPE], &result);
+ DCHECK(ret) << "getproperty(type)";
+ if (ret) {
+ DCHECK(NPVARIANT_IS_STRING(result));
+ // Opera doesn't zero terminate its utf8 strings.
+ const NPString& type = NPVARIANT_TO_STRING(result);
+ std::string zero_terminated(type.UTF8Characters, type.UTF8Length);
+ DLOG(INFO) << "handleEvent: " << zero_terminated;
+ delegate_->OnEvent(zero_terminated.c_str());
+ }
+}
+
+NPClass* NPObjectEventListener::PluginClass() {
+ static NPClass _np_class = {
+ NP_CLASS_STRUCT_VERSION,
+ reinterpret_cast<NPAllocateFunctionPtr>(AllocateObject),
+ reinterpret_cast<NPDeallocateFunctionPtr>(DeallocateObject),
+ NULL, // invalidate
+ reinterpret_cast<NPHasMethodFunctionPtr>(HasMethod),
+ reinterpret_cast<NPInvokeFunctionPtr>(Invoke),
+ NULL, // InvokeDefault,
+ NULL, // HasProperty,
+ NULL, // GetProperty,
+ NULL, // SetProperty,
+ NULL // construct
+ };
+
+ return &_np_class;
+}
+
+bool NPObjectEventListener::HasMethod(NPObjectEventListener::Npo* npo,
+ NPIdentifier name) {
+ NPIdentifier* ids = GetCachedStringIds();
+ if (name == ids[HANDLE_EVENT])
+ return true;
+
+ return false;
+}
+
+bool NPObjectEventListener::Invoke(NPObjectEventListener::Npo* npo,
+ NPIdentifier name, const NPVariant* args,
+ uint32_t arg_count, NPVariant* result) {
+ NPIdentifier* ids = GetCachedStringIds();
+ if (name != ids[HANDLE_EVENT])
+ return false;
+
+ if (arg_count != 1 || !NPVARIANT_IS_OBJECT(args[0])) {
+ NOTREACHED();
+ } else {
+ NPObject* ev = NPVARIANT_TO_OBJECT(args[0]);
+ npo->listener()->HandleEvent(npo, ev);
+ }
+
+ return true;
+}
+
+NPObject* NPObjectEventListener::AllocateObject(NPP instance,
+ NPClass* class_name) {
+ return new Npo(instance);
+}
+
+void NPObjectEventListener::DeallocateObject(NPObjectEventListener::Npo* npo) {
+ delete npo;
+}
+
+NPIdentifier* NPObjectEventListener::GetCachedStringIds() {
+ static NPIdentifier _identifiers[IDENTIFIER_COUNT] = {0};
+ if (!_identifiers[0]) {
+ const NPUTF8* identifier_names[] = {
+ "handleEvent",
+ "type",
+ "addEventListener",
+ "removeEventListener",
+ "tagName",
+ "parentElement",
+ };
+ COMPILE_ASSERT(arraysize(identifier_names) == arraysize(_identifiers),
+ mismatched_array_size);
+ npapi::GetStringIdentifiers(identifier_names, IDENTIFIER_COUNT,
+ _identifiers);
+ for (int i = 0; i < IDENTIFIER_COUNT; ++i) {
+ DCHECK(_identifiers[i] != 0);
+ }
+ }
+ return _identifiers;
+}
diff --git a/chrome_frame/np_event_listener.h b/chrome_frame/np_event_listener.h
new file mode 100644
index 0000000..18cb4eb
--- /dev/null
+++ b/chrome_frame/np_event_listener.h
@@ -0,0 +1,187 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_NP_EVENT_LISTENER_H_
+#define CHROME_FRAME_NP_EVENT_LISTENER_H_
+
+#include "base/logging.h"
+
+#include "chrome_frame/utils.h"
+#include "chrome_frame/np_browser_functions.h"
+
+// Avoid conflicts with basictypes and the gecko sdk.
+// (different definitions of uint32).
+#define NO_NSPR_10_SUPPORT
+#include "third_party/xulrunner-sdk/win/include/dom/nsIDOMEventListener.h"
+
+
+class nsIDOMElement;
+
+class NpEventDelegate {
+ public:
+ virtual void OnEvent(const char* event_name) = 0;
+};
+
+class NpEventListener {
+ public:
+ NS_IMETHOD_(nsrefcnt) AddRef() = 0;
+ NS_IMETHOD_(nsrefcnt) Release() = 0;
+ virtual bool Subscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count) = 0;
+ virtual bool Unsubscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count) = 0;
+};
+
+// A little helper class to implement simple ref counting
+// and assert on single threadedness.
+template <class T>
+class NpEventListenerBase : public NpEventListener {
+ public:
+ NpEventListenerBase(NpEventDelegate* delegate)
+ : ref_count_(0), delegate_(delegate) {
+ DCHECK(delegate_);
+ thread_id_ = ::GetCurrentThreadId();
+ }
+
+ ~NpEventListenerBase() {
+ DCHECK(thread_id_ == ::GetCurrentThreadId());
+ }
+
+ NS_IMETHOD_(nsrefcnt) AddRef() {
+ DCHECK(thread_id_ == ::GetCurrentThreadId());
+ ref_count_++;
+ return ref_count_;
+ }
+
+ NS_IMETHOD_(nsrefcnt) Release() {
+ DCHECK(thread_id_ == ::GetCurrentThreadId());
+ ref_count_--;
+
+ if (!ref_count_) {
+ T* me = static_cast<T*>(this);
+ delete me;
+ return 0;
+ }
+
+ return ref_count_;
+ }
+
+ protected:
+ nsrefcnt ref_count_;
+ NpEventDelegate* delegate_;
+ AddRefModule module_ref_;
+ // used to DCHECK on expected single-threaded usage
+ unsigned long thread_id_;
+};
+
+// Implements nsIDOMEventListener in order to receive events from DOM
+// elements inside an HTML page.
+class DomEventListener
+ : public nsIDOMEventListener,
+ public NpEventListenerBase<DomEventListener> {
+ public:
+ DomEventListener(NpEventDelegate* delegate);
+ ~DomEventListener();
+
+ // Implementation of NpEventListener
+ virtual bool Subscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count);
+ virtual bool Unsubscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count);
+ protected:
+ // We implement QueryInterface etc ourselves in order to avoid
+ // extra dependencies brought on by the NS_IMPL_* macros.
+ NS_IMETHOD QueryInterface(REFNSIID iid, void** ptr);
+ NS_IMETHOD_(nsrefcnt) AddRef() {
+ return NpEventListenerBase<DomEventListener>::AddRef();
+ }
+
+ NS_IMETHOD_(nsrefcnt) Release() {
+ return NpEventListenerBase<DomEventListener>::Release();
+ }
+
+ // Implementation of nsIDOMEventListener
+ NS_IMETHOD HandleEvent(nsIDOMEvent *event);
+
+ private:
+ static bool GetObjectElement(NPP instance, nsIDOMElement** element);
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(DomEventListener);
+};
+
+class NPObjectEventListener
+ : public NpEventListenerBase<NPObjectEventListener> {
+ public:
+ NPObjectEventListener(NpEventDelegate* delegate);
+ ~NPObjectEventListener();
+
+ // Implementation of NpEventListener
+ virtual bool Subscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count);
+ virtual bool Unsubscribe(NPP instance,
+ const char* event_names[],
+ int event_name_count);
+
+ protected:
+ // NPObject structure which is exposed by NPObjectEventListener.
+ class Npo : public NPObject {
+ public:
+ Npo(NPP npp) : npp_(npp), listener_(NULL) {
+ }
+
+ void Initialize(NPObjectEventListener* listener) {
+ listener_ = listener;
+ }
+
+ inline NPObjectEventListener* listener() const {
+ return listener_;
+ }
+
+ inline NPP npp() const {
+ return npp_;
+ }
+
+ protected:
+ NPP npp_;
+ NPObjectEventListener* listener_;
+ AddRefModule module_ref_;
+ };
+
+ static NPClass* PluginClass();
+
+ static bool HasMethod(Npo* npo, NPIdentifier name);
+ static bool Invoke(Npo* npo, NPIdentifier name, const NPVariant* args,
+ uint32_t arg_count, NPVariant* result);
+ static NPObject* AllocateObject(NPP instance, NPClass* class_name);
+ static void DeallocateObject(Npo* npo);
+
+ typedef enum {
+ HANDLE_EVENT,
+ TYPE,
+ ADD_EVENT_LISTENER,
+ REMOVE_EVENT_LISTENER,
+ TAG_NAME,
+ PARENT_ELEMENT,
+ IDENTIFIER_COUNT,
+ } CachedStringIdentifiers;
+
+ static NPIdentifier* GetCachedStringIds();
+
+ void HandleEvent(Npo* npo, NPObject* event);
+ static NPObject* GetObjectElement(NPP instance);
+
+ private:
+ // Our NPObject.
+ ScopedNpObject<Npo> npo_;
+
+ DISALLOW_COPY_AND_ASSIGN(NPObjectEventListener);
+};
+
+#endif // CHROME_FRAME_NP_EVENT_LISTENER_H_
diff --git a/chrome_frame/np_proxy_service.cc b/chrome_frame/np_proxy_service.cc
new file mode 100644
index 0000000..cb26dac
--- /dev/null
+++ b/chrome_frame/np_proxy_service.cc
@@ -0,0 +1,306 @@
+// Copyright (c) 2009 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 "base/string_util.h"
+#include "chrome/common/automation_constants.h"
+#include "chrome/common/json_value_serializer.h"
+#include "chrome_frame/np_proxy_service.h"
+#include "chrome_frame/np_browser_functions.h"
+
+#include "net/proxy/proxy_config.h"
+
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsXPCOM.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsIObserverService.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsISupportsUtils.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsStringAPI.h"
+
+ASSOCIATE_IID(NS_IOBSERVERSERVICE_IID_STR, nsIObserverService);
+ASSOCIATE_IID(NS_IPREFBRANCH_IID_STR, nsIPrefBranch);
+
+// Firefox preference names.
+const char* kProxyObserverRoot = "network.";
+const char* kProxyObserverBranch = "proxy.";
+const char* kProxyType = "proxy.type";
+const char* kProxyAutoconfigUrl = "proxy.autoconfig_url";
+const char* kProxyBypassList = "proxy.no_proxies_on";
+
+const int kInvalidIntPref = -1;
+
+// These are the proxy schemes that Chrome knows about at the moment.
+// SOCKS is a notable ommission here, this will need to be updated when
+// Chrome supports SOCKS proxies.
+const NpProxyService::ProxyNames NpProxyService::kProxyInfo[] = {
+ {"http", "proxy.http", "proxy.http_port"},
+ {"https", "proxy.ssl", "proxy.ssl_port"},
+ {"ftp", "proxy.ftp", "proxy.ftp_port"} };
+
+NpProxyService::NpProxyService(void)
+ : type_(PROXY_CONFIG_LAST), auto_detect_(false), no_proxy_(false),
+ system_config_(false), automation_client_(NULL) {
+}
+
+NpProxyService::~NpProxyService(void) {
+}
+
+bool NpProxyService::Initialize(NPP instance,
+ ChromeFrameAutomationClient* automation_client) {
+ DCHECK(automation_client);
+ automation_client_ = automation_client;
+
+ // Get the pref service
+ bool result = false;
+ ScopedNsPtr<nsISupports> service_manager_base;
+ npapi::GetValue(instance, NPNVserviceManager, service_manager_base.Receive());
+ if (service_manager_base != NULL) {
+ service_manager_.QueryFrom(service_manager_base);
+ if (service_manager_.get() == NULL) {
+ DLOG(ERROR) << "Failed to create ServiceManager. This only works in FF.";
+ } else {
+ service_manager_->GetServiceByContractID(
+ NS_PREFSERVICE_CONTRACTID, NS_GET_IID(nsIPrefService),
+ reinterpret_cast<void**>(pref_service_.Receive()));
+ if (!pref_service_) {
+ DLOG(ERROR) << "Failed to create PreferencesService";
+ } else {
+ result = InitializePrefBranch(pref_service_);
+ }
+ }
+ }
+ return result;
+}
+
+bool NpProxyService::InitializePrefBranch(nsIPrefService* pref_service) {
+ DCHECK(pref_service);
+ // Note that we cannot persist a reference to the pref branch because we
+ // also act as an observer of changes to the branch. As per
+ // nsIPrefBranch2.h, this would result in a circular reference between us
+ // and the pref branch, which can impede cleanup. There are workarounds,
+ // but let's try just not caching the branch reference for now.
+ bool result = false;
+ ScopedNsPtr<nsIPrefBranch> pref_branch;
+
+ pref_service->ReadUserPrefs(nsnull);
+ pref_service->GetBranch(kProxyObserverRoot, pref_branch.Receive());
+
+ if (!pref_branch) {
+ DLOG(ERROR) << "Failed to get nsIPrefBranch";
+ } else {
+ if (!ReadProxySettings(pref_branch.get())) {
+ DLOG(ERROR) << "Could not read proxy settings.";
+ } else {
+ observer_pref_branch_.QueryFrom(pref_branch);
+ if (!observer_pref_branch_) {
+ DLOG(ERROR) << "Failed to get observer nsIPrefBranch2";
+ } else {
+ nsresult res = observer_pref_branch_->AddObserver(kProxyObserverBranch,
+ this, PR_FALSE);
+ result = NS_SUCCEEDED(res);
+ }
+ }
+ }
+ return result;
+}
+
+bool NpProxyService::UnInitialize() {
+ // Fail early if this was never created - we may not be running on FF.
+ if (!pref_service_)
+ return false;
+
+ // Unhook ourselves as an observer.
+ nsresult res = NS_ERROR_FAILURE;
+ if (observer_pref_branch_)
+ res = observer_pref_branch_->RemoveObserver(kProxyObserverBranch, this);
+
+ return NS_SUCCEEDED(res);
+}
+
+NS_IMETHODIMP NpProxyService::Observe(nsISupports* subject, const char* topic,
+ const PRUnichar* data) {
+ if (!subject || !topic) {
+ NOTREACHED();
+ return NS_ERROR_UNEXPECTED;
+ }
+
+ std::string topic_str(topic);
+ nsresult res = NS_OK;
+ if (topic_str == NS_PREFBRANCH_PREFCHANGE_TOPIC_ID) {
+ // Looks like our proxy settings changed. We need to reload!
+ // I have observed some extremely strange behaviour here. Specifically,
+ // we are supposed to be able to QI |subject| and get from it an
+ // nsIPrefBranch from which we can query new values. This has erratic
+ // behaviour, specifically subject starts returning null on all member
+ // queries. So I am using the cached nsIPrefBranch2 (that we used to add
+ // the observer) to do the querying.
+ if (NS_SUCCEEDED(res)) {
+ if (!ReadProxySettings(observer_pref_branch_)) {
+ res = NS_ERROR_UNEXPECTED;
+ } else {
+ std::string proxy_settings;
+ if (GetProxyValueJSONString(&proxy_settings))
+ automation_client_->SetProxySettings(proxy_settings);
+ }
+ }
+ } else {
+ NOTREACHED();
+ }
+
+ return res;
+}
+
+std::string NpProxyService::GetStringPref(nsIPrefBranch* pref_branch,
+ const char* pref_name) {
+ nsCString pref_string;
+ std::string result;
+ nsresult rv = pref_branch->GetCharPref(pref_name, getter_Copies(pref_string));
+ if (SUCCEEDED(rv) && pref_string.get()) {
+ result = pref_string.get();
+ }
+ return result;
+}
+
+int NpProxyService::GetIntPref(nsIPrefBranch* pref_branch,
+ const char* pref_name) {
+ PRInt32 pref_int;
+ int result = kInvalidIntPref;
+ nsresult rv = pref_branch->GetIntPref(pref_name, &pref_int);
+ if (SUCCEEDED(rv)) {
+ result = pref_int;
+ }
+ return result;
+}
+
+bool NpProxyService::GetBoolPref(nsIPrefBranch* pref_branch,
+ const char* pref_name) {
+ PRBool pref_bool;
+ bool result = false;
+ nsresult rv = pref_branch->GetBoolPref(pref_name, &pref_bool);
+ if (SUCCEEDED(rv)) {
+ result = pref_bool == PR_TRUE;
+ }
+ return result;
+}
+
+void NpProxyService::Reset() {
+ type_ = PROXY_CONFIG_LAST;
+ auto_detect_ = false;
+ no_proxy_ = false;
+ system_config_ = false;
+ manual_proxies_.clear();
+ pac_url_.clear();
+ proxy_bypass_list_.clear();
+}
+
+bool NpProxyService::ReadProxySettings(nsIPrefBranch* pref_branch) {
+ DCHECK(pref_branch);
+
+ // Clear our current settings.
+ Reset();
+ type_ = GetIntPref(pref_branch, kProxyType);
+ if (type_ == kInvalidIntPref) {
+ NOTREACHED();
+ return false;
+ }
+
+ switch (type_) {
+ case PROXY_CONFIG_DIRECT:
+ case PROXY_CONFIG_DIRECT4X:
+ no_proxy_ = true;
+ break;
+ case PROXY_CONFIG_SYSTEM:
+ // _SYSTEM is documented as "Use system settings if available, otherwise
+ // DIRECT". It isn't clear under what circumstances system settings would
+ // be unavailable, but I'll special-case this nonetheless and have
+ // GetProxyValueJSONString() return empty if we get this proxy type.
+ DLOG(WARNING) << "Received PROXY_CONFIG_SYSTEM proxy type.";
+ system_config_ = true;
+ break;
+ case PROXY_CONFIG_WPAD:
+ auto_detect_ = true;
+ break;
+ case PROXY_CONFIG_PAC:
+ pac_url_ = GetStringPref(pref_branch, kProxyAutoconfigUrl);
+ break;
+ case PROXY_CONFIG_MANUAL:
+ // Read in the values for each of the known schemes.
+ for (int i = 0; i < arraysize(kProxyInfo); i++) {
+ ManualProxyEntry entry;
+ entry.url = GetStringPref(pref_branch, kProxyInfo[i].pref_name);
+ entry.port = GetIntPref(pref_branch, kProxyInfo[i].port_pref_name);
+ if (!entry.url.empty() && entry.port != kInvalidIntPref) {
+ entry.scheme = kProxyInfo[i].chrome_scheme;
+ manual_proxies_.push_back(entry);
+ }
+ }
+
+ // Also pick up the list of URLs we bypass proxies for.
+ proxy_bypass_list_ = GetStringPref(pref_branch, kProxyBypassList);
+ break;
+ default:
+ NOTREACHED();
+ return false;
+ }
+ return true;
+}
+
+DictionaryValue* NpProxyService::BuildProxyValueSet() {
+ scoped_ptr<DictionaryValue> proxy_settings_value(new DictionaryValue);
+
+ if (auto_detect_) {
+ proxy_settings_value->SetBoolean(automation::kJSONProxyAutoconfig,
+ auto_detect_);
+ }
+
+ if (no_proxy_) {
+ proxy_settings_value->SetBoolean(automation::kJSONProxyNoProxy, no_proxy_);
+ }
+
+ if (!pac_url_.empty()) {
+ proxy_settings_value->SetString(automation::kJSONProxyPacUrl, pac_url_);
+ }
+
+ if (!proxy_bypass_list_.empty()) {
+ proxy_settings_value->SetString(automation::kJSONProxyBypassList,
+ proxy_bypass_list_);
+ }
+
+ // Fill in the manual proxy settings. Build a string representation that
+ // corresponds to the format of the input parameter to
+ // ProxyConfig::ProxyRules::ParseFromString.
+ std::string manual_proxy_settings;
+ ManualProxyList::const_iterator iter(manual_proxies_.begin());
+ for (; iter != manual_proxies_.end(); iter++) {
+ DCHECK(!iter->scheme.empty());
+ DCHECK(!iter->url.empty());
+ DCHECK(iter->port != kInvalidIntPref);
+ manual_proxy_settings += iter->scheme;
+ manual_proxy_settings += "=";
+ manual_proxy_settings += iter->url;
+ manual_proxy_settings += ":";
+ manual_proxy_settings += IntToString(iter->port);
+ manual_proxy_settings += ";";
+ }
+
+ if (!manual_proxy_settings.empty()) {
+ proxy_settings_value->SetString(automation::kJSONProxyServer,
+ manual_proxy_settings);
+ }
+
+ return proxy_settings_value.release();
+}
+
+bool NpProxyService::GetProxyValueJSONString(std::string* output) {
+ DCHECK(output);
+ output->empty();
+
+ // If we detected a PROXY_CONFIG_SYSTEM config type or failed to obtain the
+ // pref service then return false here to make Chrome continue using its
+ // default proxy settings.
+ if (system_config_ || !pref_service_)
+ return false;
+
+ scoped_ptr<DictionaryValue> proxy_settings_value(BuildProxyValueSet());
+
+ JSONStringValueSerializer serializer(output);
+ return serializer.Serialize(*static_cast<Value*>(proxy_settings_value.get()));
+}
diff --git a/chrome_frame/np_proxy_service.h b/chrome_frame/np_proxy_service.h
new file mode 100644
index 0000000..c0a85c8
--- /dev/null
+++ b/chrome_frame/np_proxy_service.h
@@ -0,0 +1,137 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_NP_PROXY_SERVICE_H_
+#define CHROME_FRAME_NP_PROXY_SERVICE_H_
+
+#include <string>
+#include <vector>
+#include "base/values.h"
+#include "base/scoped_ptr.h"
+
+// Avoid conflicts with basictypes and the gecko sdk.
+// (different definitions of uint32).
+#define NO_NSPR_10_SUPPORT
+
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/ns_associate_iid_win.h"
+#include "chrome_frame/ns_isupports_impl.h"
+#include "chrome_frame/scoped_ns_ptr_win.h"
+#include "third_party/WebKit/WebCore/bridge/npapi.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsIObserver.h"
+#include "third_party/xulrunner-sdk/win/include/pref/nsIPrefBranch2.h"
+#include "third_party/xulrunner-sdk/win/include/pref/nsIPrefService.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsIServiceManager.h"
+
+ASSOCIATE_IID(NS_IOBSERVER_IID_STR, nsIObserver);
+ASSOCIATE_IID(NS_ISERVICEMANAGER_IID_STR, nsIServiceManager);
+ASSOCIATE_IID(NS_IPREFSERVICE_IID_STR, nsIPrefService);
+ASSOCIATE_IID(NS_IPREFBRANCH2_IID_STR, nsIPrefBranch2);
+
+class nsIServiceManager;
+class nsIPrefService;
+class nsIPrefBranch2;
+
+// This class reads in proxy settings from firefox.
+// TODO(robertshield): The change notification code is currently broken.
+// Fix it and implement calling back through to the automation proxy with
+// proxy updates.
+class NpProxyService : public NsISupportsImplBase<NpProxyService>,
+ public nsIObserver {
+ public:
+ // These values correspond to the integer network.proxy.type preference.
+ enum ProxyConfigType {
+ PROXY_CONFIG_DIRECT,
+ PROXY_CONFIG_MANUAL,
+ PROXY_CONFIG_PAC,
+ PROXY_CONFIG_DIRECT4X,
+ PROXY_CONFIG_WPAD,
+ PROXY_CONFIG_SYSTEM, // use system settings if available, otherwise DIRECT
+ PROXY_CONFIG_LAST
+ };
+
+ // nsISupports
+ NS_IMETHODIMP_(nsrefcnt) AddRef(void) {
+ return NsISupportsImplBase<NpProxyService>::AddRef();
+ }
+
+ NS_IMETHODIMP_(nsrefcnt) Release(void) {
+ return NsISupportsImplBase<NpProxyService>::Release();
+ }
+
+ NS_IMETHOD QueryInterface(REFNSIID iid, void** ptr) {
+ nsresult res =
+ NsISupportsImplBase<NpProxyService>::QueryInterface(iid, ptr);
+ if (NS_FAILED(res) &&
+ memcmp(&iid, &__uuidof(nsIObserver), sizeof(nsIID)) == 0) {
+ *ptr = static_cast<nsIObserver*>(this);
+ AddRef();
+ res = NS_OK;
+ }
+ return res;
+ }
+
+ // nsIObserver
+ NS_IMETHOD Observe(nsISupports* subject, const char* topic,
+ const PRUnichar* data);
+
+ NpProxyService();
+ ~NpProxyService();
+
+ virtual bool Initialize(NPP instance,
+ ChromeFrameAutomationClient* automation_client);
+ bool UnInitialize();
+
+ // Places the current Firefox settings as a JSON string suitable for posting
+ // over to Chromium into output. Returns true if the settings were correctly
+ // serialized into a JSON string, false otherwise.
+ // TODO(robertshield): I haven't yet nailed down how much of this should go
+ // here and how much should go in the AutomationProxy. Will do that in a
+ // near-future patch.
+ bool GetProxyValueJSONString(std::string* output);
+
+ private:
+ bool InitializePrefBranch(nsIPrefService* pref_service);
+ bool ReadProxySettings(nsIPrefBranch* pref_branch);
+
+ std::string GetStringPref(nsIPrefBranch* pref_branch, const char* pref_name);
+ int GetIntPref(nsIPrefBranch* pref_branch, const char* pref_name);
+ bool GetBoolPref(nsIPrefBranch* pref_branch, const char* pref_name);
+
+ void Reset();
+ DictionaryValue* BuildProxyValueSet();
+
+ ChromeFrameAutomationClient* automation_client_;
+
+ ScopedNsPtr<nsIServiceManager> service_manager_;
+ ScopedNsPtr<nsIPrefService> pref_service_;
+ ScopedNsPtr<nsIPrefBranch2> observer_pref_branch_;
+
+ struct ProxyNames {
+ // Proxy type (http, https, ftp, etc...).
+ const char* chrome_scheme;
+ // Firefox preference name of the URL for this proxy type.
+ const char* pref_name;
+ // Firefox preference name for the port for this proxy type.
+ const char* port_pref_name;
+ };
+ static const ProxyNames kProxyInfo[];
+
+ struct ManualProxyEntry {
+ std::string scheme;
+ std::string url;
+ int port;
+ };
+ typedef std::vector<ManualProxyEntry> ManualProxyList;
+
+ bool system_config_;
+ bool auto_detect_;
+ bool no_proxy_;
+ int type_;
+ std::string pac_url_;
+ std::string proxy_bypass_list_;
+ ManualProxyList manual_proxies_;
+};
+
+#endif // CHROME_FRAME_NP_PROXY_SERVICE_H_
diff --git a/chrome_frame/npapi_url_request.cc b/chrome_frame/npapi_url_request.cc
new file mode 100644
index 0000000..33052a1
--- /dev/null
+++ b/chrome_frame/npapi_url_request.cc
@@ -0,0 +1,157 @@
+// Copyright 2009, Google Inc.
+// All rights reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following disclaimer
+// in the documentation and/or other materials provided with the
+// distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived from
+// this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+#include "chrome_frame/npapi_url_request.h"
+
+#include "base/string_util.h"
+#include "chrome_frame/np_browser_functions.h"
+#include "net/base/net_errors.h"
+
+int NPAPIUrlRequest::instance_count_ = 0;
+
+NPAPIUrlRequest::NPAPIUrlRequest(NPP instance)
+ : ref_count_(0), instance_(instance), stream_(NULL),
+ pending_read_size_(0),
+ status_(URLRequestStatus::FAILED, net::ERR_FAILED),
+ thread_(PlatformThread::CurrentId()) {
+ DLOG(INFO) << "Created request. Count: " << ++instance_count_;
+}
+
+NPAPIUrlRequest::~NPAPIUrlRequest() {
+ DLOG(INFO) << "Deleted request. Count: " << --instance_count_;
+}
+
+// NPAPIUrlRequest member defines.
+bool NPAPIUrlRequest::Start() {
+ NPError result = NPERR_GENERIC_ERROR;
+ DLOG(INFO) << "Starting URL request: " << url();
+ if (LowerCaseEqualsASCII(method(), "get")) {
+ // TODO(joshia): if we have extra headers for HTTP GET, then implement
+ // it using XHR
+ result = npapi::GetURLNotify(instance_, url().c_str(), NULL, this);
+ } else if (LowerCaseEqualsASCII(method(), "post")) {
+ result = npapi::PostURLNotify(instance_, url().c_str(), NULL,
+ extra_headers().length(), extra_headers().c_str(), false, this);
+ } else {
+ NOTREACHED() << "PluginUrlRequest only supports 'GET'/'POST'";
+ }
+
+ if (NPERR_NO_ERROR == result) {
+ request_handler()->AddRequest(this);
+ } else {
+ int os_error = net::ERR_FAILED;
+ switch (result) {
+ case NPERR_INVALID_URL:
+ os_error = net::ERR_INVALID_URL;
+ break;
+ default:
+ break;
+ }
+
+ OnResponseEnd(URLRequestStatus(URLRequestStatus::FAILED, os_error));
+ return false;
+ }
+
+ return true;
+}
+
+void NPAPIUrlRequest::Stop() {
+ DLOG(INFO) << "Finished request: Url - " << url() << " result: "
+ << static_cast<int>(status_.status());
+ if (stream_) {
+ npapi::DestroyStream(instance_, stream_, NPRES_USER_BREAK);
+ stream_ = NULL;
+ }
+
+ request_handler()->RemoveRequest(this);
+ if (!status_.is_io_pending())
+ OnResponseEnd(status_);
+}
+
+bool NPAPIUrlRequest::Read(int bytes_to_read) {
+ pending_read_size_ = bytes_to_read;
+ return true;
+}
+
+bool NPAPIUrlRequest::OnStreamCreated(const char* mime_type, NPStream* stream) {
+ stream_ = stream;
+ status_.set_status(URLRequestStatus::IO_PENDING);
+ // TODO(iyengar)
+ // Add support for passing persistent cookies and information about any URL
+ // redirects to Chrome.
+ OnResponseStarted(mime_type, stream->headers, stream->end,
+ base::Time::FromTimeT(stream->lastmodified), std::string(),
+ std::string(), 0);
+ return true;
+}
+
+void NPAPIUrlRequest::OnStreamDestroyed(NPReason reason) {
+ URLRequestStatus::Status status = URLRequestStatus::FAILED;
+ switch (reason) {
+ case NPRES_DONE:
+ status_.set_status(URLRequestStatus::SUCCESS);
+ status_.set_os_error(0);
+ break;
+ case NPRES_USER_BREAK:
+ status_.set_status(URLRequestStatus::CANCELED);
+ status_.set_os_error(net::ERR_ABORTED);
+ break;
+ case NPRES_NETWORK_ERR:
+ default:
+ status_.set_status(URLRequestStatus::FAILED);
+ status_.set_os_error(net::ERR_CONNECTION_CLOSED);
+ break;
+ }
+}
+
+int NPAPIUrlRequest::OnWriteReady() {
+ return pending_read_size_;
+}
+
+int NPAPIUrlRequest::OnWrite(void* buffer, int len) {
+ pending_read_size_ = 0;
+ OnReadComplete(buffer, len);
+ return len;
+}
+
+STDMETHODIMP_(ULONG) NPAPIUrlRequest::AddRef() {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+ ++ref_count_;
+ return ref_count_;
+}
+
+STDMETHODIMP_(ULONG) NPAPIUrlRequest::Release() {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+ unsigned long ret = --ref_count_;
+ if (!ret)
+ delete this;
+
+ return ret;
+}
+
diff --git a/chrome_frame/npapi_url_request.h b/chrome_frame/npapi_url_request.h
new file mode 100644
index 0000000..9151fcf
--- /dev/null
+++ b/chrome_frame/npapi_url_request.h
@@ -0,0 +1,45 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_NPAPI_URL_REQUEST_H_
+#define CHROME_FRAME_NPAPI_URL_REQUEST_H_
+
+#include "base/platform_thread.h"
+#include "chrome_frame/plugin_url_request.h"
+#include "third_party/WebKit/WebCore/bridge/npapi.h"
+
+class NPAPIUrlRequest : public PluginUrlRequest {
+ public:
+ explicit NPAPIUrlRequest(NPP instance);
+ ~NPAPIUrlRequest();
+
+ virtual bool Start();
+ virtual void Stop();
+ virtual bool Read(int bytes_to_read);
+
+ // Called from NPAPI
+ bool OnStreamCreated(const char* mime_type, NPStream* stream);
+ void OnStreamDestroyed(NPReason reason);
+ int OnWriteReady();
+ int OnWrite(void* buffer, int len);
+
+ // Thread unsafe implementation of ref counting, since
+ // this will be called on the plugin UI thread only.
+ virtual unsigned long API_CALL AddRef();
+ virtual unsigned long API_CALL Release();
+
+ private:
+ unsigned long ref_count_;
+ NPP instance_;
+ NPStream* stream_;
+ size_t pending_read_size_;
+ URLRequestStatus status_;
+
+ PlatformThreadId thread_;
+ static int instance_count_;
+ DISALLOW_COPY_AND_ASSIGN(NPAPIUrlRequest);
+};
+
+#endif // CHROME_FRAME_NPAPI_URL_REQUEST_H_
+
diff --git a/chrome_frame/ns_associate_iid_win.h b/chrome_frame/ns_associate_iid_win.h
new file mode 100644
index 0000000..db6c234
--- /dev/null
+++ b/chrome_frame/ns_associate_iid_win.h
@@ -0,0 +1,15 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_NS_ASSOCIATE_IID_WIN_H_
+#define CHROME_FRAME_NS_ASSOCIATE_IID_WIN_H_
+
+#define ASSOCIATE_IID(iid, class_name) class __declspec(uuid(iid)) class_name
+
+// Associate IIDs with ns interfaces for convenience.
+// This makes ScopedNsPtr more user friendly and avoids the nsIXyz::GetIID()
+// calls which can't be used in templates (runtime vs compile time).
+ASSOCIATE_IID("00000000-0000-0000-c000-000000000046", nsISupports);
+
+#endif // CHROME_FRAME_NS_ASSOCIATE_IID_WIN_H_
diff --git a/chrome_frame/ns_isupports_impl.h b/chrome_frame/ns_isupports_impl.h
new file mode 100644
index 0000000..4873163
--- /dev/null
+++ b/chrome_frame/ns_isupports_impl.h
@@ -0,0 +1,68 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_NS_ISUPPORTS_IMPL_H_
+#define CHROME_FRAME_NS_ISUPPORTS_IMPL_H_
+
+#include "base/basictypes.h"
+#include "base/logging.h"
+#include "base/platform_thread.h"
+#include "chrome_frame/utils.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nscore.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsid.h"
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsISupportsBase.h"
+
+// A simple single-threaded implementation of methods needed to support
+// nsISupports. Must be inherited by classes that also inherit from nsISupports.
+template<class Derived>
+class NsISupportsImplBase {
+ public:
+ NsISupportsImplBase() : nsiimpl_ref_count_(0) {
+ nsiimpl_thread_id_ = PlatformThread::CurrentId();
+ }
+
+ virtual ~NsISupportsImplBase() {
+ DCHECK(nsiimpl_thread_id_ == PlatformThread::CurrentId());
+ }
+
+ NS_IMETHOD QueryInterface(REFNSIID iid, void** ptr) {
+ DCHECK(nsiimpl_thread_id_ == PlatformThread::CurrentId());
+ nsresult res = NS_NOINTERFACE;
+
+ if (memcmp(&iid, &__uuidof(nsISupports), sizeof(nsIID)) == 0) {
+ *ptr = static_cast<nsISupports*>(static_cast<typename Derived*>(this));
+ AddRef();
+ res = NS_OK;
+ }
+
+ return res;
+ }
+
+ NS_IMETHOD_(nsrefcnt) AddRef() {
+ DCHECK(nsiimpl_thread_id_ == PlatformThread::CurrentId());
+ nsiimpl_ref_count_++;
+ return nsiimpl_ref_count_;
+ }
+
+ NS_IMETHOD_(nsrefcnt) Release() {
+ DCHECK(nsiimpl_thread_id_ == PlatformThread::CurrentId());
+ nsiimpl_ref_count_--;
+
+ if (!nsiimpl_ref_count_) {
+ Derived* me = static_cast<Derived*>(this);
+ delete me;
+ return 0;
+ }
+
+ return nsiimpl_ref_count_;
+ }
+
+ protected:
+ nsrefcnt nsiimpl_ref_count_;
+ AddRefModule nsiimpl_module_ref_;
+ // used to DCHECK on expected single-threaded usage
+ uint64 nsiimpl_thread_id_;
+};
+
+#endif // CHROME_FRAME_NS_ISUPPORTS_IMPL_H_
diff --git a/chrome_frame/ole_document_impl.h b/chrome_frame/ole_document_impl.h
new file mode 100644
index 0000000..4c346d8
--- /dev/null
+++ b/chrome_frame/ole_document_impl.h
@@ -0,0 +1,253 @@
+// Copyright (c) 2009 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.
+
+// TODO(slightlyoff): Add any required LICENSE block changes for MSFT code
+// inclusion.
+
+// ole_document_impl.h : IOleDocument implementation
+//
+// This file is a modified version of the OleDocument.h file, which is
+// part of the ActiveDoc MSDN sample. The modifications are largely
+// conversions to Google coding guidelines. Below if the original header
+// from the file.
+
+// This is a part of the Active Template Library.
+// Copyright (c) Microsoft Corporation. All rights reserved.
+//
+// This source code is only intended as a supplement to the
+// Active Template Library Reference and related
+// electronic documentation provided with the library.
+// See these sources for detailed information regarding the
+// Active Template Library product.
+
+#ifndef CHROME_FRAME_OLE_DOCUMENT_IMPL_H_
+#define CHROME_FRAME_OLE_DOCUMENT_IMPL_H_
+
+// TODO(sanjeevr): Revisit this impl file and cleanup dependencies
+#include <atlbase.h>
+#include <docobj.h>
+
+#include "base/logging.h"
+
+//////////////////////////////////////////////////////////////////////////////
+// IOleDocumentImpl
+template <class T>
+class ATL_NO_VTABLE IOleDocumentImpl : public IOleDocument {
+ public:
+ STDMETHOD(CreateView)(IOleInPlaceSite* in_place_site,
+ IStream* stream,
+ DWORD reserved ,
+ IOleDocumentView** new_view) {
+ DLOG(INFO) << __FUNCTION__;
+ if (new_view == NULL)
+ return E_POINTER;
+ T* t = static_cast<T*>(this);
+ // If we've already created a view then we can't create another as we
+ // currently only support the ability to create one view
+ if (t->m_spInPlaceSite) {
+ return E_FAIL;
+ }
+ IOleDocumentView* view;
+ t->GetUnknown()->QueryInterface(IID_IOleDocumentView,
+ reinterpret_cast<void**>(&view));
+ // If we support IOleDocument we should support IOleDocumentView
+ ATLENSURE(view != NULL);
+ // If they've given us a site then use it
+ if (in_place_site != NULL) {
+ view->SetInPlaceSite(in_place_site);
+ }
+ // If they have given us an IStream pointer then use it to
+ // initialize the view
+ if (stream != NULL) {
+ view->ApplyViewState(stream);
+ }
+ // Return the view
+ *new_view = view;
+ return S_OK;
+ }
+
+ STDMETHOD(GetDocMiscStatus)(DWORD* status) {
+ DLOG(INFO) << __FUNCTION__;
+ if (NULL == status) {
+ return E_POINTER;
+ }
+ *status = DOCMISC_NOFILESUPPORT;
+ return S_OK;
+ }
+
+ STDMETHOD(EnumViews)(IEnumOleDocumentViews** enum_views,
+ IOleDocumentView** view) {
+ DLOG(INFO) << __FUNCTION__;
+ if (view == NULL)
+ return E_POINTER;
+ T* t = static_cast<T*>(this);
+ // We only support one view
+ return t->_InternalQueryInterface(IID_IOleDocumentView,
+ reinterpret_cast<void**>(view));
+ }
+};
+
+//////////////////////////////////////////////////////////////////////////////
+// IOleDocumentViewImpl
+
+template <class T>
+class ATL_NO_VTABLE IOleDocumentViewImpl : public IOleDocumentView {
+ public:
+ STDMETHOD(SetInPlaceSite)(IOleInPlaceSite* in_place_site) {
+ DLOG(INFO) << __FUNCTION__;
+ T* t = static_cast<T*>(this);
+ if (t->m_spInPlaceSite) {
+ // If we already have a site get rid of it
+ UIActivate(FALSE);
+ HRESULT hr = t->InPlaceDeactivate();
+ if (FAILED(hr)) {
+ return hr;
+ }
+ DCHECK(!t->m_bInPlaceActive);
+ }
+ if (in_place_site != NULL) {
+ t->m_spInPlaceSite.Release();
+ in_place_site->QueryInterface(
+ IID_IOleInPlaceSiteWindowless,
+ reinterpret_cast<void **>(&t->m_spInPlaceSite));
+ if (!t->m_spInPlaceSite) {
+ // TODO(sanjeevr): This is a super-hack because m_spInPlaceSite
+ // is an IOleInPlaceSiteWindowless pointer and we are setting
+ // an IOleInPlaceSite pointer into it. The problem is that ATL
+ // (CComControlBase) uses this in a schizophrenic manner based
+ // on the m_bWndLess flag. Ouch, ouch, ouch! Find a way to clean
+ // this up.
+ // Disclaimer: I did not invent this hack, it exists in the MSDN
+ // sample from where this code has been derived and it also exists
+ // in ATL itself (look at atlctl.h line 938).
+ t->m_spInPlaceSite =
+ reinterpret_cast<IOleInPlaceSiteWindowless*>(in_place_site);
+ }
+ }
+ return S_OK;
+ }
+
+ STDMETHOD(GetInPlaceSite)(IOleInPlaceSite** in_place_site) {
+ DLOG(INFO) << __FUNCTION__;
+ if (in_place_site == NULL) {
+ return E_POINTER;
+ }
+ T* t = static_cast<T*>(this);
+ return t->m_spInPlaceSite->QueryInterface(
+ IID_IOleInPlaceSite,
+ reinterpret_cast<LPVOID *>(in_place_site));
+ }
+
+ STDMETHOD(GetDocument)(IUnknown** document) {
+ DLOG(INFO) << __FUNCTION__;
+ if (document == NULL) {
+ return E_POINTER;
+ }
+ T* t = static_cast<T*>(this);
+ *document = t->GetUnknown();
+ (*document)->AddRef();
+ return S_OK;
+ }
+
+ STDMETHOD(SetRect)(LPRECT view_rect) {
+ static bool is_resizing = false;
+ if (is_resizing) {
+ return S_OK;
+ }
+ is_resizing = true;
+ DLOG(INFO) << __FUNCTION__ << " " << view_rect->left << "," <<
+ view_rect->top << "," << view_rect->right << "," << view_rect->bottom;
+ T* t = static_cast<T*>(this);
+ t->SetObjectRects(view_rect, view_rect);
+ is_resizing = false;
+ return S_OK;
+ }
+
+ STDMETHOD(GetRect)(LPRECT view_rect) {
+ DLOG(INFO) << __FUNCTION__;
+ if (view_rect == NULL) {
+ return E_POINTER;
+ }
+ T* t = static_cast<T*>(this);
+ *view_rect = t->m_rcPos;
+ return S_OK;
+ }
+
+ STDMETHOD(SetRectComplex)(LPRECT view_rect,
+ LPRECT hscroll_rect,
+ LPRECT vscroll_rect,
+ LPRECT size_box_rect) {
+ DLOG(INFO) << __FUNCTION__ << " not implemented";
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Show)(BOOL show) {
+ DLOG(INFO) << __FUNCTION__;
+ T* t = static_cast<T*>(this);
+ HRESULT hr = S_OK;
+ if (show) {
+ if (!t->m_bUIActive)
+ hr = t->ActiveXDocActivate(OLEIVERB_INPLACEACTIVATE);
+ } else {
+ hr = t->UIActivate(FALSE);
+ ::ShowWindow(t->m_hWnd, SW_HIDE);
+ }
+ return hr;
+ }
+
+ STDMETHOD(UIActivate)(BOOL ui_activate) {
+ DLOG(INFO) << __FUNCTION__;
+ T* t = static_cast<T*>(this);
+ HRESULT hr = S_OK;
+ if (ui_activate) {
+ // We must know the client site first
+ if (t->m_spInPlaceSite == NULL) {
+ return E_UNEXPECTED;
+ }
+ if (!t->m_bUIActive) {
+ hr = t->ActiveXDocActivate(OLEIVERB_UIACTIVATE);
+ }
+ } else {
+ t->InPlaceMenuDestroy();
+ // t->DestroyToolbar();
+ hr = t->UIDeactivate();
+ }
+ return hr;
+ }
+
+ STDMETHOD(Open)() {
+ DLOG(INFO) << __FUNCTION__ << " not implemented";
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(CloseView)(DWORD reserved) {
+ DLOG(INFO) << __FUNCTION__;
+ T* t = static_cast<T*>(this);
+ t->Show(FALSE);
+ t->SetInPlaceSite(NULL);
+ return S_OK;
+ }
+
+ STDMETHOD(SaveViewState)(LPSTREAM stream) {
+ DLOG(INFO) << __FUNCTION__ << " not implemented";
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(ApplyViewState)(LPSTREAM stream) {
+ DLOG(INFO) << __FUNCTION__ << " not implemented";
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Clone)(IOleInPlaceSite* new_in_place_site,
+ IOleDocumentView** new_view) {
+ DLOG(INFO) << __FUNCTION__ << " not implemented";
+ return E_NOTIMPL;
+ }
+
+ HRESULT ActiveXDocActivate(LONG verb) {
+ return E_NOTIMPL;
+ }
+};
+
+#endif // CHROME_FRAME_OLE_DOCUMENT_IMPL_H_
diff --git a/chrome_frame/plugin_url_request.cc b/chrome_frame/plugin_url_request.cc
new file mode 100644
index 0000000..1d34b0c
--- /dev/null
+++ b/chrome_frame/plugin_url_request.cc
@@ -0,0 +1,65 @@
+// Copyright (c) 2009 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 "chrome_frame/plugin_url_request.h"
+
+#include "chrome/test/automation/automation_messages.h"
+#include "chrome_frame/np_browser_functions.h"
+
+PluginUrlRequest::PluginUrlRequest()
+ : request_handler_(NULL), tab_(0), remote_request_id_(0),
+ status_(URLRequestStatus::IO_PENDING),
+ frame_busting_enabled_(false) {
+}
+
+PluginUrlRequest::~PluginUrlRequest() {
+}
+
+bool PluginUrlRequest::Initialize(PluginRequestHandler* request_handler,
+ int tab, int remote_request_id, const std::string& url,
+ const std::string& method, const std::string& referrer,
+ const std::string& extra_headers, net::UploadData* upload_data,
+ bool enable_frame_busting) {
+ request_handler_ = request_handler;
+ tab_ = tab;
+ remote_request_id_ = remote_request_id;
+ url_ = url;
+ method_ = method;
+ referrer_ = referrer;
+ extra_headers_ = extra_headers;
+ upload_data_ = upload_data;
+ frame_busting_enabled_ = enable_frame_busting;
+ return true;
+}
+
+void PluginUrlRequest::OnResponseStarted(const char* mime_type,
+ const char* headers, int size, base::Time last_modified,
+ const std::string& persistent_cookies,
+ const std::string& redirect_url, int redirect_status) {
+ const IPC::AutomationURLResponse response = {
+ mime_type,
+ headers ? headers : "",
+ size,
+ last_modified,
+ persistent_cookies,
+ redirect_url,
+ redirect_status
+ };
+ request_handler_->Send(new AutomationMsg_RequestStarted(0, tab_,
+ remote_request_id_, response));
+}
+
+void PluginUrlRequest::OnResponseEnd(const URLRequestStatus& status) {
+ DCHECK(!status.is_io_pending());
+ DCHECK(status.is_success() || status.os_error());
+ request_handler_->Send(new AutomationMsg_RequestEnd(0, tab_,
+ remote_request_id_, status));
+}
+
+void PluginUrlRequest::OnReadComplete(const void* buffer, int len) {
+ std::string data(reinterpret_cast<const char*>(buffer), len);
+ request_handler_->Send(new AutomationMsg_RequestData(0, tab_,
+ remote_request_id_, data));
+}
+
diff --git a/chrome_frame/plugin_url_request.h b/chrome_frame/plugin_url_request.h
new file mode 100644
index 0000000..fb28c77
--- /dev/null
+++ b/chrome_frame/plugin_url_request.h
@@ -0,0 +1,114 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_PLUGIN_URL_REQUEST_H_
+#define CHROME_FRAME_PLUGIN_URL_REQUEST_H_
+
+#include <string>
+#include <vector>
+
+#include "base/time.h"
+#include "ipc/ipc_message.h"
+#include "net/base/upload_data.h"
+#include "net/url_request/url_request_status.h"
+#include "base/ref_counted.h"
+
+class PluginUrlRequest;
+
+// Interface for a class that keeps a collection of outstanding
+// reqeusts and offers an outgoing channel.
+class PluginRequestHandler : public IPC::Message::Sender {
+ public:
+ virtual bool AddRequest(PluginUrlRequest* request) = 0;
+ virtual void RemoveRequest(PluginUrlRequest* request) = 0;
+};
+
+// A reference counting solution that's compatible with COM IUnknown
+class UrlRequestReference {
+ public:
+ virtual unsigned long API_CALL AddRef() = 0;
+ virtual unsigned long API_CALL Release() = 0;
+};
+
+class PluginUrlRequest : public UrlRequestReference {
+ public:
+ PluginUrlRequest();
+ ~PluginUrlRequest();
+
+ bool Initialize(PluginRequestHandler* handler, int tab,
+ int remote_request_id, const std::string& url,
+ const std::string& method, const std::string& referrer,
+ const std::string& extra_headers,
+ net::UploadData* upload_data,
+ bool intercept_frame_options);
+
+ // Called in response to automation IPCs
+ virtual bool Start() = 0;
+ virtual void Stop() = 0;
+ virtual bool Read(int bytes_to_read) = 0;
+
+ // Persistent cookies are read from the host browser and passed off to Chrome
+ // These cookies are sent when we receive a response for every URL request
+ // initiated by Chrome. Ideally we should only send cookies for the top level
+ // URL and any subframes. However we don't receive information from Chrome
+ // about the context for a URL, i.e. whether it is a subframe, etc.
+ // Additionally cookies for a URL should be sent once for the page. This
+ // is not done now as it is difficult to track URLs, specifically if they
+ // are redirected, etc.
+ void OnResponseStarted(const char* mime_type, const char* headers, int size,
+ base::Time last_modified, const std::string& peristent_cookies,
+ const std::string& redirect_url, int redirect_status);
+
+ void OnReadComplete(const void* buffer, int len);
+ void OnResponseEnd(const URLRequestStatus& status);
+
+ PluginRequestHandler* request_handler() const {
+ return request_handler_;
+ }
+ int id() const {
+ return remote_request_id_;
+ }
+ const std::string& url() const {
+ return url_;
+ }
+ const std::string& method() const {
+ return method_;
+ }
+ const std::string& referrer() const {
+ return referrer_;
+ }
+ const std::string& extra_headers() const {
+ return extra_headers_;
+ }
+ scoped_refptr<net::UploadData> upload_data() {
+ return upload_data_;
+ }
+
+ bool is_done() const {
+ return (URLRequestStatus::IO_PENDING != status_);
+ }
+
+ void set_url(const std::string& url) {
+ url_ = url;
+ }
+
+ protected:
+ void SendData();
+ bool frame_busting_enabled_;
+
+ private:
+ PluginRequestHandler* request_handler_;
+ int tab_;
+ int remote_request_id_;
+ std::string url_;
+ std::string method_;
+ std::string referrer_;
+ std::string extra_headers_;
+ scoped_refptr<net::UploadData> upload_data_;
+ URLRequestStatus::Status status_;
+};
+
+
+#endif // CHROME_FRAME_PLUGIN_URL_REQUEST_H_
+
diff --git a/chrome_frame/precompiled.cc b/chrome_frame/precompiled.cc
new file mode 100644
index 0000000..6b08ac8
--- /dev/null
+++ b/chrome_frame/precompiled.cc
@@ -0,0 +1,9 @@
+// Copyright (c) 2009 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.
+
+// precompiled.cc : source file that includes just the standard includes
+// precompiled.pch will be the pre-compiled header
+// precompiled.obj will contain the pre-compiled type information
+
+#include "precompiled.h"
diff --git a/chrome_frame/precompiled.h b/chrome_frame/precompiled.h
new file mode 100644
index 0000000..0a13651
--- /dev/null
+++ b/chrome_frame/precompiled.h
@@ -0,0 +1,18 @@
+// Copyright (c) 2009 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.
+
+// precompiled.h : include file for standard system include files,
+// or project specific include files that are used frequently,
+// but are changed infrequently
+
+#ifndef CHROME_FRAME_PRECOMPILED_H_
+#define CHROME_FRAME_PRECOMPILED_H_
+
+#include "resource.h"
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <atlctl.h>
+
+#endif // CHROME_FRAME_PRECOMPILED_H_
diff --git a/chrome_frame/protocol_sink_wrap.cc b/chrome_frame/protocol_sink_wrap.cc
new file mode 100644
index 0000000..a567e9f
--- /dev/null
+++ b/chrome_frame/protocol_sink_wrap.cc
@@ -0,0 +1,615 @@
+// Copyright (c) 2009 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 <htiframe.h>
+
+#include "chrome_frame/protocol_sink_wrap.h"
+
+#include "base/scoped_bstr_win.h"
+#include "base/logging.h"
+#include "base/registry.h"
+#include "base/scoped_bstr_win.h"
+#include "base/singleton.h"
+#include "base/string_util.h"
+
+#include "chrome_frame/utils.h"
+#include "chrome_frame/vtable_patch_manager.h"
+
+// BINDSTATUS_SERVER_MIMETYPEAVAILABLE == 54. Introduced in IE 8, so
+// not in everyone's headers yet. See:
+// http://msdn.microsoft.com/en-us/library/ms775133(VS.85,loband).aspx
+#ifndef BINDSTATUS_SERVER_MIMETYPEAVAILABLE
+#define BINDSTATUS_SERVER_MIMETYPEAVAILABLE 54
+#endif
+
+static const wchar_t* kChromeMimeType = L"application/chromepage";
+static const char kTextHtmlMimeType[] = "text/html";
+
+static const int kInternetProtocolStartIndex = 3;
+static const int kInternetProtocolReadIndex = 9;
+static const int kInternetProtocolStartExIndex = 13;
+
+BEGIN_VTABLE_PATCHES(IInternetProtocol)
+ VTABLE_PATCH_ENTRY(kInternetProtocolStartIndex, ProtocolSinkWrap::OnStart)
+ VTABLE_PATCH_ENTRY(kInternetProtocolReadIndex, ProtocolSinkWrap::OnRead)
+END_VTABLE_PATCHES()
+
+BEGIN_VTABLE_PATCHES(IInternetProtocolEx)
+ VTABLE_PATCH_ENTRY(kInternetProtocolStartIndex, ProtocolSinkWrap::OnStart)
+ VTABLE_PATCH_ENTRY(kInternetProtocolReadIndex, ProtocolSinkWrap::OnRead)
+ VTABLE_PATCH_ENTRY(kInternetProtocolStartExIndex, ProtocolSinkWrap::OnStartEx)
+END_VTABLE_PATCHES()
+
+//
+// ProtocolSinkWrap implementation
+//
+
+// Static map initialization
+ProtocolSinkWrap::ProtocolSinkMap ProtocolSinkWrap::sink_map_;
+CComAutoCriticalSection ProtocolSinkWrap::sink_map_lock_;
+
+ProtocolSinkWrap::ProtocolSinkWrap()
+ : protocol_(NULL), renderer_type_(UNDETERMINED),
+ buffer_size_(0), buffer_pos_(0), is_saved_result_(false),
+ result_code_(0), result_error_(0), report_data_recursiveness_(0) {
+ memset(buffer_, 0, arraysize(buffer_));
+}
+
+ProtocolSinkWrap::~ProtocolSinkWrap() {
+ CComCritSecLock<CComAutoCriticalSection> lock(sink_map_lock_);
+ DCHECK(sink_map_.end() != sink_map_.find(protocol_));
+ sink_map_.erase(protocol_);
+ protocol_ = NULL;
+ DLOG(INFO) << "ProtocolSinkWrap: active sinks: " << sink_map_.size();
+}
+
+bool ProtocolSinkWrap::PatchProtocolHandler(const wchar_t* dll,
+ const CLSID& handler_clsid) {
+ HMODULE module = ::GetModuleHandle(dll);
+ if (!module) {
+ NOTREACHED() << "urlmon is not yet loaded. Error: " << GetLastError();
+ return false;
+ }
+
+ typedef HRESULT (WINAPI* DllGetClassObject_Fn)(REFCLSID, REFIID, LPVOID*);
+ DllGetClassObject_Fn fn = reinterpret_cast<DllGetClassObject_Fn>(
+ ::GetProcAddress(module, "DllGetClassObject"));
+ if (!fn) {
+ NOTREACHED() << "DllGetClassObject not found in urlmon.dll";
+ return false;
+ }
+
+ ScopedComPtr<IClassFactory> protocol_class_factory;
+ HRESULT hr = fn(handler_clsid, IID_IClassFactory,
+ reinterpret_cast<LPVOID*>(protocol_class_factory.Receive()));
+ if (FAILED(hr)) {
+ NOTREACHED() << "DllGetclassObject failed. Error: " << hr;
+ return false;
+ }
+
+ ScopedComPtr<IInternetProtocol> handler_instance;
+ hr = protocol_class_factory->CreateInstance(NULL, IID_IInternetProtocol,
+ reinterpret_cast<void**>(handler_instance.Receive()));
+ if (FAILED(hr)) {
+ NOTREACHED() << "ClassFactory::CreateInstance failed for InternetProtocol."
+ << " Error: " << hr;
+ return false;
+ }
+
+ ScopedComPtr<IInternetProtocolEx> ipex;
+ ipex.QueryFrom(handler_instance);
+ if (ipex) {
+ vtable_patch::PatchInterfaceMethods(ipex, IInternetProtocolEx_PatchInfo);
+ } else {
+ vtable_patch::PatchInterfaceMethods(handler_instance,
+ IInternetProtocol_PatchInfo);
+ }
+
+ return true;
+}
+
+// IInternetProtocol/Ex method implementation.
+HRESULT ProtocolSinkWrap::OnStart(InternetProtocol_Start_Fn orig_start,
+ IInternetProtocol* protocol, LPCWSTR url, IInternetProtocolSink* prot_sink,
+ IInternetBindInfo* bind_info, DWORD flags, HANDLE_PTR reserved) {
+ DCHECK(orig_start);
+ DLOG_IF(INFO, url != NULL) << "OnStart: " << url;
+
+ ScopedComPtr<IInternetProtocolSink> sink_to_use(MaybeWrapSink(protocol,
+ prot_sink, url));
+ return orig_start(protocol, url, sink_to_use, bind_info, flags, reserved);
+}
+
+HRESULT ProtocolSinkWrap::OnStartEx(InternetProtocol_StartEx_Fn orig_start_ex,
+ IInternetProtocolEx* protocol, IUri* uri, IInternetProtocolSink* prot_sink,
+ IInternetBindInfo* bind_info, DWORD flags, HANDLE_PTR reserved) {
+ DCHECK(orig_start_ex);
+
+ ScopedBstr url;
+ uri->GetPropertyBSTR(Uri_PROPERTY_ABSOLUTE_URI, url.Receive(), 0);
+ DLOG_IF(INFO, url != NULL) << "OnStartEx: " << url;
+
+ ScopedComPtr<IInternetProtocolSink> sink_to_use(MaybeWrapSink(protocol,
+ prot_sink, url));
+ return orig_start_ex(protocol, uri, sink_to_use, bind_info, flags, reserved);
+}
+
+HRESULT ProtocolSinkWrap::OnRead(InternetProtocol_Read_Fn orig_read,
+ IInternetProtocol* protocol, void* buffer, ULONG size, ULONG* size_read) {
+ DCHECK(orig_read);
+
+ scoped_refptr<ProtocolSinkWrap> instance =
+ ProtocolSinkWrap::InstanceFromProtocol(protocol);
+ HRESULT hr;
+ if (instance) {
+ DCHECK(instance->protocol_ == protocol);
+ hr = instance->OnReadImpl(buffer, size, size_read, orig_read);
+ } else {
+ hr = orig_read(protocol, buffer, size, size_read);
+ }
+
+ return hr;
+}
+
+bool ProtocolSinkWrap::Initialize(IInternetProtocol* protocol,
+ IInternetProtocolSink* original_sink, const wchar_t* url) {
+ DCHECK(original_sink);
+ delegate_ = original_sink;
+ protocol_ = protocol;
+ if (url)
+ url_ = url;
+
+ CComCritSecLock<CComAutoCriticalSection> lock(sink_map_lock_);
+ DCHECK(sink_map_.end() == sink_map_.find(protocol));
+ sink_map_[protocol] = this;
+ DLOG(INFO) << "ProtocolSinkWrap: active sinks: " << sink_map_.size();
+ return true;
+}
+
+HRESULT WINAPI ProtocolSinkWrap::CheckOutgoingInterface(void* obj,
+ REFIID iid, LPVOID* ret, DWORD cookie) {
+ ProtocolSinkWrap* instance = reinterpret_cast<ProtocolSinkWrap*>(obj);
+ HRESULT hr = E_NOINTERFACE;
+ if (instance && instance->delegate_)
+ hr = instance->delegate_->QueryInterface(iid, ret);
+
+#ifndef NDEBUG
+ if (SUCCEEDED(hr)) {
+ wchar_t iid_string[64] = {0};
+ StringFromGUID2(iid, iid_string, arraysize(iid_string));
+ DLOG(INFO) << "Giving out wrapped interface: " << iid_string;
+ }
+#endif
+
+ return hr;
+}
+
+HRESULT WINAPI ProtocolSinkWrap::IfDelegateSupports(void* obj,
+ REFIID iid, LPVOID* ret, DWORD cookie) {
+ HRESULT hr = E_NOINTERFACE;
+ ProtocolSinkWrap* instance = reinterpret_cast<ProtocolSinkWrap*>(obj);
+ if (instance && instance->delegate_) {
+ ScopedComPtr<IUnknown> original;
+ hr = instance->delegate_->QueryInterface(iid,
+ reinterpret_cast<void**>(original.Receive()));
+ if (original) {
+ IUnknown* supported_interface = reinterpret_cast<IUnknown*>(
+ reinterpret_cast<DWORD_PTR>(obj) + cookie);
+ supported_interface->AddRef();
+ *ret = supported_interface;
+ hr = S_OK;
+ }
+ }
+
+ return hr;
+}
+
+// IInternetProtocolSink methods
+STDMETHODIMP ProtocolSinkWrap::Switch(PROTOCOLDATA* protocol_data) {
+ HRESULT hr = E_FAIL;
+ if (delegate_)
+ hr = delegate_->Switch(protocol_data);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::ReportProgress(ULONG status_code,
+ LPCWSTR status_text) {
+ DLOG(INFO) << "ProtocolSinkWrap::ReportProgress: Code:" << status_code <<
+ " Text: " << (status_text ? status_text : L"");
+ if ((BINDSTATUS_MIMETYPEAVAILABLE == status_code) ||
+ (BINDSTATUS_VERIFIEDMIMETYPEAVAILABLE == status_code)) {
+ // If we have a MIMETYPE and that MIMETYPE is not "text/html". we don't
+ // want to do anything with this.
+ if (status_text) {
+ size_t status_text_length = lstrlenW(status_text);
+ const wchar_t* status_text_end = status_text + std::min(
+ status_text_length, arraysize(kTextHtmlMimeType) - 1);
+ if (!LowerCaseEqualsASCII(status_text, status_text_end,
+ kTextHtmlMimeType)) {
+ renderer_type_ = OTHER;
+ }
+ }
+ }
+
+ HRESULT hr = E_FAIL;
+ if (delegate_)
+ hr = delegate_->ReportProgress(status_code, status_text);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::ReportData(DWORD flags, ULONG progress,
+ ULONG max_progress) {
+ DCHECK(protocol_);
+ DCHECK(delegate_);
+ DLOG(INFO) << "ProtocolSinkWrap::ReportData: flags: " << flags <<
+ " progress: " << progress << " progress_max: " << max_progress;
+
+ scoped_refptr<ProtocolSinkWrap> self_ref(this);
+
+ // Maintain a stack depth to make a determination. ReportData is called
+ // recursively in IE8. If the request can be served in a single Read, the
+ // situation ends up like this:
+ // orig_prot
+ // |--> ProtocolSinkWrap::ReportData (BSCF_FIRSTDATANOTIFICATION)
+ // |--> orig_prot->Read(...) - 1st read - S_OK and data
+ // |--> ProtocolSinkWrap::ReportData (BSCF_LASTDATANOTIFICATION)
+ // |--> orig_prot->Read(...) - 2nd read S_FALSE, 0 bytes
+ //
+ // Inner call returns S_FALSE and no data. We try to make a determination
+ // of render type then and incorrectly set it to 'OTHER' as we don't have
+ // any data yet. However, we can make a determination in the context of
+ // outer ReportData since the first read will return S_OK with data. Then
+ // the next Read in the loop will return S_FALSE and we will enter the
+ // determination logic.
+
+ // NOTE: We use the report_data_recursiveness_ variable to detect situations
+ // in which calls to ReportData are re-entrant (such as when the entire
+ // contents of a page fit inside a single packet). In these cases, we
+ // don't care about re-entrant calls beyond the second, and so we compare
+ // report_data_recursiveness_ inside the while loop, making sure we skip
+ // what would otherwise be spurious calls to ReportProgress().
+ report_data_recursiveness_++;
+
+ HRESULT hr = S_OK;
+ if (is_undetermined()) {
+ HRESULT hr_read = S_OK;
+ while (hr_read == S_OK) {
+ ULONG size_read = 0;
+ hr_read = protocol_->Read(buffer_ + buffer_size_,
+ kMaxContentSniffLength - buffer_size_, &size_read);
+ buffer_size_ += size_read;
+
+ // Attempt to determine the renderer type if we have received
+ // sufficient data. Do not attempt this when we are called recursively.
+ if (report_data_recursiveness_ < 2 && (S_FALSE == hr_read) ||
+ (buffer_size_ >= kMaxContentSniffLength)) {
+ DetermineRendererType();
+ if (renderer_type() == CHROME) {
+ // Workaround for IE 8 and "nosniff". See:
+ // http://blogs.msdn.com/ie/archive/2008/09/02/ie8-security-part-vi-beta-2-update.aspx
+ delegate_->ReportProgress(
+ BINDSTATUS_SERVER_MIMETYPEAVAILABLE, kChromeMimeType);
+ // For IE < 8.
+ delegate_->ReportProgress(
+ BINDSTATUS_VERIFIEDMIMETYPEAVAILABLE, kChromeMimeType);
+
+ delegate_->ReportData(
+ BSCF_LASTDATANOTIFICATION | BSCF_DATAFULLYAVAILABLE, 0, 0);
+ }
+ break;
+ }
+ }
+ }
+
+ // we call original only if the renderer type is other
+ if (renderer_type() == OTHER) {
+ hr = delegate_->ReportData(flags, progress, max_progress);
+
+ if (is_saved_result_) {
+ is_saved_result_ = false;
+ delegate_->ReportResult(result_code_, result_error_,
+ result_text_.c_str());
+ }
+ }
+
+ report_data_recursiveness_--;
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::ReportResult(HRESULT result, DWORD error,
+ LPCWSTR result_text) {
+ DLOG(INFO) << "ProtocolSinkWrap::ReportResult: result: " << result <<
+ " error: " << error << " Text: " << (result_text ? result_text : L"");
+
+ // If this request failed, we don't want to have anything to do with this.
+ if (FAILED(result))
+ renderer_type_ = OTHER;
+
+ // if we are still not sure about the renderer type, cache the result,
+ // othewise urlmon will get confused about getting reported about a
+ // success result for which it never received any data.
+ if (is_undetermined()) {
+ is_saved_result_ = true;
+ result_code_ = result;
+ result_error_ = error;
+ if (result_text)
+ result_text_ = result_text;
+ return S_OK;
+ }
+
+ HRESULT hr = E_FAIL;
+ if (delegate_)
+ hr = delegate_->ReportResult(result, error, result_text);
+
+ return hr;
+}
+
+// IInternetBindInfoEx
+STDMETHODIMP ProtocolSinkWrap::GetBindInfo(
+ DWORD* flags, BINDINFO* bind_info_ret) {
+ ScopedComPtr<IInternetBindInfo> bind_info;
+ HRESULT hr = bind_info.QueryFrom(delegate_);
+ if (bind_info)
+ hr = bind_info->GetBindInfo(flags, bind_info_ret);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::GetBindString(ULONG string_type,
+ LPOLESTR* string_array, ULONG array_size, ULONG* size_returned) {
+ ScopedComPtr<IInternetBindInfo> bind_info;
+ HRESULT hr = bind_info.QueryFrom(delegate_);
+ if (bind_info)
+ hr = bind_info->GetBindString(string_type, string_array,
+ array_size, size_returned);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::GetBindInfoEx(DWORD *flags, BINDINFO* bind_info,
+ DWORD* bindf2, DWORD *reserved) {
+ ScopedComPtr<IInternetBindInfoEx> bind_info_ex;
+ HRESULT hr = bind_info_ex.QueryFrom(delegate_);
+ if (bind_info_ex)
+ hr = bind_info_ex->GetBindInfoEx(flags, bind_info, bindf2, reserved);
+ return hr;
+}
+
+// IServiceProvider
+STDMETHODIMP ProtocolSinkWrap::QueryService(REFGUID service_guid,
+ REFIID riid, void** service) {
+ ScopedComPtr<IServiceProvider> service_provider;
+ HRESULT hr = service_provider.QueryFrom(delegate_);
+ if (service_provider)
+ hr = service_provider->QueryService(service_guid, riid, service);
+ return hr;
+}
+
+// IAuthenticate
+STDMETHODIMP ProtocolSinkWrap::Authenticate(HWND* window,
+ LPWSTR* user_name, LPWSTR* password) {
+ ScopedComPtr<IAuthenticate> authenticate;
+ HRESULT hr = authenticate.QueryFrom(delegate_);
+ if (authenticate)
+ hr = authenticate->Authenticate(window, user_name, password);
+ return hr;
+}
+
+// IInternetProtocolEx
+STDMETHODIMP ProtocolSinkWrap::Start(LPCWSTR url,
+ IInternetProtocolSink *protocol_sink, IInternetBindInfo* bind_info,
+ DWORD flags, HANDLE_PTR reserved) {
+ ScopedComPtr<IInternetProtocolRoot> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Start(url, protocol_sink, bind_info, flags, reserved);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::Continue(PROTOCOLDATA* protocol_data) {
+ ScopedComPtr<IInternetProtocolRoot> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Continue(protocol_data);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::Abort(HRESULT reason, DWORD options) {
+ ScopedComPtr<IInternetProtocolRoot> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Abort(reason, options);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::Terminate(DWORD options) {
+ ScopedComPtr<IInternetProtocolRoot> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Terminate(options);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::Suspend() {
+ ScopedComPtr<IInternetProtocolRoot> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Suspend();
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::Resume() {
+ ScopedComPtr<IInternetProtocolRoot> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Resume();
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::Read(void *buffer, ULONG size,
+ ULONG* size_read) {
+ ScopedComPtr<IInternetProtocol> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Read(buffer, size, size_read);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::Seek(LARGE_INTEGER move, DWORD origin,
+ ULARGE_INTEGER* new_pos) {
+ ScopedComPtr<IInternetProtocol> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->Seek(move, origin, new_pos);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::LockRequest(DWORD options) {
+ ScopedComPtr<IInternetProtocol> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->LockRequest(options);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::UnlockRequest() {
+ ScopedComPtr<IInternetProtocol> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->UnlockRequest();
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::StartEx(IUri* uri,
+ IInternetProtocolSink* protocol_sink, IInternetBindInfo* bind_info,
+ DWORD flags, HANDLE_PTR reserved) {
+ ScopedComPtr<IInternetProtocolEx> protocol;
+ HRESULT hr = protocol.QueryFrom(delegate_);
+ if (protocol)
+ hr = protocol->StartEx(uri, protocol_sink, bind_info, flags, reserved);
+ return hr;
+}
+
+// IInternetPriority
+STDMETHODIMP ProtocolSinkWrap::SetPriority(LONG priority) {
+ ScopedComPtr<IInternetPriority> internet_priority;
+ HRESULT hr = internet_priority.QueryFrom(delegate_);
+ if (internet_priority)
+ hr = internet_priority->SetPriority(priority);
+ return hr;
+}
+
+STDMETHODIMP ProtocolSinkWrap::GetPriority(LONG* priority) {
+ ScopedComPtr<IInternetPriority> internet_priority;
+ HRESULT hr = internet_priority.QueryFrom(delegate_);
+ if (internet_priority)
+ hr = internet_priority->GetPriority(priority);
+ return hr;
+}
+
+// IWrappedProtocol
+STDMETHODIMP ProtocolSinkWrap::GetWrapperCode(LONG *code, DWORD_PTR reserved) {
+ ScopedComPtr<IWrappedProtocol> wrapped_protocol;
+ HRESULT hr = wrapped_protocol.QueryFrom(delegate_);
+ if (wrapped_protocol)
+ hr = wrapped_protocol->GetWrapperCode(code, reserved);
+ return hr;
+}
+
+
+// public IUriContainer
+STDMETHODIMP ProtocolSinkWrap::GetIUri(IUri** uri) {
+ ScopedComPtr<IUriContainer> uri_container;
+ HRESULT hr = uri_container.QueryFrom(delegate_);
+ if (uri_container)
+ hr = uri_container->GetIUri(uri);
+ return hr;
+}
+
+// Protected helpers
+
+void ProtocolSinkWrap::DetermineRendererType() {
+ if (is_undetermined()) {
+ if (IsOptInUrl(url_.c_str())) {
+ renderer_type_ = CHROME;
+ } else {
+ std::wstring xua_compat_content;
+ // Note that document_contents_ may have NULL characters in it. While
+ // browsers may handle this properly, we don't and will stop scanning for
+ // the XUACompat content value if we encounter one.
+ DCHECK(buffer_size_ < arraysize(buffer_));
+ buffer_[buffer_size_] = 0;
+ std::wstring html_contents;
+ // TODO(joshia): detect and handle different content encodings
+ UTF8ToWide(buffer_, buffer_size_, &html_contents);
+ UtilGetXUACompatContentValue(html_contents, &xua_compat_content);
+ if (StrStrI(xua_compat_content.c_str(), kChromeContentPrefix)) {
+ renderer_type_ = CHROME;
+ } else {
+ renderer_type_ = OTHER;
+ }
+ }
+ }
+}
+
+HRESULT ProtocolSinkWrap::OnReadImpl(void* buffer, ULONG size, ULONG* size_read,
+ InternetProtocol_Read_Fn orig_read) {
+ // We want to switch the renderer to chrome, we cannot return any
+ // data now.
+ if (CHROME == renderer_type())
+ return S_FALSE;
+
+ // Serve data from our buffer first.
+ if (OTHER == renderer_type()) {
+ const ULONG bytes_to_copy = std::min(buffer_size_ - buffer_pos_, size);
+ if (bytes_to_copy) {
+ memcpy(buffer, buffer_ + buffer_pos_, bytes_to_copy);
+ *size_read = bytes_to_copy;
+ buffer_pos_ += bytes_to_copy;
+ return S_OK;
+ }
+ }
+
+ return orig_read(protocol_, buffer, size, size_read);
+}
+
+scoped_refptr<ProtocolSinkWrap> ProtocolSinkWrap::InstanceFromProtocol(
+ IInternetProtocol* protocol) {
+ CComCritSecLock<CComAutoCriticalSection> lock(sink_map_lock_);
+ scoped_refptr<ProtocolSinkWrap> instance;
+ ProtocolSinkMap::iterator it = sink_map_.find(protocol);
+ if (sink_map_.end() != it)
+ instance = it->second;
+ return instance;
+}
+
+HRESULT ProtocolSinkWrap::WebBrowserFromProtocolSink(
+ IInternetProtocolSink* sink, IWebBrowser2** web_browser) {
+ // TODO(tommi): GUID_NULL doesn't work when loading from history.
+ // asking for IID_IHttpNegotiate as the service id works, but
+ // getting the IWebBrowser2 interface still doesn't work.
+ ScopedComPtr<IHttpNegotiate> http_negotiate;
+ HRESULT hr = DoQueryService(GUID_NULL, sink, http_negotiate.Receive());
+ if (http_negotiate)
+ hr = DoQueryService(IID_ITargetFrame2, http_negotiate, web_browser);
+
+ return hr;
+}
+
+ScopedComPtr<IInternetProtocolSink> ProtocolSinkWrap::MaybeWrapSink(
+ IInternetProtocol* protocol, IInternetProtocolSink* prot_sink,
+ const wchar_t* url) {
+ ScopedComPtr<IInternetProtocolSink> sink_to_use;
+ sink_to_use.QueryFrom(prot_sink);
+ ScopedComPtr<IWebBrowser2> web_browser;
+ WebBrowserFromProtocolSink(prot_sink, web_browser.Receive());
+ if (web_browser) {
+ CComObject<ProtocolSinkWrap>* wrap = NULL;
+ CComObject<ProtocolSinkWrap>::CreateInstance(&wrap);
+ DCHECK(wrap);
+ if (wrap->Initialize(protocol, prot_sink, url)) {
+ sink_to_use = wrap;
+ }
+ }
+
+ return sink_to_use;
+}
diff --git a/chrome_frame/protocol_sink_wrap.h b/chrome_frame/protocol_sink_wrap.h
new file mode 100644
index 0000000..e4f9cfb
--- /dev/null
+++ b/chrome_frame/protocol_sink_wrap.h
@@ -0,0 +1,221 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_PROTOCOL_SINK_WRAP_H_
+#define CHROME_FRAME_PROTOCOL_SINK_WRAP_H_
+
+#include <exdisp.h>
+#include <urlmon.h>
+#include <atlbase.h>
+#include <atlcom.h>
+#include <map>
+#include <string>
+
+#include "base/basictypes.h"
+#include "base/ref_counted.h"
+#include "base/scoped_comptr_win.h"
+#include "googleurl/src/gurl.h"
+#include "chrome_frame/ie8_types.h"
+
+// Typedefs for IInternetProtocol and related methods that we patch.
+typedef HRESULT (STDMETHODCALLTYPE* InternetProtocol_Start_Fn)(
+ IInternetProtocol* this_object, LPCWSTR url,
+ IInternetProtocolSink* prot_sink, IInternetBindInfo* bind_info,
+ DWORD flags, HANDLE_PTR reserved);
+typedef HRESULT (STDMETHODCALLTYPE* InternetProtocol_Read_Fn)(
+ IInternetProtocol* this_object, void* buffer, ULONG size,
+ ULONG* size_read);
+typedef HRESULT (STDMETHODCALLTYPE* InternetProtocol_StartEx_Fn)(
+ IInternetProtocolEx* this_object, IUri* uri,
+ IInternetProtocolSink* prot_sink, IInternetBindInfo* bind_info,
+ DWORD flags, HANDLE_PTR reserved);
+typedef HRESULT (STDMETHODCALLTYPE* InternetProtocolRoot_Continue_Fn)(
+ IInternetProtocolRoot* me, PROTOCOLDATA* data);
+
+// A class to wrap protocol sink in IInternetProtocol::Start[Ex] for
+// HTTP and HTTPS protocols.
+//
+// This is an alternative to a mime filter and we have to do this in order
+// to inspect initial portion of HTML for 'chrome' meta tag and report
+// a different mime type in that case.
+//
+// We implement several documented interfaces
+// supported by the original sink provided by urlmon. There are a few
+// undocumented interfaces that we have chosen not to implement
+// but delegate simply the QI.
+class ProtocolSinkWrap
+ : public CComObjectRootEx<CComMultiThreadModel>,
+ public IInternetProtocolSink,
+ public IInternetBindInfoEx,
+ public IServiceProvider,
+ public IAuthenticate,
+ public IInternetProtocolEx,
+ public IInternetPriority,
+ public IWrappedProtocol,
+ // public IPreBindingSupport, // undocumented
+ // public ITransProtocolSink, // Undocumented
+ // public ITransactionInternal, // undocumented
+ public IUriContainer {
+ public:
+
+#define COM_INTERFACE_ENTRY_IF_DELEGATE_SUPPORTS(x) \
+ COM_INTERFACE_ENTRY_FUNC(_ATL_IIDOF(x), \
+ offsetofclass(x, _ComMapClass), \
+ IfDelegateSupports)
+
+BEGIN_COM_MAP(ProtocolSinkWrap)
+ COM_INTERFACE_ENTRY(IInternetProtocolSink)
+ COM_INTERFACE_ENTRY(IInternetBindInfo)
+ COM_INTERFACE_ENTRY(IInternetBindInfoEx)
+ COM_INTERFACE_ENTRY(IServiceProvider)
+ COM_INTERFACE_ENTRY(IAuthenticate)
+ COM_INTERFACE_ENTRY(IInternetProtocolRoot)
+ COM_INTERFACE_ENTRY(IInternetProtocol)
+ COM_INTERFACE_ENTRY(IInternetProtocolEx)
+ COM_INTERFACE_ENTRY(IInternetPriority)
+ COM_INTERFACE_ENTRY(IWrappedProtocol)
+ COM_INTERFACE_ENTRY_IF_DELEGATE_SUPPORTS(IUriContainer)
+ COM_INTERFACE_ENTRY_FUNC_BLIND(0, CheckOutgoingInterface)
+END_COM_MAP()
+
+ ProtocolSinkWrap();
+ virtual ~ProtocolSinkWrap();
+
+ bool Initialize(IInternetProtocol* protocol,
+ IInternetProtocolSink* original_sink, const wchar_t* url);
+
+ static bool PatchProtocolHandler(const wchar_t* dll,
+ const CLSID& handler_clsid);
+
+ // IInternetProtocol/Ex patches.
+ static HRESULT STDMETHODCALLTYPE OnStart(InternetProtocol_Start_Fn orig_start,
+ IInternetProtocol* protocol, LPCWSTR url,
+ IInternetProtocolSink* prot_sink, IInternetBindInfo* bind_info,
+ DWORD flags, HANDLE_PTR reserved);
+
+ static HRESULT STDMETHODCALLTYPE OnStartEx(
+ InternetProtocol_StartEx_Fn orig_start_ex, IInternetProtocolEx* protocol,
+ IUri* uri, IInternetProtocolSink* prot_sink,
+ IInternetBindInfo* bind_info, DWORD flags, HANDLE_PTR reserved);
+
+ static HRESULT STDMETHODCALLTYPE OnRead(InternetProtocol_Read_Fn orig_read,
+ IInternetProtocol* protocol, void* buffer, ULONG size, ULONG* size_read);
+
+ // IInternetProtocolSink methods
+ STDMETHOD(Switch)(PROTOCOLDATA* protocol_data);
+ STDMETHOD(ReportProgress)(ULONG status_code, LPCWSTR status_text);
+ STDMETHOD(ReportData)(DWORD flags, ULONG progress, ULONG max_progress);
+ STDMETHOD(ReportResult)(HRESULT result, DWORD error, LPCWSTR result_text);
+
+ // IInternetBindInfoEx
+ STDMETHOD(GetBindInfo)(DWORD* flags, BINDINFO* bind_info);
+ STDMETHOD(GetBindString)(ULONG string_type, LPOLESTR* string_array,
+ ULONG array_size, ULONG* size_returned);
+ STDMETHOD(GetBindInfoEx)(DWORD *flags, BINDINFO* bind_info,
+ DWORD* bindf2, DWORD *reserved);
+
+ // IServiceProvider
+ STDMETHOD(QueryService)(REFGUID service_guid, REFIID riid, void** service);
+
+ // IAuthenticate
+ STDMETHOD(Authenticate)(HWND* window, LPWSTR* user_name, LPWSTR* password);
+
+ // IInternetProtocolEx
+ STDMETHOD(Start)(LPCWSTR url, IInternetProtocolSink *protocol_sink,
+ IInternetBindInfo* bind_info, DWORD flags, HANDLE_PTR reserved);
+ STDMETHOD(Continue)(PROTOCOLDATA* protocol_data);
+ STDMETHOD(Abort)(HRESULT reason, DWORD options);
+ STDMETHOD(Terminate)(DWORD options);
+ STDMETHOD(Suspend)();
+ STDMETHOD(Resume)();
+ STDMETHOD(Read)(void *buffer, ULONG size, ULONG* size_read);
+ STDMETHOD(Seek)(LARGE_INTEGER move, DWORD origin, ULARGE_INTEGER* new_pos);
+ STDMETHOD(LockRequest)(DWORD options);
+ STDMETHOD(UnlockRequest)();
+ STDMETHOD(StartEx)(IUri* uri, IInternetProtocolSink* protocol_sink,
+ IInternetBindInfo* bind_info, DWORD flags, HANDLE_PTR reserved);
+
+ // IInternetPriority
+ STDMETHOD(SetPriority)(LONG priority);
+ STDMETHOD(GetPriority)(LONG* priority);
+
+ // IWrappedProtocol
+ STDMETHOD(GetWrapperCode)(LONG *code, DWORD_PTR reserved);
+
+ // public IUriContainer
+ STDMETHOD(GetIUri)(IUri** uri);
+
+ // IPreBindingSupport, // undocumented
+ // ITransProtocolSink, // Undocumented
+ // ITransactionInternal, // undocumented
+
+ protected:
+ enum RendererType {
+ UNDETERMINED,
+ CHROME,
+ OTHER
+ };
+
+ typedef std::map<IInternetProtocol*, ProtocolSinkWrap*> ProtocolSinkMap;
+ static const int kMaxContentSniffLength = 1024;
+
+ static scoped_refptr<ProtocolSinkWrap> InstanceFromProtocol(
+ IInternetProtocol* protocol);
+ static HRESULT WebBrowserFromProtocolSink(IInternetProtocolSink* sink,
+ IWebBrowser2** web_browser);
+ static ScopedComPtr<IInternetProtocolSink> MaybeWrapSink(
+ IInternetProtocol* protocol, IInternetProtocolSink* prot_sink,
+ const wchar_t* url);
+ static HRESULT WINAPI CheckOutgoingInterface(void* obj, REFIID iid,
+ LPVOID* ret, DWORD cookie);
+ static HRESULT WINAPI IfDelegateSupports(void* obj, REFIID iid,
+ LPVOID* ret, DWORD cookie);
+
+ void DetermineRendererType();
+ HRESULT OnReadImpl(void* buffer, ULONG size, ULONG* size_read,
+ InternetProtocol_Read_Fn orig_read);
+
+ bool is_undetermined() const {
+ return (UNDETERMINED == renderer_type_);
+ }
+ RendererType renderer_type() const {
+ return renderer_type_;
+ }
+
+ // WARNING: Don't use GURL variables here. Please see
+ // http://b/issue?id=2102171 for details.
+
+ // Remember original sink
+ CComPtr<IInternetProtocolSink> delegate_;
+ // Cannot take a reference on the protocol.
+ IInternetProtocol* protocol_;
+ RendererType renderer_type_;
+
+ // Buffer for accumulated data including 1 extra for NULL-terminator
+ char buffer_[kMaxContentSniffLength + 1];
+ unsigned long buffer_size_;
+ unsigned long buffer_pos_;
+
+ // Accumulated result
+ bool is_saved_result_;
+ HRESULT result_code_;
+ DWORD result_error_;
+ std::wstring result_text_;
+ // For tracking re-entrency and preventing duplicate Read()s from
+ // distorting the outcome of ReportData.
+ int report_data_recursiveness_;
+
+ static ProtocolSinkMap sink_map_;
+ // TODO(joshia): Replace with Lock
+ static CComAutoCriticalSection sink_map_lock_;
+
+ std::wstring url_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(ProtocolSinkWrap);
+};
+
+
+#endif // CHROME_FRAME_PROTOCOL_SINK_WRAP_H_
+
diff --git a/chrome_frame/rename_me_to_supplement.gypi b/chrome_frame/rename_me_to_supplement.gypi
new file mode 100644
index 0000000..fbe6e55
--- /dev/null
+++ b/chrome_frame/rename_me_to_supplement.gypi
@@ -0,0 +1,24 @@
+# Copyright (c) 2009 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.
+
+# It is used to override the Chrome Frame appid, as well as
+# add extra required source files and defines. The file is checked in as
+# rename_me_to_supplement.gypi so as to not be active on the official builder.
+# This is required because with these fields present, Chrome will build in
+# Chrome Frame mode, which isn't what the Chrome official builder wants yet.
+#
+# Renaming this file to supplement.gypi will cause gyp to pick it up.
+# supplement.gypi is a magic gyp include file that gets pulled in before any
+# other includes.
+#
+# The official builder will define these extra vars and whatnot in the build
+# scripts themselves.
+{
+ 'variables': {
+ 'google_update_appid': '{8BA986DA-5100-405E-AA35-86F34A02ACBF}',
+ 'extra_installer_util_sources': 1,
+ 'branding': 'Chrome',
+ 'experimental_build_define': 1,
+ },
+} \ No newline at end of file
diff --git a/chrome_frame/resource.h b/chrome_frame/resource.h
new file mode 100644
index 0000000..53f51a5
--- /dev/null
+++ b/chrome_frame/resource.h
@@ -0,0 +1,19 @@
+//{{NO_DEPENDENCIES}}
+
+// Copyright (c) 2009 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.
+
+// Microsoft Visual C++ generated include file.
+// Used by resources/tlb_resource.rc
+
+// Default values for new objects
+//
+#ifdef APSTUDIO_INVOKED
+#ifndef APSTUDIO_READONLY_SYMBOLS
+#define _APS_NEXT_RESOURCE_VALUE 204
+#define _APS_NEXT_COMMAND_VALUE 32770
+#define _APS_NEXT_CONTROL_VALUE 205
+#define _APS_NEXT_SYMED_VALUE 108
+#endif
+#endif
diff --git a/chrome_frame/resources/chrome_frame_resources.grd b/chrome_frame/resources/chrome_frame_resources.grd
new file mode 100644
index 0000000..c249ce8
--- /dev/null
+++ b/chrome_frame/resources/chrome_frame_resources.grd
@@ -0,0 +1,55 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+Copyright (c) 2009 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.
+-->
+
+<!--
+Embeded strings, branding resource, etc. See chrome_frame_strings.grd
+for localizable strings
+-->
+
+<grit latest_public_release="0" current_release="1">
+ <outputs>
+ <output filename="grit/chrome_frame_resources.h" type="rc_header">
+ <emit emit_type='prepend'></emit>
+ </output>
+ <output filename="chrome_frame_resources.rc" type="rc_all" lang="en"/>
+ <output filename="chrome_frame_resources.pak" type="data_package" />
+ </outputs>
+ <release seq="1">
+ <messages>
+ <!-- TODO(slightlyoff): should these be in chrome_frame_strings.grd
+ instead? -->
+ <message name="IDS_PROJNAME">
+ ChromeTab
+ </message>
+ <message name="IDS_VERSIONMISMATCH_HEADER">
+ ChromeFrame Update.
+ </message>
+ <message name="IDS_VERSIONMISMATCH">
+ ChromeFrame has been updated. Please restart your browser. Chrome version: <ph name="TODO_0001">%ls<ex>TODO</ex></ph>, Chrome Frame version: <ph name="TODO_0002">%ls<ex>TODO</ex></ph>
+ </message>
+ <message name="IDS_VERSIONUNKNOWN">
+ Very old
+ </message>
+ </messages>
+ <structures first_id="50000">
+ <structure name="IDD_FIND_DIALOG" file="structured_resources.rc" type="dialog" >
+ </structure>
+ </structures>
+ <includes>
+ <include name="IDB_CHROME_ACTIVE_DOCUMENT" file="../chrome_active_document.bmp"
+ type="BITMAP" />
+ <include name="IDR_BHO" file="../bho.rgs" type="REGISTRY" />
+ <include name="IDR_CHROMETAB" file="../chrome_tab.rgs" type="REGISTRY" />
+ <include name="IDR_CHROMEPROTOCOL" file="../chrome_protocol.rgs" type="REGISTRY" />
+ <include name="IDR_CHROMEACTIVEDOCUMENT" file="../chrome_active_document.rgs"
+ type="REGISTRY" />
+ <include name="IDR_CHROMEFRAME" file="../chrome_frame_activex.rgs" type="REGISTRY" />
+ <include name="IDR_CHROMEFRAME_NPAPI" file="../chrome_frame_npapi.rgs" type="REGISTRY" />
+ </includes>
+ </release>
+</grit>
diff --git a/chrome_frame/resources/chrome_frame_strings.grd b/chrome_frame/resources/chrome_frame_strings.grd
new file mode 100644
index 0000000..39bcfd9
--- /dev/null
+++ b/chrome_frame/resources/chrome_frame_strings.grd
@@ -0,0 +1,40 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+Copyright (c) 2009 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.
+-->
+
+<!-- Definitions of resources that will be translated for each locale.
+-->
+
+<grit base_dir="." latest_public_release="0" current_release="1"
+ source_lang_id="en" enc_check="möl">
+ <outputs>
+ <output filename="grit/chrome_frame_strings.h" type="rc_header">
+ <emit emit_type='prepend'></emit>
+ </output>
+ <output filename="chrome_frame_strings.rc" type="rc_all" lang="en" />
+ </outputs>
+ <translations>
+ <!--
+ <file path="chrome_frame_strings_en-GB.xtb" lang="en-GB" />
+ ...
+ -->
+ </translations>
+ <release seq="1" allow_pseudo="false">
+ <messages fallback_to_english="true" first_id="30000">
+
+ <!-- Menus -->
+ <message name="IDS_CHROME_FRAME_MENU_ABOUT" desc="About Chrome Frame label">
+ About Chrome Frame...
+ </message>
+
+ <!-- General application strings -->
+ <message name="IDS_CHROME_FRAME_NAME" desc="Official plugin name.">
+ Google Chrome Frame
+ </message>
+ </messages>
+ </release>
+</grit>
diff --git a/chrome_frame/resources/structured_resources.rc b/chrome_frame/resources/structured_resources.rc
new file mode 100644
index 0000000..22e0337
--- /dev/null
+++ b/chrome_frame/resources/structured_resources.rc
@@ -0,0 +1,24 @@
+// Copyright (c) 2009 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.
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Dialog
+//
+
+IDD_FIND_DIALOG DIALOGEX 0, 0, 278, 60
+STYLE DS_SETFONT | DS_MODALFRAME | DS_FIXEDSYS | WS_POPUP | WS_VISIBLE | WS_CAPTION | WS_SYSMENU
+EXSTYLE WS_EX_CONTROLPARENT
+CAPTION "Find"
+FONT 8, "MS Shell Dlg", 400, 0, 0x1
+BEGIN
+ EDITTEXT IDC_FIND_TEXT,51,7,154,14,ES_AUTOHSCROLL
+ DEFPUSHBUTTON "&Find Next",IDOK,221,7,50,14
+ PUSHBUTTON "Cancel",IDCANCEL,221,24,50,14
+ CONTROL "Match &case",IDC_MATCH_CASE,"Button",BS_AUTOCHECKBOX | WS_TABSTOP,6,24,52,10
+ GROUPBOX "Direction",IDC_STATIC,85,24,119,24
+ CONTROL "&Down",IDC_DIRECTION_DOWN,"Button",BS_AUTORADIOBUTTON | WS_GROUP | WS_TABSTOP,101,34,34,10
+ CONTROL "&Up",IDC_DIRECTION_UP,"Button",BS_AUTORADIOBUTTON,155,34,38,10
+ LTEXT "Fi&nd what:",IDC_STATIC,6,7,35,8
+END
diff --git a/chrome_frame/resources/tlb_resource.rc b/chrome_frame/resources/tlb_resource.rc
new file mode 100644
index 0000000..b16c9d4
--- /dev/null
+++ b/chrome_frame/resources/tlb_resource.rc
@@ -0,0 +1,68 @@
+// Copyright (c) 2009 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.
+
+// NOTE(slightlyoff): the below is generated, and apparently, magical.
+// Seemingly innocuous edits totally destroy registration of the DLL.
+// Edit at your own peril.
+
+
+#define APSTUDIO_READONLY_SYMBOLS
+/////////////////////////////////////////////////////////////////////////////
+//
+// Generated from the TEXTINCLUDE 2 resource.
+//
+#include "winres.h"
+
+/////////////////////////////////////////////////////////////////////////////
+#undef APSTUDIO_READONLY_SYMBOLS
+
+/////////////////////////////////////////////////////////////////////////////
+// English (U.S.) resources
+
+#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU)
+#ifdef _WIN32
+LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US
+#pragma code_page(1252)
+#endif //_WIN32
+
+#ifdef APSTUDIO_INVOKED
+/////////////////////////////////////////////////////////////////////////////
+//
+// TEXTINCLUDE
+//
+
+1 TEXTINCLUDE
+BEGIN
+ "resource.h\0"
+END
+
+2 TEXTINCLUDE
+BEGIN
+ "#include ""winres.h""\r\n"
+ "\0"
+END
+
+3 TEXTINCLUDE
+BEGIN
+ "1 TYPELIB ""chrome_tab.tlb""\r\n"
+ "\0"
+END
+#else
+#include "chrome_tab_version.rc"
+#endif // APSTUDIO_INVOKED
+
+#endif // English (U.S.) resources
+/////////////////////////////////////////////////////////////////////////////
+
+
+
+#ifndef APSTUDIO_INVOKED
+/////////////////////////////////////////////////////////////////////////////
+//
+// Generated from the TEXTINCLUDE 3 resource.
+//
+1 TYPELIB "chrome_tab.tlb"
+
+/////////////////////////////////////////////////////////////////////////////
+#endif // not APSTUDIO_INVOKED
diff --git a/chrome_frame/scoped_ns_ptr_win.h b/chrome_frame/scoped_ns_ptr_win.h
new file mode 100644
index 0000000..91780fd
--- /dev/null
+++ b/chrome_frame/scoped_ns_ptr_win.h
@@ -0,0 +1,149 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_SCOPED_NS_PTR_WIN_H_
+#define CHROME_FRAME_SCOPED_NS_PTR_WIN_H_
+
+#include "base/logging.h"
+#include "base/ref_counted.h"
+
+#include "third_party/xulrunner-sdk/win/include/xpcom/nsISupports.h"
+
+
+// Utility template to prevent users of ScopedNsPtr from calling AddRef and/or
+// Release() without going through the ScopedNsPtr class.
+template <class nsInterface>
+class BlocknsISupportsMethods : public nsInterface {
+ private:
+ NS_IMETHOD QueryInterface(REFNSIID iid, void** object) = 0;
+ NS_IMETHOD_(nsrefcnt) AddRef() = 0;
+ NS_IMETHOD_(nsrefcnt) Release() = 0;
+};
+
+// A smart pointer class for nsISupports.
+// Based on ScopedComPtr.
+// We have our own class instead of nsCOMPtr. nsCOMPtr has parts of its
+// implementation in the xpcomglue lib which we can't use since that lib
+// is built with incompatible build flags to ours.
+template <class nsInterface,
+ const nsIID* interface_id =
+ reinterpret_cast<const nsIID*>(&__uuidof(nsInterface))>
+class ScopedNsPtr : public scoped_refptr<nsInterface> {
+ public:
+ typedef scoped_refptr<nsInterface> ParentClass;
+
+ ScopedNsPtr() {
+ }
+
+ explicit ScopedNsPtr(nsInterface* p) : ParentClass(p) {
+ }
+
+ explicit ScopedNsPtr(const ScopedNsPtr<nsInterface, interface_id>& p)
+ : ParentClass(p) {
+ }
+
+ ~ScopedNsPtr() {
+ // We don't want the smart pointer class to be bigger than the pointer
+ // it wraps.
+ COMPILE_ASSERT(sizeof(ScopedNsPtr<nsInterface, interface_id>) ==
+ sizeof(nsInterface*), ScopedNsPtrSize);
+ }
+
+ // Explicit Release() of the held object. Useful for reuse of the
+ // ScopedNsPtr instance.
+ // Note that this function equates to nsISupports::Release and should not
+ // be confused with e.g. scoped_ptr::release().
+ void Release() {
+ if (ptr_ != NULL) {
+ ptr_->Release();
+ ptr_ = NULL;
+ }
+ }
+
+ // Sets the internal pointer to NULL and returns the held object without
+ // releasing the reference.
+ nsInterface* Detach() {
+ nsInterface* p = ptr_;
+ ptr_ = NULL;
+ return p;
+ }
+
+ // Accepts an interface pointer that has already been addref-ed.
+ void Attach(nsInterface* p) {
+ DCHECK(ptr_ == NULL);
+ ptr_ = p;
+ }
+
+ // Retrieves the pointer address.
+ // Used to receive object pointers as out arguments (and take ownership).
+ // The function DCHECKs on the current value being NULL.
+ // Usage: Foo(p.Receive());
+ nsInterface** Receive() {
+ DCHECK(ptr_ == NULL) << "Object leak. Pointer must be NULL";
+ return &ptr_;
+ }
+
+ template <class Query>
+ nsresult QueryInterface(Query** p) {
+ DCHECK(p != NULL);
+ DCHECK(ptr_ != NULL);
+ return ptr_->QueryInterface(Query::GetIID(), reinterpret_cast<void**>(p));
+ }
+
+ template <class Query>
+ nsresult QueryInterface(const nsIID& iid, Query** p) {
+ DCHECK(p != NULL);
+ DCHECK(ptr_ != NULL);
+ return ptr_->QueryInterface(iid, reinterpret_cast<void**>(p));
+ }
+
+ // Queries |other| for the interface this object wraps and returns the
+ // error code from the other->QueryInterface operation.
+ nsresult QueryFrom(nsISupports* other) {
+ DCHECK(other != NULL);
+ return other->QueryInterface(iid(), reinterpret_cast<void**>(Receive()));
+ }
+
+ // Checks if the identity of |other| and this object is the same.
+ bool IsSameObject(nsISupports* other) {
+ if (!other && !ptr_)
+ return true;
+
+ if (!other || !ptr_)
+ return false;
+
+ nsIID iid = NS_ISUPPORTS_IID;
+ ScopedNsPtr<nsISupports, iid> my_identity;
+ QueryInterface(my_identity.Receive());
+
+ ScopedNsPtr<nsISupports, iid> other_identity;
+ other->QueryInterface(other_identity.Receive());
+
+ return static_cast<nsISupports*>(my_identity) ==
+ static_cast<nsISupports*>(other_identity);
+ }
+
+ // Provides direct access to the interface.
+ // Here we use a well known trick to make sure we block access to
+ // IUknown methods so that something bad like this doesn't happen:
+ // ScopedNsPtr<nsISupports> p(Foo());
+ // p->Release();
+ // ... later the destructor runs, which will Release() again.
+ // and to get the benefit of the DCHECKs we add to QueryInterface.
+ // There's still a way to call these methods if you absolutely must
+ // by statically casting the ScopedNsPtr instance to the wrapped interface
+ // and then making the call... but generally that shouldn't be necessary.
+ BlocknsISupportsMethods<nsInterface>* operator->() const {
+ DCHECK(ptr_ != NULL);
+ return reinterpret_cast<BlocknsISupportsMethods<nsInterface>*>(ptr_);
+ }
+
+ // static methods
+
+ static const nsIID& iid() {
+ return *interface_id;
+ }
+};
+
+#endif // CHROME_FRAME_SCOPED_NS_PTR_WIN_H_
diff --git a/chrome_frame/script_security_manager.h b/chrome_frame/script_security_manager.h
new file mode 100644
index 0000000..a586c84
--- /dev/null
+++ b/chrome_frame/script_security_manager.h
@@ -0,0 +1,785 @@
+/* ***** BEGIN LICENSE BLOCK *****
+ * Version: MPL 1.1/GPL 2.0/LGPL 2.1
+ *
+ * The contents of this file are subject to the Mozilla Public License Version
+ * 1.1 (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ * http://www.mozilla.org/MPL/
+ *
+ * Software distributed under the License is distributed on an "AS IS" basis,
+ * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
+ * for the specific language governing rights and limitations under the
+ * License.
+ *
+ * The Original Code is mozilla.org code.
+ *
+ * The Initial Developer of the Original Code is
+ * Netscape Communications Corporation.
+ * Portions created by the Initial Developer are Copyright (C) 1999
+ * the Initial Developer. All Rights Reserved.
+ *
+ * Contributor(s):
+ *
+ * Alternatively, the contents of this file may be used under the terms of
+ * either of the GNU General Public License Version 2 or later (the "GPL"),
+ * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
+ * in which case the provisions of the GPL or the LGPL are applicable instead
+ * of those above. If you wish to allow use of your version of this file only
+ * under the terms of either the GPL or the LGPL, and not to allow others to
+ * use your version of this file under the terms of the MPL, indicate your
+ * decision by deleting the provisions above and replace them with the notice
+ * and other provisions required by the GPL or the LGPL. If you do not delete
+ * the provisions above, a recipient may use your version of this file under
+ * the terms of any one of the MPL, the GPL or the LGPL.
+ *
+ * ***** END LICENSE BLOCK ***** */
+
+#ifndef CHROME_FRAME_SCRIPT_SECURITY_MANAGER_H_
+#define CHROME_FRAME_SCRIPT_SECURITY_MANAGER_H_
+
+// Gecko headers need this on Windows.
+#ifndef XP_WIN
+#define XP_WIN
+#endif
+
+#include "chrome_frame/ns_associate_iid_win.h"
+#include "third_party/xulrunner-sdk/win/include/caps/nsIScriptSecurityManager.h"
+
+ASSOCIATE_IID(NS_ISCRIPTSECURITYMANAGER_IID_STR, nsIScriptSecurityManager);
+
+// Because we need to support both Firefox 3.0.x and 3.5.x, and because Mozilla
+// changed these unfrozen interfaces, both are declared here, with specific
+// names for specific versions. Doing this makes it easier to adopt new
+// version of the interfaces as they evolve in future version of Firefox.
+
+// The xxx_FF30 declatations below were taken from the file
+// nsIScriptSecurityManager.h in the gecko 1.9.0.5 SDK.
+// The xxx_FF35 declarations below were taken from the file
+// nsIScriptSecurityManager.h in the gecko 1.9.1 SDK.
+
+#define NS_IXPCSECURITYMANAGER_IID_FF30 \
+ {0x31431440, 0xf1ce, 0x11d2, \
+ { 0x98, 0x5a, 0x00, 0x60, 0x08, 0x96, 0x24, 0x22 }}
+
+class NS_NO_VTABLE nsIXPCSecurityManager_FF30 : public nsISupports {
+ public:
+
+ NS_DECLARE_STATIC_IID_ACCESSOR(NS_IXPCSECURITYMANAGER_IID_FF30)
+
+ /**
+ * These flags are used when calling nsIXPConnect::SetSecurityManager
+ */
+ enum { HOOK_CREATE_WRAPPER = 1U };
+
+ enum { HOOK_CREATE_INSTANCE = 2U };
+
+ enum { HOOK_GET_SERVICE = 4U };
+
+ enum { HOOK_CALL_METHOD = 8U };
+
+ enum { HOOK_GET_PROPERTY = 16U };
+
+ enum { HOOK_SET_PROPERTY = 32U };
+
+ enum { HOOK_ALL = 63U };
+
+ /**
+ * For each of these hooks returning NS_OK means 'let the action continue'.
+ * Returning an error code means 'veto the action'. XPConnect will return
+ * JS_FALSE to the js engine if the action is vetoed. The implementor of this
+ * interface is responsible for setting a JS exception into the JSContext
+ * if that is appropriate.
+ */
+ /* void CanCreateWrapper (in JSContextPtr aJSContext, in nsIIDRef aIID,
+ in nsISupports aObj, in nsIClassInfo aClassInfo,
+ inout voidPtr aPolicy); */
+ NS_IMETHOD CanCreateWrapper(JSContext* aJSContext, const nsIID & aIID,
+ nsISupports* aObj, nsIClassInfo* aClassInfo,
+ void** aPolicy) = 0;
+
+ /* void CanCreateInstance (in JSContextPtr aJSContext, in nsCIDRef aCID); */
+ NS_IMETHOD CanCreateInstance(JSContext* aJSContext, const nsCID& aCID) = 0;
+
+ /* void CanGetService (in JSContextPtr aJSContext, in nsCIDRef aCID); */
+ NS_IMETHOD CanGetService(JSContext* aJSContext, const nsCID& aCID) = 0;
+
+ enum { ACCESS_CALL_METHOD = 0U };
+
+ enum { ACCESS_GET_PROPERTY = 1U };
+
+ enum { ACCESS_SET_PROPERTY = 2U };
+
+ /* void CanAccess (in PRUint32 aAction,
+ in nsAXPCNativeCallContextPtr aCallContext,
+ in JSContextPtr aJSContext, in JSObjectPtr aJSObject,
+ in nsISupports aObj, in nsIClassInfo aClassInfo,
+ in JSVal aName, inout voidPtr aPolicy); */
+ NS_IMETHOD CanAccess(PRUint32 aAction,
+ nsAXPCNativeCallContext* aCallContext,
+ JSContext* aJSContext,
+ JSObject* aJSObject,
+ nsISupports* aObj,
+ nsIClassInfo* aClassInfo,
+ jsval aName, void** aPolicy) = 0;
+
+};
+
+NS_DEFINE_STATIC_IID_ACCESSOR(nsIXPCSecurityManager_FF30,
+ NS_IXPCSECURITYMANAGER_IID_FF30);
+
+
+#define NS_ISCRIPTSECURITYMANAGER_IID_STR_FF30 \
+ "3fffd8e8-3fea-442e-a0ed-2ba81ae197d5"
+
+#define NS_ISCRIPTSECURITYMANAGER_IID_FF30 \
+ {0x3fffd8e8, 0x3fea, 0x442e, \
+ { 0xa0, 0xed, 0x2b, 0xa8, 0x1a, 0xe1, 0x97, 0xd5 }}
+
+/**
+ * WARNING!! The JEP needs to call GetSubjectPrincipal()
+ * to support JavaScript-to-Java LiveConnect. So every change to the
+ * nsIScriptSecurityManager interface (big enough to change its IID) also
+ * breaks JavaScript-to-Java LiveConnect on mac.
+ *
+ * If you REALLY have to change this interface, please mark your bug as
+ * blocking bug 293973.
+ */
+class NS_NO_VTABLE NS_SCRIPTABLE nsIScriptSecurityManager_FF30 :
+ public nsIXPCSecurityManager_FF30 {
+ public:
+
+ NS_DECLARE_STATIC_IID_ACCESSOR(NS_ISCRIPTSECURITYMANAGER_IID_FF30)
+
+ /**
+ * Checks whether the running script is allowed to access aProperty.
+ */
+ /* [noscript] void checkPropertyAccess (in JSContextPtr aJSContext,
+ in JSObjectPtr aJSObject,
+ in string aClassName,
+ in JSVal aProperty,
+ in PRUint32 aAction); */
+ NS_IMETHOD CheckPropertyAccess(JSContext* aJSContext, JSObject* aJSObject,
+ const char* aClassName, jsval aProperty,
+ PRUint32 aAction) = 0;
+
+ /**
+ * Checks whether the running script is allowed to connect to aTargetURI
+ */
+ /* [noscript] void checkConnect (in JSContextPtr aJSContext,
+ in nsIURI aTargetURI, in string aClassName,
+ in string aProperty); */
+ NS_IMETHOD CheckConnect(JSContext* aJSContext, nsIURI* aTargetURI,
+ const char* aClassName, const char* aProperty) = 0;
+
+ /**
+ * Check that the script currently running in context "cx" can load "uri".
+ *
+ * Will return error code NS_ERROR_DOM_BAD_URI if the load request
+ * should be denied.
+ *
+ * @param cx the JSContext of the script causing the load
+ * @param uri the URI that is being loaded
+ */
+ /* [noscript] void checkLoadURIFromScript (in JSContextPtr cx,
+ in nsIURI uri); */
+ NS_IMETHOD CheckLoadURIFromScript(JSContext* cx, nsIURI* uri) = 0;
+
+ /**
+ * Default CheckLoadURI permissions
+ */
+ enum { STANDARD = 0U };
+
+ enum { LOAD_IS_AUTOMATIC_DOCUMENT_REPLACEMENT = 1U };
+
+ enum { ALLOW_CHROME = 2U };
+
+ enum { DISALLOW_INHERIT_PRINCIPAL = 4U };
+
+ enum { DISALLOW_SCRIPT_OR_DATA = 4U };
+
+ enum { DISALLOW_SCRIPT = 8U };
+
+ /**
+ * Check that content with principal aPrincipal can load "uri".
+ *
+ * Will return error code NS_ERROR_DOM_BAD_URI if the load request
+ * should be denied.
+ *
+ * @param aPrincipal the principal identifying the actor causing the load
+ * @param uri the URI that is being loaded
+ * @param flags the permission set, see above
+ */
+ /* void checkLoadURIWithPrincipal (in nsIPrincipal aPrincipal, in nsIURI uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURIWithPrincipal(nsIPrincipal* aPrincipal,
+ nsIURI* uri, PRUint32 flags) = 0;
+
+ /**
+ * Check that content from "from" can load "uri".
+ *
+ * Will return error code NS_ERROR_DOM_BAD_URI if the load request
+ * should be denied.
+ *
+ * @param from the URI causing the load
+ * @param uri the URI that is being loaded
+ * @param flags the permission set, see above
+ *
+ * @deprecated Use checkLoadURIWithPrincipal instead of this function.
+ */
+ /* void checkLoadURI (in nsIURI from, in nsIURI uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURI(nsIURI* from, nsIURI* uri,
+ PRUint32 flags) = 0;
+
+ /**
+ * Similar to checkLoadURIWithPrincipal but there are two differences:
+ *
+ * 1) The URI is a string, not a URI object.
+ * 2) This function assumes that the URI may still be subject to fixup (and
+ * hence will check whether fixed-up versions of the URI are allowed to
+ * load as well); if any of the versions of this URI is not allowed, this
+ * function will return error code NS_ERROR_DOM_BAD_URI.
+ */
+ /* void checkLoadURIStrWithPrincipal (in nsIPrincipal aPrincipal,
+ in AUTF8String uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURIStrWithPrincipal(
+ nsIPrincipal* aPrincipal, const nsACString& uri, PRUint32 flags) = 0;
+
+ /**
+ * Same as CheckLoadURI but takes string arguments for ease of use
+ * by scripts
+ *
+ * @deprecated Use checkLoadURIStrWithPrincipal instead of this function.
+ */
+ /* void checkLoadURIStr (in AUTF8String from, in AUTF8String uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURIStr(const nsACString& from,
+ const nsACString& uri,
+ PRUint32 flags) = 0;
+
+ /**
+ * Check that the function 'funObj' is allowed to run on 'targetObj'
+ *
+ * Will return error code NS_ERROR_DOM_SECURITY_ERR if the function
+ * should not run
+ *
+ * @param cx The current active JavaScript context.
+ * @param funObj The function trying to run..
+ * @param targetObj The object the function will run on.
+ */
+ /* [noscript] void checkFunctionAccess (in JSContextPtr cx,
+ in voidPtr funObj,
+ in voidPtr targetObj); */
+ NS_IMETHOD CheckFunctionAccess(JSContext* cx, void* funObj,
+ void* targetObj) = 0;
+
+ /**
+ * Return true if content from the given principal is allowed to
+ * execute scripts.
+ */
+ /* [noscript] boolean canExecuteScripts (in JSContextPtr cx,
+ in nsIPrincipal principal); */
+ NS_IMETHOD CanExecuteScripts(JSContext* cx, nsIPrincipal* principal,
+ PRBool* _retval) = 0;
+
+ /**
+ * Return the principal of the innermost frame of the currently
+ * executing script. Will return null if there is no script
+ * currently executing.
+ */
+ /* [noscript] nsIPrincipal getSubjectPrincipal (); */
+ NS_IMETHOD GetSubjectPrincipal(nsIPrincipal** _retval) = 0;
+
+ /**
+ * Return the all-powerful system principal.
+ */
+ /* [noscript] nsIPrincipal getSystemPrincipal (); */
+ NS_IMETHOD GetSystemPrincipal(nsIPrincipal** _retval) = 0;
+
+ /**
+ * Return a principal with the specified certificate fingerprint, subject
+ * name (the full name or concatenated set of names of the entity
+ * represented by the certificate), pretty name, certificate, and
+ * codebase URI. The certificate fingerprint and subject name MUST be
+ * nonempty; otherwise an error will be thrown. Similarly, aCert must
+ * not be null.
+ */
+ /* [noscript] nsIPrincipal getCertificatePrincipal (
+ in AUTF8String aCertFingerprint, in AUTF8String aSubjectName,
+ in AUTF8String aPrettyName, in nsISupports aCert, in nsIURI aURI); */
+ NS_IMETHOD GetCertificatePrincipal(const nsACString& aCertFingerprint,
+ const nsACString& aSubjectName,
+ const nsACString& aPrettyName,
+ nsISupports* aCert,
+ nsIURI* aURI, nsIPrincipal** _retval) = 0;
+
+ /**
+ * Return a principal that has the same origin as aURI.
+ */
+ /* nsIPrincipal getCodebasePrincipal (in nsIURI aURI); */
+ NS_SCRIPTABLE NS_IMETHOD GetCodebasePrincipal(nsIURI* aURI,
+ nsIPrincipal **_retval) = 0;
+
+ /**
+ * Request that 'capability' can be enabled by scripts or applets
+ * running with 'principal'. Will prompt user if
+ * necessary. Returns nsIPrincipal::ENABLE_GRANTED or
+ * nsIPrincipal::ENABLE_DENIED based on user's choice.
+ */
+ /* [noscript] short requestCapability (in nsIPrincipal principal,
+ in string capability); */
+ NS_IMETHOD RequestCapability(nsIPrincipal* principal,
+ const char* capability, PRInt16* _retval) = 0;
+
+ /**
+ * Return true if the currently executing script has 'capability' enabled.
+ */
+ /* boolean isCapabilityEnabled (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD IsCapabilityEnabled(const char* capability,
+ PRBool* _retval) = 0;
+
+ /**
+ * Enable 'capability' in the innermost frame of the currently executing
+ * script.
+ */
+ /* void enableCapability (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD EnableCapability(const char* capability) = 0;
+
+ /**
+ * Remove 'capability' from the innermost frame of the currently
+ * executing script. Any setting of 'capability' from enclosing
+ * frames thus comes into effect.
+ */
+ /* void revertCapability (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD RevertCapability(const char* capability) = 0;
+
+ /**
+ * Disable 'capability' in the innermost frame of the currently executing
+ * script.
+ */
+ /* void disableCapability (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD DisableCapability(const char* capability) = 0;
+
+ /**
+ * Allow 'certificateID' to enable 'capability.' Can only be performed
+ * by code signed by the system certificate.
+ */
+ /* void setCanEnableCapability (in AUTF8String certificateFingerprint,
+ in string capability, in short canEnable); */
+ NS_SCRIPTABLE NS_IMETHOD SetCanEnableCapability(
+ const nsACString& certificateFingerprint, const char* capability,
+ PRInt16 canEnable) = 0;
+
+ /**
+ * Return the principal of the specified object in the specified context.
+ */
+ /* [noscript] nsIPrincipal getObjectPrincipal (in JSContextPtr cx,
+ in JSObjectPtr obj); */
+ NS_IMETHOD GetObjectPrincipal(JSContext* cx, JSObject* obj,
+ nsIPrincipal** _retval) = 0;
+
+ /**
+ * Returns true if the principal of the currently running script is the
+ * system principal, false otherwise.
+ */
+ /* [noscript] boolean subjectPrincipalIsSystem (); */
+ NS_IMETHOD SubjectPrincipalIsSystem(PRBool* _retval) = 0;
+
+ /**
+ * Returns OK if aJSContext and target have the same "origin"
+ * (scheme, host, and port).
+ */
+ /* [noscript] void checkSameOrigin (in JSContextPtr aJSContext,
+ in nsIURI aTargetURI); */
+ NS_IMETHOD CheckSameOrigin(JSContext* aJSContext, nsIURI* aTargetURI) = 0;
+
+ /**
+ * Returns OK if aSourceURI and target have the same "origin"
+ * (scheme, host, and port).
+ * ReportError flag suppresses error reports for functions that
+ * don't need reporting.
+ */
+ /* void checkSameOriginURI (in nsIURI aSourceURI, in nsIURI aTargetURI,
+ in boolean reportError); */
+ NS_SCRIPTABLE NS_IMETHOD CheckSameOriginURI(nsIURI* aSourceURI,
+ nsIURI* aTargetURI,
+ PRBool reportError) = 0;
+
+ /**
+ * Returns the principal of the global object of the given context, or null
+ * if no global or no principal.
+ */
+ /* [noscript] nsIPrincipal getPrincipalFromContext (in JSContextPtr cx); */
+ NS_IMETHOD GetPrincipalFromContext(JSContext* cx,
+ nsIPrincipal** _retval) = 0;
+
+ /**
+ * Get the principal for the given channel. This will typically be the
+ * channel owner if there is one, and the codebase principal for the
+ * channel's URI otherwise. aChannel must not be null.
+ */
+ /* nsIPrincipal getChannelPrincipal (in nsIChannel aChannel); */
+ NS_SCRIPTABLE NS_IMETHOD GetChannelPrincipal(nsIChannel* aChannel,
+ nsIPrincipal** _retval) = 0;
+
+ /**
+ * Check whether a given principal is a system principal. This allows us
+ * to avoid handing back the system principal to script while allowing
+ * script to check whether a given principal is system.
+ */
+ /* boolean isSystemPrincipal (in nsIPrincipal aPrincipal); */
+ NS_SCRIPTABLE NS_IMETHOD IsSystemPrincipal(nsIPrincipal* aPrincipal,
+ PRBool* _retval) = 0;
+
+ /**
+ * Same as getSubjectPrincipal(), only faster. cx must *never* be
+ * passed null, and it must be the context on the top of the
+ * context stack. Does *not* reference count the returned
+ * principal.
+ */
+ /* [noscript, notxpcom] nsIPrincipal getCxSubjectPrincipal (
+ in JSContextPtr cx); */
+ NS_IMETHOD_(nsIPrincipal *) GetCxSubjectPrincipal(JSContext* cx) = 0;
+
+};
+
+NS_DEFINE_STATIC_IID_ACCESSOR(nsIScriptSecurityManager_FF30,
+ NS_ISCRIPTSECURITYMANAGER_IID_FF30);
+
+ASSOCIATE_IID(NS_ISCRIPTSECURITYMANAGER_IID_STR_FF30,
+ nsIScriptSecurityManager_FF30);
+
+
+#define NS_ISCRIPTSECURITYMANAGER_IID_STR_FF35 \
+ "f8e350b9-9f31-451a-8c8f-d10fea26b780"
+
+#define NS_ISCRIPTSECURITYMANAGER_IID_FF35 \
+ {0xf8e350b9, 0x9f31, 0x451a, \
+ { 0x8c, 0x8f, 0xd1, 0x0f, 0xea, 0x26, 0xb7, 0x80 }}
+
+#ifndef NS_OUTPARAM
+#define NS_OUTPARAM
+#endif
+
+/**
+ * WARNING!! The JEP needs to call GetSubjectPrincipal()
+ * to support JavaScript-to-Java LiveConnect. So every change to the
+ * nsIScriptSecurityManager interface (big enough to change its IID) also
+ * breaks JavaScript-to-Java LiveConnect on mac.
+ *
+ * If you REALLY have to change this interface, please mark your bug as
+ * blocking bug 293973.
+ */
+class NS_NO_VTABLE NS_SCRIPTABLE nsIScriptSecurityManager_FF35 :
+ public nsIXPCSecurityManager_FF30 {
+ public:
+
+ NS_DECLARE_STATIC_IID_ACCESSOR(NS_ISCRIPTSECURITYMANAGER_IID_FF35)
+
+ /**
+ * Checks whether the running script is allowed to access aProperty.
+ */
+ /* [noscript] void checkPropertyAccess (in JSContextPtr aJSContext,
+ in JSObjectPtr aJSObject,
+ in string aClassName,
+ in JSVal aProperty,
+ in PRUint32 aAction); */
+ NS_IMETHOD CheckPropertyAccess(JSContext* aJSContext, JSObject* aJSObject,
+ const char* aClassName, jsval aProperty,
+ PRUint32 aAction) = 0;
+
+ /**
+ * Checks whether the running script is allowed to connect to aTargetURI
+ */
+ /* [noscript] void checkConnect (in JSContextPtr aJSContext,
+ in nsIURI aTargetURI, in string aClassName,
+ in string aProperty); */
+ NS_IMETHOD CheckConnect(JSContext* aJSContext, nsIURI* aTargetURI,
+ const char* aClassName, const char* aProperty) = 0;
+
+ /**
+ * Check that the script currently running in context "cx" can load "uri".
+ *
+ * Will return error code NS_ERROR_DOM_BAD_URI if the load request
+ * should be denied.
+ *
+ * @param cx the JSContext of the script causing the load
+ * @param uri the URI that is being loaded
+ */
+ /* [noscript] void checkLoadURIFromScript (in JSContextPtr cx,
+ in nsIURI uri); */
+ NS_IMETHOD CheckLoadURIFromScript(JSContext* cx, nsIURI* uri) = 0;
+
+ /**
+ * Default CheckLoadURI permissions
+ */
+ enum { STANDARD = 0U };
+
+ enum { LOAD_IS_AUTOMATIC_DOCUMENT_REPLACEMENT = 1U };
+
+ enum { ALLOW_CHROME = 2U };
+
+ enum { DISALLOW_INHERIT_PRINCIPAL = 4U };
+
+ enum { DISALLOW_SCRIPT_OR_DATA = 4U };
+
+ enum { DISALLOW_SCRIPT = 8U };
+
+ /**
+ * Check that content with principal aPrincipal can load "uri".
+ *
+ * Will return error code NS_ERROR_DOM_BAD_URI if the load request
+ * should be denied.
+ *
+ * @param aPrincipal the principal identifying the actor causing the load
+ * @param uri the URI that is being loaded
+ * @param flags the permission set, see above
+ */
+ /* void checkLoadURIWithPrincipal (in nsIPrincipal aPrincipal, in nsIURI uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURIWithPrincipal(nsIPrincipal* aPrincipal,
+ nsIURI* uri,
+ PRUint32 flags) = 0;
+
+ /**
+ * Check that content from "from" can load "uri".
+ *
+ * Will return error code NS_ERROR_DOM_BAD_URI if the load request
+ * should be denied.
+ *
+ * @param from the URI causing the load
+ * @param uri the URI that is being loaded
+ * @param flags the permission set, see above
+ *
+ * @deprecated Use checkLoadURIWithPrincipal instead of this function.
+ */
+ /* void checkLoadURI (in nsIURI from, in nsIURI uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURI(nsIURI* from, nsIURI* uri,
+ PRUint32 flags) = 0;
+
+ /**
+ * Similar to checkLoadURIWithPrincipal but there are two differences:
+ *
+ * 1) The URI is a string, not a URI object.
+ * 2) This function assumes that the URI may still be subject to fixup (and
+ * hence will check whether fixed-up versions of the URI are allowed to
+ * load as well); if any of the versions of this URI is not allowed, this
+ * function will return error code NS_ERROR_DOM_BAD_URI.
+ */
+ /* void checkLoadURIStrWithPrincipal (in nsIPrincipal aPrincipal,
+ in AUTF8String uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURIStrWithPrincipal(
+ nsIPrincipal* aPrincipal, const nsACString& uri, PRUint32 flags) = 0;
+
+ /**
+ * Same as CheckLoadURI but takes string arguments for ease of use
+ * by scripts
+ *
+ * @deprecated Use checkLoadURIStrWithPrincipal instead of this function.
+ */
+ /* void checkLoadURIStr (in AUTF8String from, in AUTF8String uri,
+ in unsigned long flags); */
+ NS_SCRIPTABLE NS_IMETHOD CheckLoadURIStr(const nsACString& from,
+ const nsACString & uri,
+ PRUint32 flags) = 0;
+
+ /**
+ * Check that the function 'funObj' is allowed to run on 'targetObj'
+ *
+ * Will return error code NS_ERROR_DOM_SECURITY_ERR if the function
+ * should not run
+ *
+ * @param cx The current active JavaScript context.
+ * @param funObj The function trying to run..
+ * @param targetObj The object the function will run on.
+ */
+ /* [noscript] void checkFunctionAccess (in JSContextPtr cx,
+ in voidPtr funObj,
+ in voidPtr targetObj); */
+ NS_IMETHOD CheckFunctionAccess(JSContext* cx, void* funObj,
+ void* targetObj) = 0;
+
+ /**
+ * Return true if content from the given principal is allowed to
+ * execute scripts.
+ */
+ /* [noscript] boolean canExecuteScripts (in JSContextPtr cx,
+ in nsIPrincipal principal); */
+ NS_IMETHOD CanExecuteScripts(JSContext* cx, nsIPrincipal* principal,
+ PRBool* _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Return the principal of the innermost frame of the currently
+ * executing script. Will return null if there is no script
+ * currently executing.
+ */
+ /* [noscript] nsIPrincipal getSubjectPrincipal (); */
+ NS_IMETHOD GetSubjectPrincipal(nsIPrincipal **_retval NS_OUTPARAM) = 0;
+
+ /**
+ * Return the all-powerful system principal.
+ */
+ /* [noscript] nsIPrincipal getSystemPrincipal (); */
+ NS_IMETHOD GetSystemPrincipal(nsIPrincipal **_retval NS_OUTPARAM) = 0;
+
+ /**
+ * Return a principal with the specified certificate fingerprint, subject
+ * name (the full name or concatenated set of names of the entity
+ * represented by the certificate), pretty name, certificate, and
+ * codebase URI. The certificate fingerprint and subject name MUST be
+ * nonempty; otherwise an error will be thrown. Similarly, aCert must
+ * not be null.
+ */
+ /* [noscript] nsIPrincipal getCertificatePrincipal (
+ in AUTF8String aCertFingerprint, in AUTF8String aSubjectName,
+ in AUTF8String aPrettyName, in nsISupports aCert, in nsIURI aURI); */
+ NS_IMETHOD GetCertificatePrincipal(const nsACString& aCertFingerprint,
+ const nsACString& aSubjectName,
+ const nsACString& aPrettyName,
+ nsISupports* aCert, nsIURI *aURI,
+ nsIPrincipal** _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Return a principal that has the same origin as aURI.
+ */
+ /* nsIPrincipal getCodebasePrincipal (in nsIURI aURI); */
+ NS_SCRIPTABLE NS_IMETHOD GetCodebasePrincipal(nsIURI* aURI,
+ nsIPrincipal** _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Request that 'capability' can be enabled by scripts or applets
+ * running with 'principal'. Will prompt user if
+ * necessary. Returns nsIPrincipal::ENABLE_GRANTED or
+ * nsIPrincipal::ENABLE_DENIED based on user's choice.
+ */
+ /* [noscript] short requestCapability (in nsIPrincipal principal,
+ in string capability); */
+ NS_IMETHOD RequestCapability(nsIPrincipal* principal, const char* capability,
+ PRInt16* _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Return true if the currently executing script has 'capability' enabled.
+ */
+ /* boolean isCapabilityEnabled (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD IsCapabilityEnabled(const char* capability,
+ PRBool* _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Enable 'capability' in the innermost frame of the currently executing
+ * script.
+ */
+ /* void enableCapability (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD EnableCapability(const char* capability) = 0;
+
+ /**
+ * Remove 'capability' from the innermost frame of the currently
+ * executing script. Any setting of 'capability' from enclosing
+ * frames thus comes into effect.
+ */
+ /* void revertCapability (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD RevertCapability(const char* capability) = 0;
+
+ /**
+ * Disable 'capability' in the innermost frame of the currently executing
+ * script.
+ */
+ /* void disableCapability (in string capability); */
+ NS_SCRIPTABLE NS_IMETHOD DisableCapability(const char* capability) = 0;
+
+ /**
+ * Allow 'certificateID' to enable 'capability.' Can only be performed
+ * by code signed by the system certificate.
+ */
+ /* void setCanEnableCapability (in AUTF8String certificateFingerprint,
+ in string capability, in short canEnable); */
+ NS_SCRIPTABLE NS_IMETHOD SetCanEnableCapability(
+ const nsACString& certificateFingerprint, const char *capability,
+ PRInt16 canEnable) = 0;
+
+ /**
+ * Return the principal of the specified object in the specified context.
+ */
+ /* [noscript] nsIPrincipal getObjectPrincipal (in JSContextPtr cx,
+ in JSObjectPtr obj); */
+ NS_IMETHOD GetObjectPrincipal(JSContext* cx, JSObject* obj,
+ nsIPrincipal** _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Returns true if the principal of the currently running script is the
+ * system principal, false otherwise.
+ */
+ /* [noscript] boolean subjectPrincipalIsSystem (); */
+ NS_IMETHOD SubjectPrincipalIsSystem(PRBool* _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Returns OK if aJSContext and target have the same "origin"
+ * (scheme, host, and port).
+ */
+ /* [noscript] void checkSameOrigin (in JSContextPtr aJSContext,
+ in nsIURI aTargetURI); */
+ NS_IMETHOD CheckSameOrigin(JSContext* aJSContext, nsIURI* aTargetURI) = 0;
+
+ /**
+ * Returns OK if aSourceURI and target have the same "origin"
+ * (scheme, host, and port).
+ * ReportError flag suppresses error reports for functions that
+ * don't need reporting.
+ */
+ /* void checkSameOriginURI (in nsIURI aSourceURI, in nsIURI aTargetURI,
+ in boolean reportError); */
+ NS_SCRIPTABLE NS_IMETHOD CheckSameOriginURI(nsIURI* aSourceURI,
+ nsIURI* aTargetURI,
+ PRBool reportError) = 0;
+
+ /**
+ * Returns the principal of the global object of the given context, or null
+ * if no global or no principal.
+ */
+ /* [noscript] nsIPrincipal getPrincipalFromContext (in JSContextPtr cx); */
+ NS_IMETHOD GetPrincipalFromContext(JSContext* cx,
+ nsIPrincipal** _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Get the principal for the given channel. This will typically be the
+ * channel owner if there is one, and the codebase principal for the
+ * channel's URI otherwise. aChannel must not be null.
+ */
+ /* nsIPrincipal getChannelPrincipal (in nsIChannel aChannel); */
+ NS_SCRIPTABLE NS_IMETHOD GetChannelPrincipal(nsIChannel* aChannel,
+ nsIPrincipal** _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Check whether a given principal is a system principal. This allows us
+ * to avoid handing back the system principal to script while allowing
+ * script to check whether a given principal is system.
+ */
+ /* boolean isSystemPrincipal (in nsIPrincipal aPrincipal); */
+ NS_SCRIPTABLE NS_IMETHOD IsSystemPrincipal(nsIPrincipal* aPrincipal,
+ PRBool* _retval NS_OUTPARAM) = 0;
+
+ /**
+ * Same as getSubjectPrincipal(), only faster. cx must *never* be
+ * passed null, and it must be the context on the top of the
+ * context stack. Does *not* reference count the returned
+ * principal.
+ */
+ /* [noscript, notxpcom] nsIPrincipal getCxSubjectPrincipal (
+ in JSContextPtr cx); */
+ NS_IMETHOD_(nsIPrincipal*) GetCxSubjectPrincipal(JSContext* cx) = 0;
+
+ /* [noscript, notxpcom] nsIPrincipal getCxSubjectPrincipalAndFrame (in JSContextPtr cx, out JSStackFramePtr fp); */
+ NS_IMETHOD_(nsIPrincipal*) GetCxSubjectPrincipalAndFrame(JSContext* cx,
+ JSStackFrame** fp NS_OUTPARAM) = 0;
+
+};
+
+NS_DEFINE_STATIC_IID_ACCESSOR(nsIScriptSecurityManager_FF35,
+ NS_ISCRIPTSECURITYMANAGER_IID_FF35);
+
+ASSOCIATE_IID(NS_ISCRIPTSECURITYMANAGER_IID_STR_FF35,
+ nsIScriptSecurityManager_FF35);
+
+#endif // CHROME_FRAME_SCRIPT_SECURITY_MANAGER_H_
diff --git a/chrome_frame/support.gyp b/chrome_frame/support.gyp
new file mode 100644
index 0000000..30f5889
--- /dev/null
+++ b/chrome_frame/support.gyp
@@ -0,0 +1,17 @@
+# Copyright (c) 2009 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.
+
+{
+ 'targets': [
+ {
+ 'target_name': 'support',
+ 'type': 'none',
+ 'dependencies': [
+ 'chrome_frame.gyp:*',
+ ],
+ },
+ ],
+}
+
+# vim: shiftwidth=2:et:ai:tabstop=2
diff --git a/chrome_frame/sync_msg_reply_dispatcher.cc b/chrome_frame/sync_msg_reply_dispatcher.cc
new file mode 100644
index 0000000..b301698
--- /dev/null
+++ b/chrome_frame/sync_msg_reply_dispatcher.cc
@@ -0,0 +1,61 @@
+// Copyright (c) 2009 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 "chrome_frame/sync_msg_reply_dispatcher.h"
+
+#include "ipc/ipc_sync_message.h"
+
+void SyncMessageReplyDispatcher::Push(IPC::SyncMessage* msg, void* callback,
+ void* key) {
+ MessageSent pending(IPC::SyncMessage::GetMessageId(*msg),
+ msg->type(), callback, key);
+ AutoLock lock(message_queue_lock_);
+ message_queue_.push_back(pending);
+}
+
+bool SyncMessageReplyDispatcher::HandleMessageType(const IPC::Message& msg,
+ const MessageSent& origin) {
+ return false;
+}
+
+bool SyncMessageReplyDispatcher::OnMessageReceived(const IPC::Message& msg) {
+ MessageSent origin;
+ if (!Pop(msg, &origin)) {
+ return false;
+ }
+
+ // No callback e.g. no return values and/or don't care
+ if (origin.callback == NULL)
+ return true;
+
+ return HandleMessageType(msg, origin);
+}
+
+void SyncMessageReplyDispatcher::Cancel(void* key) {
+ DCHECK(key != NULL);
+ AutoLock lock(message_queue_lock_);
+ PendingSyncMessageQueue::iterator it;
+ for (it = message_queue_.begin(); it != message_queue_.end(); ++it) {
+ if (it->key == key) {
+ it->key = NULL;
+ }
+ }
+}
+
+bool SyncMessageReplyDispatcher::Pop(const IPC::Message& msg, MessageSent* t) {
+ if (!msg.is_reply())
+ return false;
+
+ int id = IPC::SyncMessage::GetMessageId(msg);
+ AutoLock lock(message_queue_lock_);
+ PendingSyncMessageQueue::iterator it;
+ for (it = message_queue_.begin(); it != message_queue_.end(); ++it) {
+ if (it->id == id) {
+ *t = *it;
+ message_queue_.erase(it);
+ return true;
+ }
+ }
+ return false;
+}
diff --git a/chrome_frame/sync_msg_reply_dispatcher.h b/chrome_frame/sync_msg_reply_dispatcher.h
new file mode 100644
index 0000000..05e5162
--- /dev/null
+++ b/chrome_frame/sync_msg_reply_dispatcher.h
@@ -0,0 +1,96 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_SYNC_MSG_REPLY_DISPATCHER_H_
+#define CHROME_FRAME_SYNC_MSG_REPLY_DISPATCHER_H_
+
+#include <deque>
+
+#include "base/lock.h"
+#include "ipc/ipc_channel_proxy.h"
+
+// Base class used to allow synchronous IPC messages to be sent and
+// received in an asynchronous manner. To use this class add it as a filter to
+// your IPC channel using ChannelProxy::AddFilter(). From then on, before
+// sending a synchronous message, call SyncMessageReplyDispatcher::Push() with
+// a callback and a key. This class will then handle the message response and
+// will call the callback when it is received.
+//
+// This class is intended to be extended by classes implementing
+// HandleMessageType with delegation for the messages they expect to receive in
+// cases where you care about the return values of synchronous messages.
+//
+// Sample usage pattern:
+//
+// // Add handling for desired message types.
+// class SyncMessageReplyDispatcherImpl : public SyncMessageReplyDispatcher {
+// virtual bool HandleMessageType(const IPC::Message& msg,
+// const MessageSent& origin) {
+// switch (origin.type) {
+// case AutomationMsg_CreateExternalTab::ID:
+// InvokeCallback<Tuple3<HWND, HWND, int> >(msg, origin);
+// break;
+// [HANDLING FOR OTHER EXPECTED MESSAGE TYPES]
+// }
+// }
+//
+// // Add the filter
+// IPC::SyncChannel channel_;
+// channel_.AddFilter(new SyncMessageReplyDispatcherImpl());
+//
+// sync_->Push(msg, NewCallback(this, CallbackMethod, this);
+// channel_->ChannelProxy::Send(msg);
+//
+class SyncMessageReplyDispatcher : public IPC::ChannelProxy::MessageFilter {
+ public:
+ SyncMessageReplyDispatcher() {}
+ void Push(IPC::SyncMessage* msg, void* callback, void* key);
+ void Cancel(void* key);
+
+ protected:
+ struct MessageSent {
+ MessageSent() {}
+ MessageSent(int id, uint16 type, void* callback, void* key)
+ : id(id), callback(callback), type(type), key(key) {}
+ int id;
+ void* callback;
+ void* key;
+ uint16 type;
+ };
+
+ typedef std::deque<MessageSent> PendingSyncMessageQueue;
+
+ bool Pop(const IPC::Message& msg, MessageSent* t);
+ virtual bool OnMessageReceived(const IPC::Message& msg);
+
+ // Child classes must implement a handler for the message types they are
+ // interested in handling responses for. If you don't care about the replies
+ // to any of the sync messages you are handling, then you don't have to
+ // implement this.
+ virtual bool HandleMessageType(const IPC::Message& msg,
+ const MessageSent& origin);
+
+ template <typename T>
+ void InvokeCallback(const IPC::Message& msg, const MessageSent& origin) {
+ // Ensure we delete the callback
+ scoped_ptr<CallbackRunner<T> > c(
+ reinterpret_cast<CallbackRunner<T>*>(origin.callback));
+ if (origin.key == NULL) { // Canceled
+ return;
+ }
+
+ T tmp; // Acts as "initializer" for output parameters.
+ IPC::ParamDeserializer<T> deserializer(tmp);
+ if (deserializer.MessageReplyDeserializer::SerializeOutputParameters(msg)) {
+ c->RunWithParams(deserializer.out_);
+ } else {
+ // TODO(stoyan): How to handle errors?
+ }
+ }
+
+ PendingSyncMessageQueue message_queue_;
+ Lock message_queue_lock_;
+};
+
+#endif // CHROME_FRAME_SYNC_MSG_REPLY_DISPATCHER_H_
diff --git a/chrome_frame/test/ChromeTab_UnitTests.vcproj b/chrome_frame/test/ChromeTab_UnitTests.vcproj
new file mode 100644
index 0000000..9f4369b
--- /dev/null
+++ b/chrome_frame/test/ChromeTab_UnitTests.vcproj
@@ -0,0 +1,273 @@
+<?xml version="1.0" encoding="Windows-1252"?>
+<VisualStudioProject
+ ProjectType="Visual C++"
+ Version="8.00"
+ Name="chrometab_unittests"
+ ProjectGUID="{BA08FE92-567D-4411-B344-17ADAECA2B5A}"
+ RootNamespace="ChromeTab_UnitTests"
+ Keyword="Win32Proj"
+ >
+ <Platforms>
+ <Platform
+ Name="Win32"
+ />
+ </Platforms>
+ <ToolFiles>
+ </ToolFiles>
+ <Configurations>
+ <Configuration
+ Name="Debug|Win32"
+ OutputDirectory="..\..\chrome\$(ConfigurationName)"
+ IntermediateDirectory="$(ConfigurationName)"
+ ConfigurationType="1"
+ InheritedPropertySheets="$(SolutionDir)..\build\common.vsprops;$(SolutionDir)..\build\debug.vsprops;$(SolutionDir)..\skia\using_skia.vsprops;$(SolutionDir)..\third_party\libxml\build\using_libxml.vsprops;$(SolutionDir)..\third_party\libxslt\build\using_libxslt.vsprops;..\..\testing\using_gtest.vsprops;$(SolutionDir)..\chrome\third_party\wtl\using_wtl.vsprops;.\chrometab_unittests.vsprops"
+ >
+ <Tool
+ Name="VCPreBuildEventTool"
+ />
+ <Tool
+ Name="VCCustomBuildTool"
+ />
+ <Tool
+ Name="VCXMLDataGeneratorTool"
+ />
+ <Tool
+ Name="VCWebServiceProxyGeneratorTool"
+ />
+ <Tool
+ Name="VCMIDLTool"
+ />
+ <Tool
+ Name="VCCLCompilerTool"
+ Optimization="0"
+ PreprocessorDefinitions="WIN32;_DEBUG;_CONSOLE"
+ MinimalRebuild="true"
+ BasicRuntimeChecks="3"
+ RuntimeLibrary="1"
+ UsePrecompiledHeader="0"
+ WarningLevel="3"
+ Detect64BitPortabilityProblems="false"
+ DebugInformationFormat="4"
+ />
+ <Tool
+ Name="VCManagedResourceCompilerTool"
+ />
+ <Tool
+ Name="VCResourceCompilerTool"
+ />
+ <Tool
+ Name="VCPreLinkEventTool"
+ />
+ <Tool
+ Name="VCLinkerTool"
+ LinkIncremental="2"
+ GenerateDebugInformation="true"
+ SubSystem="1"
+ TargetMachine="1"
+ />
+ <Tool
+ Name="VCALinkTool"
+ />
+ <Tool
+ Name="VCManifestTool"
+ />
+ <Tool
+ Name="VCXDCMakeTool"
+ />
+ <Tool
+ Name="VCBscMakeTool"
+ />
+ <Tool
+ Name="VCFxCopTool"
+ />
+ <Tool
+ Name="VCAppVerifierTool"
+ />
+ <Tool
+ Name="VCPostBuildEventTool"
+ />
+ </Configuration>
+ <Configuration
+ Name="Release|Win32"
+ OutputDirectory="..\..\chrome\$(ConfigurationName)"
+ IntermediateDirectory="$(ConfigurationName)"
+ ConfigurationType="1"
+ InheritedPropertySheets="$(SolutionDir)..\build\common.vsprops;$(SolutionDir)..\skia\using_skia.vsprops;$(SolutionDir)..\third_party\libxml\build\using_libxml.vsprops;$(SolutionDir)..\third_party\libxslt\build\using_libxslt.vsprops;..\..\testing\using_gtest.vsprops;$(SolutionDir)..\chrome\third_party\wtl\using_wtl.vsprops;$(SolutionDir)..\build\release.vsprops;.\chrometab_unittests.vsprops"
+ UseOfATL="1"
+ ATLMinimizesCRunTimeLibraryUsage="false"
+ WholeProgramOptimization="1"
+ >
+ <Tool
+ Name="VCPreBuildEventTool"
+ />
+ <Tool
+ Name="VCCustomBuildTool"
+ />
+ <Tool
+ Name="VCXMLDataGeneratorTool"
+ />
+ <Tool
+ Name="VCWebServiceProxyGeneratorTool"
+ />
+ <Tool
+ Name="VCMIDLTool"
+ />
+ <Tool
+ Name="VCCLCompilerTool"
+ Optimization="2"
+ OmitFramePointers="false"
+ WholeProgramOptimization="false"
+ PreprocessorDefinitions="WIN32;NDEBUG;_CONSOLE;"
+ StringPooling="true"
+ BasicRuntimeChecks="0"
+ RuntimeLibrary="0"
+ UsePrecompiledHeader="0"
+ WarningLevel="3"
+ Detect64BitPortabilityProblems="false"
+ DebugInformationFormat="3"
+ />
+ <Tool
+ Name="VCManagedResourceCompilerTool"
+ />
+ <Tool
+ Name="VCResourceCompilerTool"
+ />
+ <Tool
+ Name="VCPreLinkEventTool"
+ />
+ <Tool
+ Name="VCLinkerTool"
+ LinkIncremental="1"
+ AdditionalLibraryDirectories="$(ConfigurationName)\lib;"
+ IgnoreDefaultLibraryNames=""
+ GenerateDebugInformation="true"
+ SubSystem="1"
+ OptimizeReferences="2"
+ EnableCOMDATFolding="2"
+ OptimizeForWindows98="1"
+ TargetMachine="1"
+ />
+ <Tool
+ Name="VCALinkTool"
+ />
+ <Tool
+ Name="VCManifestTool"
+ />
+ <Tool
+ Name="VCXDCMakeTool"
+ />
+ <Tool
+ Name="VCBscMakeTool"
+ />
+ <Tool
+ Name="VCFxCopTool"
+ />
+ <Tool
+ Name="VCAppVerifierTool"
+ />
+ <Tool
+ Name="VCPostBuildEventTool"
+ />
+ </Configuration>
+ </Configurations>
+ <References>
+ </References>
+ <Files>
+ <Filter
+ Name="Source Files"
+ Filter="cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx"
+ UniqueIdentifier="{4FC737F1-C7A5-4376-A066-2A32D752A2FF}"
+ >
+ <Filter
+ Name="Common"
+ >
+ <File
+ RelativePath=".\cf_test_utils.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\cf_test_utils.h"
+ >
+ </File>
+ <File
+ RelativePath=".\chrometab_unittests.h"
+ >
+ </File>
+ <File
+ RelativePath=".\run_all_unittests.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\..\base\test_suite.h"
+ >
+ </File>
+ <File
+ RelativePath="..\test_utils.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\test_utils.h"
+ >
+ </File>
+ </Filter>
+ <Filter
+ Name="Tests"
+ >
+ <File
+ RelativePath="..\chrome_frame_automation.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\chrometab_unittests.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\com_message_event.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\com_message_event_unittest.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\function_stub_unittest.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\html_util_unittests.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\html_utils.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\icu_stubs_unittests.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\util_unittests.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\utils.cc"
+ >
+ </File>
+ </Filter>
+ </Filter>
+ <Filter
+ Name="Header Files"
+ Filter="h;hpp;hxx;hm;inl;inc;xsd"
+ UniqueIdentifier="{93995380-89BD-4b04-88EB-625FBE52EBFB}"
+ >
+ </Filter>
+ <Filter
+ Name="Resource Files"
+ Filter="rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav"
+ UniqueIdentifier="{67DA6AB6-F800-4c08-8B7A-83BB121AAD01}"
+ >
+ </Filter>
+ </Files>
+ <Globals>
+ </Globals>
+</VisualStudioProject>
diff --git a/chrome_frame/test/chrome_frame_automation_mock.cc b/chrome_frame/test/chrome_frame_automation_mock.cc
new file mode 100644
index 0000000..d900176
--- /dev/null
+++ b/chrome_frame/test/chrome_frame_automation_mock.cc
@@ -0,0 +1,47 @@
+// Copyright (c) 2006-2009 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 "chrome_frame/test/chrome_frame_automation_mock.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+const int kLongWaitTimeout = 25 * 1000;
+const int kShortWaitTimeout = 5 * 1000;
+
+TEST(ChromeFrame, Launch) {
+ MessageLoopForUI loop;
+ AutomationMockLaunch mock_launch(&loop, kLongWaitTimeout);
+
+ mock_launch.Navigate("about:blank");
+ loop.Run(NULL);
+ EXPECT_EQ(true, mock_launch.launch_result());
+}
+
+TEST(ChromeFrame, Navigate) {
+ MessageLoopForUI loop;
+ AutomationMockNavigate mock_navigate(&loop, kLongWaitTimeout);
+
+ mock_navigate.NavigateRelativeFile(L"postmessage_basic_frame.html");
+ loop.Run(NULL);
+ EXPECT_EQ(true, mock_navigate.navigation_result());
+}
+
+TEST(ChromeFrame, PostMessage) {
+ MessageLoopForUI loop;
+ AutomationMockPostMessage mock_postmessage(&loop, kLongWaitTimeout);
+
+ mock_postmessage.NavigateRelativeFile(L"postmessage_basic_frame.html");
+ loop.Run(NULL);
+ EXPECT_EQ(true, mock_postmessage.postmessage_result());
+}
+
+TEST(ChromeFrame, RequestStart) {
+ MessageLoopForUI loop;
+ AutomationMockHostNetworkRequestStart mock_request_start(&loop,
+ kLongWaitTimeout);
+
+ mock_request_start.NavigateRelative(L"postmessage_basic_frame.html");
+ loop.Run(NULL);
+ EXPECT_EQ(true, mock_request_start.request_start_result());
+}
+
diff --git a/chrome_frame/test/chrome_frame_automation_mock.h b/chrome_frame/test/chrome_frame_automation_mock.h
new file mode 100644
index 0000000..4c3fe7b
--- /dev/null
+++ b/chrome_frame/test/chrome_frame_automation_mock.h
@@ -0,0 +1,208 @@
+// Copyright (c) 2006-2009 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.
+#ifndef CHROME_FRAME_TEST_CHROME_FRAME_AUTOMATION_MOCK_H_
+#define CHROME_FRAME_TEST_CHROME_FRAME_AUTOMATION_MOCK_H_
+
+#include <string>
+
+#include "base/file_path.h"
+#include "base/path_service.h"
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/chrome_frame_plugin.h"
+#include "chrome_frame/test/http_server.h"
+
+template <typename T>
+class AutomationMockDelegate
+ : public CWindowImpl<T>,
+ public ChromeFramePlugin<T> {
+ public:
+ AutomationMockDelegate(MessageLoop* caller_message_loop,
+ int launch_timeout, bool perform_version_check,
+ const std::wstring& profile_name,
+ const std::wstring& extra_chrome_arguments, bool incognito)
+ : caller_message_loop_(caller_message_loop), is_connected_(false) {
+ test_server_.SetUp();
+ automation_client_.reset(new ChromeFrameAutomationClient);
+ automation_client_->Initialize(this, launch_timeout, perform_version_check,
+ profile_name, extra_chrome_arguments, incognito);
+ }
+ ~AutomationMockDelegate() {
+ if (automation_client_.get()) {
+ automation_client_->Uninitialize();
+ automation_client_.reset();
+ }
+ if (IsWindow())
+ DestroyWindow();
+
+ test_server_.TearDown();
+ }
+
+ // Navigate external tab to the specified url through automation
+ bool Navigate(const std::string& url) {
+ url_ = GURL(url);
+ return automation_client_->InitiateNavigation(url);
+ }
+
+ // Navigate the external to a 'file://' url for unit test files
+ bool NavigateRelativeFile(const std::wstring& file) {
+ FilePath cf_source_path;
+ PathService::Get(base::DIR_SOURCE_ROOT, &cf_source_path);
+ std::wstring file_url(L"file://");
+ file_url.append(cf_source_path.Append(
+ FILE_PATH_LITERAL("chrome_frame")).Append(
+ FILE_PATH_LITERAL("test")).Append(
+ FILE_PATH_LITERAL("data")).Append(file).value());
+ return Navigate(WideToUTF8(file_url));
+ }
+
+ bool NavigateRelative(const std::wstring& relative_url) {
+ return Navigate(test_server_.Resolve(relative_url.c_str()).spec());
+ }
+
+ virtual void OnAutomationServerReady() {
+ if (automation_client_.get()) {
+ Create(NULL, 0, NULL, WS_OVERLAPPEDWINDOW);
+ DCHECK(IsWindow());
+ is_connected_ = true;
+ } else {
+ NOTREACHED();
+ }
+ }
+
+ virtual void OnAutomationServerLaunchFailed() {
+ QuitMessageLoop();
+ }
+
+ virtual void OnLoad(int tab_handle, const GURL& url) {
+ if (url_ == url) {
+ navigation_result_ = true;
+ } else {
+ QuitMessageLoop();
+ }
+ }
+
+ virtual void OnLoadFailed(int error_code, const std::string& url) {
+ QuitMessageLoop();
+ }
+
+ ChromeFrameAutomationClient* automation() {
+ return automation_client_.get();
+ }
+ const GURL& url() const {
+ return url_;
+ }
+ bool is_connected() const {
+ return is_connected_;
+ }
+ bool navigation_result() const {
+ return navigation_result_;
+ }
+
+ BEGIN_MSG_MAP(AutomationMockDelegate)
+ END_MSG_MAP()
+
+ protected:
+ void QuitMessageLoop() {
+ // Quit on the caller message loop has to be called on the caller
+ // thread.
+ caller_message_loop_->PostTask(FROM_HERE, new MessageLoop::QuitTask);
+ }
+
+ private:
+ ChromeFrameHTTPServer test_server_;
+ MessageLoop* caller_message_loop_;
+ GURL url_;
+ bool is_connected_;
+ bool navigation_result_;
+};
+
+class AutomationMockLaunch
+ : public AutomationMockDelegate<AutomationMockLaunch> {
+ public:
+ typedef AutomationMockDelegate<AutomationMockLaunch> Base;
+ AutomationMockLaunch(MessageLoop* caller_message_loop,
+ int launch_timeout)
+ : Base(caller_message_loop, launch_timeout, true, L"", L"", false) {
+ }
+ virtual void OnAutomationServerReady() {
+ Base::OnAutomationServerReady();
+ QuitMessageLoop();
+ }
+ bool launch_result() const {
+ return is_connected();
+ }
+};
+
+class AutomationMockNavigate
+ : public AutomationMockDelegate<AutomationMockNavigate> {
+ public:
+ typedef AutomationMockDelegate<AutomationMockNavigate> Base;
+ AutomationMockNavigate(MessageLoop* caller_message_loop,
+ int launch_timeout)
+ : Base(caller_message_loop, launch_timeout, true, L"", L"", false) {
+ }
+ virtual void OnLoad(int tab_handle, const GURL& url) {
+ Base::OnLoad(tab_handle, url);
+ QuitMessageLoop();
+ }
+};
+
+class AutomationMockPostMessage
+ : public AutomationMockDelegate<AutomationMockPostMessage> {
+ public:
+ typedef AutomationMockDelegate<AutomationMockPostMessage> Base;
+ AutomationMockPostMessage(MessageLoop* caller_message_loop,
+ int launch_timeout)
+ : Base(caller_message_loop, launch_timeout, true, L"", L"", false),
+ postmessage_result_(false) {}
+ bool postmessage_result() const {
+ return postmessage_result_;
+ }
+ virtual void OnLoad(int tab_handle, const GURL& url) {
+ Base::OnLoad(tab_handle, url);
+ if (navigation_result()) {
+ automation()->ForwardMessageFromExternalHost("Test", "null", "*");
+ }
+ }
+ virtual void OnMessageFromChromeFrame(int tab_handle,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& target) {
+ postmessage_result_ = true;
+ QuitMessageLoop();
+ }
+ private:
+ bool postmessage_result_;
+};
+
+class AutomationMockHostNetworkRequestStart
+ : public AutomationMockDelegate<AutomationMockHostNetworkRequestStart> {
+ public:
+ typedef AutomationMockDelegate<AutomationMockHostNetworkRequestStart> Base;
+ AutomationMockHostNetworkRequestStart(MessageLoop* caller_message_loop,
+ int launch_timeout)
+ : Base(caller_message_loop, launch_timeout, true, L"", L"", false),
+ request_start_result_(false) {
+ if (automation()) {
+ automation()->set_use_chrome_network(false);
+ }
+ }
+ bool request_start_result() const {
+ return request_start_result_;
+ }
+ virtual void OnRequestStart(int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request) {
+ request_start_result_ = true;
+ QuitMessageLoop();
+ }
+ virtual void OnLoad(int tab_handle, const GURL& url) {
+ Base::OnLoad(tab_handle, url);
+ }
+ private:
+ bool request_start_result_;
+};
+
+
+#endif // CHROME_FRAME_TEST_CHROME_FRAME_AUTOMATION_MOCK_H_
+
diff --git a/chrome_frame/test/chrome_frame_test_utils.cc b/chrome_frame/test/chrome_frame_test_utils.cc
new file mode 100644
index 0000000..a75d791
--- /dev/null
+++ b/chrome_frame/test/chrome_frame_test_utils.cc
@@ -0,0 +1,416 @@
+// Copyright (c) 2006-2008 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 "chrome_frame/test/chrome_frame_test_utils.h"
+
+#include <atlbase.h>
+#include <atlwin.h>
+#include <iepmapi.h>
+
+#include "base/registry.h" // to find IE and firefox
+#include "base/scoped_handle.h"
+#include "base/scoped_comptr_win.h"
+#include "base/string_util.h"
+#include "base/win_util.h"
+#include "chrome/common/chrome_switches.h"
+
+namespace chrome_frame_test {
+
+const wchar_t kIEImageName[] = L"iexplore.exe";
+const wchar_t kIEBrokerImageName[] = L"ieuser.exe";
+const wchar_t kFirefoxImageName[] = L"firefox.exe";
+const wchar_t kOperaImageName[] = L"opera.exe";
+const wchar_t kSafariImageName[] = L"safari.exe";
+const wchar_t kChromeImageName[] = L"chrome.exe";
+
+bool IsTopLevelWindow(HWND window) {
+ long style = GetWindowLong(window, GWL_STYLE); // NOLINT
+ if (!(style & WS_CHILD))
+ return true;
+
+ HWND parent = GetParent(window);
+ if (!parent)
+ return true;
+
+ if (parent == GetDesktopWindow())
+ return true;
+
+ return false;
+}
+
+// Callback function for EnumThreadWindows.
+BOOL CALLBACK CloseWindowsThreadCallback(HWND hwnd, LPARAM param) {
+ int& count = *reinterpret_cast<int*>(param);
+ if (IsWindowVisible(hwnd)) {
+ if (IsWindowEnabled(hwnd)) {
+ DWORD results = 0;
+ if (!::SendMessageTimeout(hwnd, WM_SYSCOMMAND, SC_CLOSE, 0, SMTO_BLOCK,
+ 10000, &results)) {
+ DLOG(WARNING) << "Window hung: " << StringPrintf(L"%08X", hwnd);
+ }
+ count++;
+ } else {
+ DLOG(WARNING) << "Skipping disabled window: "
+ << StringPrintf(L"%08X", hwnd);
+ }
+ }
+ return TRUE; // continue enumeration
+}
+
+// Attempts to close all non-child, visible windows on the given thread.
+// The return value is the number of visible windows a close request was
+// sent to.
+int CloseVisibleTopLevelWindowsOnThread(DWORD thread_id) {
+ int window_close_attempts = 0;
+ EnumThreadWindows(thread_id, CloseWindowsThreadCallback,
+ reinterpret_cast<LPARAM>(&window_close_attempts));
+ return window_close_attempts;
+}
+
+// Enumerates the threads of a process and attempts to close visible non-child
+// windows on all threads of the process.
+// The return value is the number of visible windows a close request was
+// sent to.
+int CloseVisibleWindowsOnAllThreads(HANDLE process) {
+ DWORD process_id = ::GetProcessId(process);
+ if (process_id == 0) {
+ NOTREACHED();
+ return 0;
+ }
+
+ ScopedHandle snapshot(CreateToolhelp32Snapshot(TH32CS_SNAPTHREAD, 0));
+ if (!snapshot.IsValid()) {
+ NOTREACHED();
+ return 0;
+ }
+
+ int window_close_attempts = 0;
+ THREADENTRY32 te = { sizeof(THREADENTRY32) };
+ if (Thread32First(snapshot, &te)) {
+ do {
+ if (RTL_CONTAINS_FIELD(&te, te.dwSize, th32OwnerProcessID) &&
+ te.th32OwnerProcessID == process_id) {
+ window_close_attempts +=
+ CloseVisibleTopLevelWindowsOnThread(te.th32ThreadID);
+ }
+ te.dwSize = sizeof(te);
+ } while (Thread32Next(snapshot, &te));
+ }
+
+ return window_close_attempts;
+}
+
+class ForegroundHelperWindow : public CWindowImpl<ForegroundHelperWindow> {
+ public:
+BEGIN_MSG_MAP(ForegroundHelperWindow)
+ MESSAGE_HANDLER(WM_HOTKEY, OnHotKey)
+END_MSG_MAP()
+
+ HRESULT SetForeground(HWND window) {
+ DCHECK(::IsWindow(window));
+ if (NULL == Create(NULL, NULL, NULL, WS_POPUP))
+ return AtlHresultFromLastError();
+
+ static const int hotkey_id = 0x0000baba;
+
+ SetWindowLongPtr(GWLP_USERDATA, reinterpret_cast<ULONG_PTR>(window));
+ RegisterHotKey(m_hWnd, hotkey_id, 0, VK_F22);
+
+ MSG msg = {0};
+ PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE);
+
+ INPUT hotkey = {0};
+ hotkey.type = INPUT_KEYBOARD;
+ hotkey.ki.wVk = VK_F22;
+ SendInput(1, &hotkey, sizeof(hotkey));
+
+ while (GetMessage(&msg, NULL, 0, 0)) {
+ TranslateMessage(&msg);
+ DispatchMessage(&msg);
+ if (WM_HOTKEY == msg.message)
+ break;
+ }
+
+ UnregisterHotKey(m_hWnd, hotkey_id);
+ DestroyWindow();
+
+ return S_OK;
+ }
+
+ LRESULT OnHotKey(UINT msg, WPARAM wp, LPARAM lp, BOOL& handled) { // NOLINT
+ HWND window = reinterpret_cast<HWND>(GetWindowLongPtr(GWLP_USERDATA));
+ SetForegroundWindow(window);
+ return 1;
+ }
+};
+
+bool ForceSetForegroundWindow(HWND window) {
+ if (GetForegroundWindow() == window)
+ return true;
+ ForegroundHelperWindow foreground_helper_window;
+ HRESULT hr = foreground_helper_window.SetForeground(window);
+ return SUCCEEDED(hr);
+}
+
+struct PidAndWindow {
+ base::ProcessId pid;
+ HWND hwnd;
+};
+
+BOOL CALLBACK FindWindowInProcessCallback(HWND hwnd, LPARAM param) {
+ PidAndWindow* paw = reinterpret_cast<PidAndWindow*>(param);
+ base::ProcessId pid;
+ GetWindowThreadProcessId(hwnd, &pid);
+ if (pid == paw->pid && IsWindowVisible(hwnd)) {
+ paw->hwnd = hwnd;
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+bool EnsureProcessInForeground(base::ProcessId process_id) {
+ HWND hwnd = GetForegroundWindow();
+ base::ProcessId current_foreground_pid = 0;
+ DWORD active_thread_id = GetWindowThreadProcessId(hwnd,
+ &current_foreground_pid);
+ if (current_foreground_pid == process_id)
+ return true;
+
+ PidAndWindow paw = { process_id };
+ EnumWindows(FindWindowInProcessCallback, reinterpret_cast<LPARAM>(&paw));
+ if (!IsWindow(paw.hwnd)) {
+ DLOG(ERROR) << "failed to find process window";
+ return false;
+ }
+
+ bool ret = ForceSetForegroundWindow(paw.hwnd);
+ DLOG_IF(ERROR, !ret) << "ForceSetForegroundWindow: " << ret;
+
+ return ret;
+}
+
+// Iterates through all the characters in the string and simulates
+// keyboard input. The input goes to the currently active application.
+bool SendString(const wchar_t* string) {
+ DCHECK(string != NULL);
+
+ INPUT input[2] = {0};
+ input[0].type = INPUT_KEYBOARD;
+ input[0].ki.dwFlags = KEYEVENTF_UNICODE; // to avoid shift, etc.
+ input[1] = input[0];
+ input[1].ki.dwFlags |= KEYEVENTF_KEYUP;
+
+ for (const wchar_t* p = string; *p; p++) {
+ input[0].ki.wScan = input[1].ki.wScan = *p;
+ SendInput(2, input, sizeof(INPUT));
+ }
+
+ return true;
+}
+
+void SendVirtualKey(int16 key) {
+ INPUT input = { INPUT_KEYBOARD };
+ input.ki.wVk = key;
+ SendInput(1, &input, sizeof(input));
+ input.ki.dwFlags = KEYEVENTF_KEYUP;
+ SendInput(1, &input, sizeof(input));
+}
+
+void SendChar(char c) {
+ SendVirtualKey(VkKeyScanA(c));
+}
+
+void SendString(const char* s) {
+ while (*s) {
+ SendChar(*s);
+ s++;
+ }
+}
+
+// Sends a keystroke to the currently active application with optional
+// modifiers set.
+bool SendMnemonic(WORD mnemonic_char, bool shift_pressed, bool control_pressed,
+ bool alt_pressed) {
+ INPUT special_keys[3] = {0};
+ for (int index = 0; index < arraysize(special_keys); ++index) {
+ special_keys[index].type = INPUT_KEYBOARD;
+ special_keys[index].ki.dwFlags = 0;
+ }
+
+ int num_special_keys = 0;
+ if (shift_pressed) {
+ special_keys[num_special_keys].ki.wVk = VK_SHIFT;
+ num_special_keys++;
+ }
+
+ if (control_pressed) {
+ special_keys[num_special_keys].ki.wVk = VK_CONTROL;
+ num_special_keys++;
+ }
+
+ if (alt_pressed) {
+ special_keys[num_special_keys].ki.wVk = VK_MENU;
+ num_special_keys++;
+ }
+
+ // Depress the modifiers.
+ SendInput(num_special_keys, special_keys, sizeof(INPUT));
+
+ Sleep(100);
+
+ INPUT mnemonic = {0};
+ mnemonic.type = INPUT_KEYBOARD;
+ mnemonic.ki.wVk = mnemonic_char;
+
+ // Depress and release the mnemonic.
+ SendInput(1, &mnemonic, sizeof(INPUT));
+ Sleep(100);
+
+ mnemonic.ki.dwFlags |= KEYEVENTF_KEYUP;
+ SendInput(1, &mnemonic, sizeof(INPUT));
+ Sleep(100);
+
+ // Now release the modifiers.
+ for (int index = 0; index < num_special_keys; index++) {
+ special_keys[index].ki.dwFlags |= KEYEVENTF_KEYUP;
+ }
+
+ SendInput(num_special_keys, special_keys, sizeof(INPUT));
+ Sleep(100);
+
+ return true;
+}
+
+std::wstring GetExecutableAppPath(const std::wstring& file) {
+ std::wstring kAppPathsKey =
+ L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\App Paths\\";
+
+ std::wstring app_path;
+ RegKey key(HKEY_LOCAL_MACHINE, (kAppPathsKey + file).c_str());
+ if (key.Handle()) {
+ key.ReadValue(NULL, &app_path);
+ }
+
+ return app_path;
+}
+
+std::wstring FormatCommandForApp(const std::wstring& exe_name,
+ const std::wstring& argument) {
+ std::wstring reg_path(StringPrintf(L"Applications\\%ls\\shell\\open\\command",
+ exe_name.c_str()));
+ RegKey key(HKEY_CLASSES_ROOT, reg_path.c_str());
+
+ std::wstring command;
+ if (key.Handle()) {
+ key.ReadValue(NULL, &command);
+ int found = command.find(L"%1");
+ if (found >= 0) {
+ command.replace(found, 2, argument);
+ }
+ }
+ return command;
+}
+
+base::ProcessHandle LaunchExecutable(const std::wstring& executable,
+ const std::wstring& argument) {
+ base::ProcessHandle process = NULL;
+ std::wstring path = GetExecutableAppPath(executable);
+ if (path.empty()) {
+ path = FormatCommandForApp(executable, argument);
+ if (path.empty()) {
+ DLOG(ERROR) << "Failed to find executable: " << executable;
+ } else {
+ CommandLine cmdline(L"");
+ cmdline.ParseFromString(path);
+ base::LaunchApp(cmdline, false, false, &process);
+ }
+ } else {
+ CommandLine cmdline(path);
+ cmdline.AppendLooseValue(argument);
+ base::LaunchApp(cmdline, false, false, &process);
+ }
+ return process;
+}
+
+base::ProcessHandle LaunchFirefox(const std::wstring& url) {
+ return LaunchExecutable(kFirefoxImageName, url);
+}
+
+base::ProcessHandle LaunchSafari(const std::wstring& url) {
+ return LaunchExecutable(kSafariImageName, url);
+}
+
+base::ProcessHandle LaunchChrome(const std::wstring& url) {
+ return LaunchExecutable(kChromeImageName,
+ StringPrintf(L"--%ls ", switches::kNoFirstRun) + url);
+}
+
+base::ProcessHandle LaunchOpera(const std::wstring& url) {
+ // NOTE: For Opera tests to work it must be configured to start up with
+ // a blank page. There is an command line switch, -nosession, that's supposed
+ // to avoid opening up the previous session, but that switch is not working.
+ // TODO(tommi): Include a special ini file (opera6.ini) for opera and launch
+ // with our required settings. This file is by default stored here:
+ // "%USERPROFILE%\Application Data\Opera\Opera\profile\opera6.ini"
+ return LaunchExecutable(kOperaImageName, url);
+}
+
+base::ProcessHandle LaunchIEOnVista(const std::wstring& url) {
+ typedef HRESULT (WINAPI* IELaunchURLPtr)(
+ const wchar_t* url,
+ PROCESS_INFORMATION *pi,
+ VOID *info);
+
+ IELaunchURLPtr launch;
+ PROCESS_INFORMATION pi = {0};
+ IELAUNCHURLINFO info = {sizeof info, 0};
+ HMODULE h = LoadLibrary(L"ieframe.dll");
+ if (!h)
+ return NULL;
+ launch = reinterpret_cast<IELaunchURLPtr>(GetProcAddress(h, "IELaunchURL"));
+ HRESULT hr = launch(url.c_str(), &pi, &info);
+ FreeLibrary(h);
+ if (SUCCEEDED(hr))
+ CloseHandle(pi.hThread);
+ return pi.hProcess;
+}
+
+base::ProcessHandle LaunchIE(const std::wstring& url) {
+ if (win_util::GetWinVersion() >= win_util::WINVERSION_VISTA) {
+ return LaunchIEOnVista(url);
+ } else {
+ return LaunchExecutable(kIEImageName, url);
+ }
+}
+
+int CloseAllIEWindows() {
+ int ret = 0;
+
+ ScopedComPtr<IShellWindows> windows;
+ HRESULT hr = ::CoCreateInstance(__uuidof(ShellWindows), NULL, CLSCTX_ALL,
+ IID_IShellWindows, reinterpret_cast<void**>(windows.Receive()));
+ DCHECK(SUCCEEDED(hr));
+
+ if (SUCCEEDED(hr)) {
+ long count = 0; // NOLINT
+ windows->get_Count(&count);
+ VARIANT i = { VT_I4 };
+ for (i.lVal = 0; i.lVal < count; ++i.lVal) {
+ ScopedComPtr<IDispatch> folder;
+ windows->Item(i, folder.Receive());
+ if (folder != NULL) {
+ ScopedComPtr<IWebBrowser2> browser;
+ if (SUCCEEDED(browser.QueryFrom(folder))) {
+ browser->Quit();
+ ++ret;
+ }
+ }
+ }
+ }
+
+ return ret;
+}
+
+} // namespace chrome_frame_test
diff --git a/chrome_frame/test/chrome_frame_test_utils.h b/chrome_frame/test/chrome_frame_test_utils.h
new file mode 100644
index 0000000..95e0c9b
--- /dev/null
+++ b/chrome_frame/test/chrome_frame_test_utils.h
@@ -0,0 +1,61 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_TEST_CHROME_FRAME_TEST_UTILS_H_
+#define CHROME_FRAME_TEST_CHROME_FRAME_TEST_UTILS_H_
+
+#include <windows.h>
+
+#include "base/basictypes.h"
+#include "base/process_util.h"
+
+namespace chrome_frame_test {
+
+bool IsTopLevelWindow(HWND window);
+int CloseVisibleWindowsOnAllThreads(HANDLE process);
+bool ForceSetForegroundWindow(HWND window);
+bool EnsureProcessInForeground(base::ProcessId process_id);
+
+// Iterates through all the characters in the string and simulates
+// keyboard input. The input goes to the currently active application.
+bool SendString(const wchar_t* s);
+
+// Sends a virtual key such as VK_TAB, VK_RETURN or a character that has been
+// translated to a virtual key.
+void SendVirtualKey(int16 key);
+
+// Translates a single char to a virtual key and calls SendVirtualKey.
+void SendChar(char c);
+
+// Sends an ascii string, char by char (calls SendChar for each).
+void SendString(const char* s);
+
+// Sends a keystroke to the currently active application with optional
+// modifiers set.
+bool SendMnemonic(WORD mnemonic_char, bool shift_pressed, bool control_pressed,
+ bool alt_pressed);
+
+base::ProcessHandle LaunchFirefox(const std::wstring& url);
+base::ProcessHandle LaunchOpera(const std::wstring& url);
+base::ProcessHandle LaunchIE(const std::wstring& url);
+base::ProcessHandle LaunchSafari(const std::wstring& url);
+base::ProcessHandle LaunchChrome(const std::wstring& url);
+
+// Attempts to close all open IE windows.
+// The return value is the number of windows closed.
+// @note: this function requires COM to be initialized on the calling thread.
+// Since the caller might be running in either MTA or STA, the function does
+// not perform this initialization itself.
+int CloseAllIEWindows();
+
+extern const wchar_t kIEImageName[];
+extern const wchar_t kIEBrokerImageName[];
+extern const wchar_t kFirefoxImageName[];
+extern const wchar_t kOperaImageName[];
+extern const wchar_t kSafariImageName[];
+extern const wchar_t kChromeImageName[];
+
+} // namespace chrome_frame_test
+
+#endif // CHROME_FRAME_CHROMETAB_UNITTESTS_CF_TEST_UTILS_H_
diff --git a/chrome_frame/test/chrome_frame_unittests.cc b/chrome_frame/test/chrome_frame_unittests.cc
new file mode 100644
index 0000000..20826b1
--- /dev/null
+++ b/chrome_frame/test/chrome_frame_unittests.cc
@@ -0,0 +1,1510 @@
+// Copyright (c) 2006-2008 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 <windows.h>
+#include <stdarg.h>
+
+// IShellWindows includes. Unfortunately we can't keep these in
+// alphabetic order since exdisp will bark if some interfaces aren't fully
+// defined.
+#include <mshtml.h>
+#include <exdisp.h>
+
+#include "base/basictypes.h"
+#include "base/file_version_info.h"
+#include "base/file_util.h"
+#include "base/scoped_bstr_win.h"
+#include "base/scoped_comptr_win.h"
+#include "base/scoped_variant_win.h"
+#include "base/sys_info.h"
+#include "gmock/gmock.h"
+#include "net/url_request/url_request_unittest.h"
+#include "chrome_frame/test/chrome_frame_unittests.h"
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/chrome_frame_delegate.h"
+#include "chrome_frame/test/chrome_frame_test_utils.h"
+#include "chrome_frame/test/helper_gmock.h"
+#include "chrome_frame/test_utils.h"
+#include "chrome_frame/utils.h"
+#include "chrome_frame/vectored_handler-impl.h"
+#include "chrome/installer/util/install_util.h"
+#include "chrome/installer/util/helper.h"
+
+const wchar_t kDocRoot[] = L"chrome_frame\\test\\data";
+const int kLongWaitTimeout = 60 * 1000;
+const int kShortWaitTimeout = 25 * 1000;
+
+_ATL_FUNC_INFO WebBrowserEventSink::kNavigateErrorInfo = {
+ CC_STDCALL, VT_EMPTY, 5, {
+ VT_DISPATCH,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_BOOL | VT_BYREF,
+ }
+};
+
+_ATL_FUNC_INFO WebBrowserEventSink::kNavigateComplete2Info = {
+ CC_STDCALL, VT_EMPTY, 2, {
+ VT_DISPATCH,
+ VT_VARIANT | VT_BYREF
+ }
+};
+
+_ATL_FUNC_INFO WebBrowserEventSink::kBeforeNavigate2Info = {
+ CC_STDCALL, VT_EMPTY, 7, {
+ VT_DISPATCH,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_VARIANT | VT_BYREF,
+ VT_BOOL | VT_BYREF
+ }
+};
+
+
+
+void ChromeFrameTestWithWebServer::SetUp() {
+ server_.SetUp();
+ results_dir_ = server_.GetDataDir();
+ file_util::AppendToPath(&results_dir_, L"dump");
+}
+
+void ChromeFrameTestWithWebServer::TearDown() {
+ CloseBrowser();
+
+ // Web browsers tend to relaunch themselves in other processes, meaning the
+ // KillProcess stuff above might not have actually cleaned up all our browser
+ // instances, so make really sure browsers are dead.
+ base::KillProcesses(chrome_frame_test::kIEImageName, 0, NULL);
+ base::KillProcesses(chrome_frame_test::kIEBrokerImageName, 0, NULL);
+ base::KillProcesses(chrome_frame_test::kFirefoxImageName, 0, NULL);
+ base::KillProcesses(chrome_frame_test::kSafariImageName, 0, NULL);
+ base::KillProcesses(chrome_frame_test::kChromeImageName, 0, NULL);
+
+ server_.TearDown();
+}
+
+bool ChromeFrameTestWithWebServer::LaunchBrowser(BrowserKind browser,
+ const wchar_t* page) {
+ std::wstring url = UTF8ToWide(server_.Resolve(page).spec());
+ browser_ = browser;
+ if (browser == IE) {
+ browser_handle_.Set(chrome_frame_test::LaunchIE(url));
+ } else if (browser == FIREFOX) {
+ browser_handle_.Set(chrome_frame_test::LaunchFirefox(url));
+ } else if (browser == OPERA) {
+ browser_handle_.Set(chrome_frame_test::LaunchOpera(url));
+ } else if (browser == SAFARI) {
+ browser_handle_.Set(chrome_frame_test::LaunchSafari(url));
+ } else if (browser == CHROME) {
+ browser_handle_.Set(chrome_frame_test::LaunchChrome(url));
+ } else {
+ NOTREACHED();
+ }
+
+ return browser_handle_.IsValid();
+}
+
+void ChromeFrameTestWithWebServer::CloseBrowser() {
+ if (!browser_handle_.IsValid())
+ return;
+
+ int attempts = 0;
+ if (browser_ == IE) {
+ attempts = chrome_frame_test::CloseAllIEWindows();
+ } else {
+ attempts = chrome_frame_test::CloseVisibleWindowsOnAllThreads(
+ browser_handle_);
+ }
+
+ if (attempts > 0) {
+ DWORD wait = ::WaitForSingleObject(browser_handle_, 20000);
+ if (wait == WAIT_OBJECT_0) {
+ browser_handle_.Close();
+ } else {
+ DLOG(ERROR) << "WaitForSingleObject returned " << wait;
+ }
+ } else {
+ DLOG(ERROR) << "No attempts to close browser windows";
+ }
+
+ if (browser_handle_.IsValid()) {
+ DWORD exit_code = 0;
+ if (!::GetExitCodeProcess(browser_handle_, &exit_code) ||
+ exit_code == STILL_ACTIVE) {
+ DLOG(ERROR) << L"Forcefully killing browser process. Exit:" << exit_code;
+ base::KillProcess(browser_handle_, 0, true);
+ }
+ browser_handle_.Close();
+ }
+}
+
+bool ChromeFrameTestWithWebServer::BringBrowserToTop() {
+ return chrome_frame_test::EnsureProcessInForeground(GetProcessId(
+ browser_handle_));
+}
+
+bool ChromeFrameTestWithWebServer::WaitForTestToComplete(int milliseconds) {
+ return server_.WaitToFinish(milliseconds);
+}
+
+bool ChromeFrameTestWithWebServer::WaitForOnLoad(int milliseconds) {
+ DWORD start = ::GetTickCount();
+ std::string data;
+ while (!ReadResultFile(L"OnLoadEvent", &data) || data.length() == 0) {
+ DWORD now = ::GetTickCount();
+ if (start > now) {
+ // Very simple check for overflow. In that case we just restart the
+ // wait.
+ start = now;
+ } else if (static_cast<int>(now - start) > milliseconds) {
+ break;
+ }
+ Sleep(100);
+ }
+
+ return data.compare("loaded") == 0;
+}
+
+bool ChromeFrameTestWithWebServer::ReadResultFile(const std::wstring& file_name,
+ std::string* data) {
+ std::wstring full_path = results_dir_;
+ file_util::AppendToPath(&full_path, file_name);
+ return file_util::ReadFileToString(full_path, data);
+}
+
+bool ChromeFrameTestWithWebServer::CheckResultFile(
+ const std::wstring& file_name, const std::string& expected_result) {
+ std::string data;
+ bool ret = ReadResultFile(file_name, &data);
+ if (ret)
+ ret = (data == expected_result);
+
+ if (!ret) {
+ DLOG(ERROR) << "Error text: " << (data.empty() ? "<empty>" : data.c_str());
+ }
+
+ return ret;
+}
+
+void ChromeFrameTestWithWebServer::SimpleBrowserTest(BrowserKind browser,
+ const wchar_t* page, const wchar_t* result_file_to_check) {
+ EXPECT_TRUE(LaunchBrowser(browser, page));
+ ASSERT_TRUE(WaitForTestToComplete(kLongWaitTimeout));
+ ASSERT_TRUE(CheckResultFile(result_file_to_check, "OK"));
+}
+
+void ChromeFrameTestWithWebServer::OptionalBrowserTest(BrowserKind browser,
+ const wchar_t* page, const wchar_t* result_file_to_check) {
+ if (!LaunchBrowser(browser, page)) {
+ DLOG(ERROR) << "Failed to launch browser " << ToString(browser);
+ } else {
+ ASSERT_TRUE(WaitForTestToComplete(kLongWaitTimeout));
+ ASSERT_TRUE(CheckResultFile(result_file_to_check, "OK"));
+ }
+}
+
+void ChromeFrameTestWithWebServer::VersionTest(BrowserKind browser,
+ const wchar_t* page, const wchar_t* result_file_to_check) {
+ std::wstring plugin_path;
+ PathService::Get(base::DIR_MODULE, &plugin_path);
+ file_util::AppendToPath(&plugin_path, L"servers/npchrome_tab.dll");
+
+ static FileVersionInfo* version_info =
+ FileVersionInfo::CreateFileVersionInfo(plugin_path);
+
+ std::wstring version;
+ if (version_info)
+ version = version_info->product_version();
+
+ // If we can't find the npchrome_tab.dll in the src tree, we turn to
+ // the directory where chrome is installed.
+ if (!version_info) {
+ installer::Version* ver_system = InstallUtil::GetChromeVersion(true);
+ installer::Version* ver_user = InstallUtil::GetChromeVersion(false);
+ ASSERT_TRUE(ver_system || ver_user);
+
+ bool system_install = ver_system ? true : false;
+ std::wstring npchrome_path(installer::GetChromeInstallPath(system_install));
+ file_util::AppendToPath(&npchrome_path,
+ ver_system ? ver_system->GetString() : ver_user->GetString());
+ file_util::AppendToPath(&npchrome_path, L"npchrome_tab.dll");
+ version_info = FileVersionInfo::CreateFileVersionInfo(npchrome_path);
+ if (version_info)
+ version = version_info->product_version();
+ }
+
+ EXPECT_TRUE(version_info);
+ EXPECT_FALSE(version.empty());
+ EXPECT_TRUE(LaunchBrowser(browser, page));
+ ASSERT_TRUE(WaitForTestToComplete(kLongWaitTimeout));
+ ASSERT_TRUE(CheckResultFile(result_file_to_check, WideToUTF8(version)));
+}
+
+const wchar_t kPostMessageBasicPage[] = L"files/postmessage_basic_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_PostMessageBasic) {
+ SimpleBrowserTest(IE, kPostMessageBasicPage, L"PostMessage");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_PostMessageBasic) {
+ SimpleBrowserTest(FIREFOX, kPostMessageBasicPage, L"PostMessage");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_PostMessageBasic) {
+ OptionalBrowserTest(OPERA, kPostMessageBasicPage, L"PostMessage");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, FullTabIE_MIMEFilterBasic) {
+ const wchar_t kMIMEFilterBasicPage[] =
+ L"files/chrome_frame_mime_filter_test.html";
+
+ SimpleBrowserTest(IE, kMIMEFilterBasicPage, L"MIMEFilter");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_Resize) {
+ SimpleBrowserTest(IE, L"files/chrome_frame_resize.html", L"Resize");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_Resize) {
+ SimpleBrowserTest(FIREFOX, L"files/chrome_frame_resize.html", L"Resize");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_Resize) {
+ OptionalBrowserTest(OPERA, L"files/chrome_frame_resize.html", L"Resize");
+}
+
+const wchar_t kNavigateURLAbsolutePage[] =
+ L"files/navigateurl_absolute_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_NavigateURLAbsolute) {
+ SimpleBrowserTest(IE, kNavigateURLAbsolutePage, L"NavigateURL");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_NavigateURLAbsolute) {
+ SimpleBrowserTest(FIREFOX, kNavigateURLAbsolutePage, L"NavigateURL");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_NavigateURLAbsolute) {
+ OptionalBrowserTest(OPERA, kNavigateURLAbsolutePage, L"NavigateURL");
+}
+
+const wchar_t kNavigateURLRelativePage[] =
+ L"files/navigateurl_relative_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_NavigateURLRelative) {
+ SimpleBrowserTest(IE, kNavigateURLRelativePage, L"NavigateURL");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_NavigateURLRelative) {
+ SimpleBrowserTest(FIREFOX, kNavigateURLRelativePage, L"NavigateURL");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_NavigateURLRelative) {
+ OptionalBrowserTest(OPERA, kNavigateURLRelativePage, L"NavigateURL");
+}
+
+const wchar_t kNavigateSimpleObjectFocus[] = L"files/simple_object_focus.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_ObjectFocus) {
+ SimpleBrowserTest(FIREFOX, kNavigateSimpleObjectFocus, L"ObjectFocus");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_ObjectFocus) {
+ SimpleBrowserTest(IE, kNavigateSimpleObjectFocus, L"ObjectFocus");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_ObjectFocus) {
+ if (!LaunchBrowser(OPERA, kNavigateSimpleObjectFocus)) {
+ DLOG(ERROR) << "Failed to launch browser " << ToString(OPERA);
+ } else {
+ ASSERT_TRUE(WaitForOnLoad(kLongWaitTimeout));
+ BringBrowserToTop();
+ // Tab through a couple of times. Once should be enough in theory
+ // but in practice activating the browser can take a few milliseconds more.
+ bool ok;
+ for (int i = 0;
+ i < 5 && (ok = CheckResultFile(L"ObjectFocus", "OK")) == false;
+ ++i) {
+ Sleep(300);
+ chrome_frame_test::SendMnemonic(VK_TAB, false, false, false);
+ }
+ ASSERT_TRUE(ok);
+ }
+}
+
+const wchar_t kiframeBasicPage[] = L"files/iframe_basic_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_iframeBasic) {
+ SimpleBrowserTest(IE, kiframeBasicPage, L"PostMessage");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_iframeBasic) {
+ SimpleBrowserTest(FIREFOX, kiframeBasicPage, L"PostMessage");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_iframeBasic) {
+ OptionalBrowserTest(OPERA, kiframeBasicPage, L"PostMessage");
+}
+
+const wchar_t kSrcPropertyTestPage[] = L"files/src_property_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_SrcProperty) {
+ SimpleBrowserTest(IE, kSrcPropertyTestPage, L"SrcProperty");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_SrcProperty) {
+ SimpleBrowserTest(FIREFOX, kSrcPropertyTestPage, L"SrcProperty");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_SrcProperty) {
+ OptionalBrowserTest(OPERA, kSrcPropertyTestPage, L"SrcProperty");
+}
+
+const wchar_t kCFInstanceBasicTestPage[] = L"files/CFInstance_basic_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceBasic) {
+ SimpleBrowserTest(IE, kCFInstanceBasicTestPage, L"CFInstanceBasic");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceBasic) {
+ SimpleBrowserTest(FIREFOX, kCFInstanceBasicTestPage, L"CFInstanceBasic");
+}
+
+const wchar_t kCFISingletonPage[] = L"files/CFInstance_singleton_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceSingleton) {
+ SimpleBrowserTest(IE, kCFISingletonPage, L"CFInstanceSingleton");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceSingleton) {
+ SimpleBrowserTest(FIREFOX, kCFISingletonPage, L"CFInstanceSingleton");
+}
+
+const wchar_t kCFIDelayPage[] = L"files/CFInstance_delay_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeIE_CFInstanceDelay) {
+ SimpleBrowserTest(IE, kCFIDelayPage, L"CFInstanceDelay");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeFF_CFInstanceDelay) {
+ SimpleBrowserTest(FIREFOX, kCFIDelayPage, L"CFInstanceDelay");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeOpera_CFInstanceDelay) {
+ OptionalBrowserTest(OPERA, kCFIDelayPage, L"CFInstanceDelay");
+}
+
+const wchar_t kCFIFallbackPage[] = L"files/CFInstance_fallback_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceFallback) {
+ SimpleBrowserTest(IE, kCFIFallbackPage, L"CFInstanceFallback");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceFallback) {
+ SimpleBrowserTest(FIREFOX, kCFIFallbackPage, L"CFInstanceFallback");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_CFInstanceFallback) {
+ OptionalBrowserTest(OPERA, kCFIFallbackPage, L"CFInstanceFallback");
+}
+
+const wchar_t kCFINoSrcPage[] = L"files/CFInstance_no_src_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceNoSrc) {
+ SimpleBrowserTest(IE, kCFINoSrcPage, L"CFInstanceNoSrc");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceNoSrc) {
+ SimpleBrowserTest(FIREFOX, kCFINoSrcPage, L"CFInstanceNoSrc");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_CFInstanceNoSrc) {
+ OptionalBrowserTest(OPERA, kCFINoSrcPage, L"CFInstanceNoSrc");
+}
+
+const wchar_t kCFIIfrOnLoadPage[] = L"files/CFInstance_iframe_onload_host.html";
+
+// disabled since it's unlikely that we care about this case
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeIE_CFInstanceIfrOnLoad) {
+ SimpleBrowserTest(IE, kCFIIfrOnLoadPage, L"CFInstanceIfrOnLoad");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceIfrOnLoad) {
+ SimpleBrowserTest(FIREFOX, kCFIIfrOnLoadPage, L"CFInstanceIfrOnLoad");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_CFInstanceIfrOnLoad) {
+ OptionalBrowserTest(OPERA, kCFIIfrOnLoadPage, L"CFInstanceIfrOnLoad");
+}
+
+const wchar_t kCFIZeroSizePage[] = L"files/CFInstance_zero_size_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceZeroSize) {
+ SimpleBrowserTest(IE, kCFIZeroSizePage, L"CFInstanceZeroSize");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceZeroSize) {
+ SimpleBrowserTest(FIREFOX, kCFIZeroSizePage, L"CFInstanceZeroSize");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_CFInstanceZeroSize) {
+ OptionalBrowserTest(OPERA, kCFIZeroSizePage, L"CFInstanceZeroSize");
+}
+
+const wchar_t kCFIIfrPostPage[] = L"files/CFInstance_iframe_post_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceIfrPost) {
+ SimpleBrowserTest(IE, kCFIIfrPostPage, L"CFInstanceIfrPost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceIfrPost) {
+ SimpleBrowserTest(FIREFOX, kCFIIfrPostPage, L"CFInstanceIfrPost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeChrome_CFInstanceIfrPost) {
+ OptionalBrowserTest(CHROME, kCFIIfrPostPage, L"CFInstanceIfrPost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeSafari_CFInstanceIfrPost) {
+ OptionalBrowserTest(SAFARI, kCFIIfrPostPage, L"CFInstanceIfrPost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_CFInstanceIfrPost) {
+ OptionalBrowserTest(OPERA, kCFIIfrPostPage, L"CFInstanceIfrPost");
+}
+
+const wchar_t kCFIPostPage[] = L"files/CFInstance_post_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstancePost) {
+ SimpleBrowserTest(IE, kCFIPostPage, L"CFInstancePost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstancePost) {
+ SimpleBrowserTest(FIREFOX, kCFIPostPage, L"CFInstancePost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeChrome_CFInstancePost) {
+ OptionalBrowserTest(CHROME, kCFIPostPage, L"CFInstancePost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeSafari_CFInstancePost) {
+ OptionalBrowserTest(SAFARI, kCFIPostPage, L"CFInstancePost");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeOpera_CFInstancePost) {
+ OptionalBrowserTest(OPERA, kCFIPostPage, L"CFInstancePost");
+}
+
+const wchar_t kCFIRPCPage[] = L"files/CFInstance_rpc_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceRPC) {
+ SimpleBrowserTest(IE, kCFIRPCPage, L"CFInstanceRPC");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceRPC) {
+ SimpleBrowserTest(FIREFOX, kCFIRPCPage, L"CFInstanceRPC");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeChrome_CFInstanceRPC) {
+ OptionalBrowserTest(CHROME, kCFIRPCPage, L"CFInstanceRPC");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeSafari_CFInstanceRPC) {
+ OptionalBrowserTest(SAFARI, kCFIRPCPage, L"CFInstanceRPC");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeOpera_CFInstanceRPC) {
+ OptionalBrowserTest(OPERA, kCFIRPCPage, L"CFInstanceRPC");
+}
+
+const wchar_t kCFIRPCInternalPage[] =
+ L"files/CFInstance_rpc_internal_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceRPCInternal) {
+ SimpleBrowserTest(IE, kCFIRPCInternalPage, L"CFInstanceRPCInternal");
+}
+
+// Disabled: http://b/issue?id=2050201
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeFF_CFInstanceRPCInternal) {
+ SimpleBrowserTest(FIREFOX, kCFIRPCInternalPage, L"CFInstanceRPCInternal");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeChrome_CFInstanceRPCInternal) {
+ OptionalBrowserTest(CHROME, kCFIRPCInternalPage, L"CFInstanceRPCInternal");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeSafari_CFInstanceRPCInternal) {
+ OptionalBrowserTest(SAFARI, kCFIRPCInternalPage, L"CFInstanceRPCInternal");
+}
+
+const wchar_t kCFIDefaultCtorPage[] =
+ L"files/CFInstance_default_ctor_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_CFInstanceDefaultCtor) {
+ SimpleBrowserTest(IE, kCFIDefaultCtorPage, L"CFInstanceDefaultCtor");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_CFInstanceDefaultCtor) {
+ SimpleBrowserTest(FIREFOX, kCFIDefaultCtorPage, L"CFInstanceDefaultCtor");
+}
+
+// Class that mocks external call from VectoredHandlerT for testing purposes.
+class EMock : public VEHTraitsBase {
+ public:
+ static inline bool WriteDump(EXCEPTION_POINTERS* p) {
+ g_dump_made = true;
+ return true;
+ }
+
+ static inline void* Register(PVECTORED_EXCEPTION_HANDLER func,
+ const void* module_start,
+ const void* module_end) {
+ VEHTraitsBase::SetModule(module_start, module_end);
+ // Return some arbitrary number, expecting to get the same on Unregister()
+ return reinterpret_cast<void*>(4);
+ }
+
+ static inline ULONG Unregister(void* handle) {
+ EXPECT_EQ(handle, reinterpret_cast<void*>(4));
+ return 1;
+ }
+
+ static inline WORD RtlCaptureStackBackTrace(DWORD FramesToSkip,
+ DWORD FramesToCapture, void** BackTrace, DWORD* BackTraceHash) {
+ EXPECT_EQ(2, FramesToSkip);
+ EXPECT_LE(FramesToSkip + FramesToCapture,
+ VectoredHandlerBase::max_back_trace);
+ memcpy(BackTrace, g_stack, g_stack_entries * sizeof(BackTrace[0]));
+ return g_stack_entries;
+ }
+
+ static inline EXCEPTION_REGISTRATION_RECORD* RtlpGetExceptionList() {
+ return g_seh_chain;
+ }
+
+ // Test helpers
+
+ // Create fake SEH chain of random filters - with and without our module.
+ static void SetHaveSEHFilter() {
+ SetSEHChain(reinterpret_cast<const char*>(g_module_start) - 0x1000,
+ reinterpret_cast<const char*>(g_module_start) + 0x1000,
+ reinterpret_cast<const char*>(g_module_end) + 0x7127,
+ NULL);
+ }
+
+ static void SetNoSEHFilter() {
+ SetSEHChain(reinterpret_cast<const char*>(g_module_start) - 0x1000,
+ reinterpret_cast<const char*>(g_module_end) + 0x7127,
+ NULL);
+ }
+
+ // Create fake stack - with and without our module.
+ static void SetOnStack() {
+ SetStack(reinterpret_cast<const char*>(g_module_start) - 0x11283,
+ reinterpret_cast<const char*>(g_module_start) - 0x278361,
+ reinterpret_cast<const char*>(g_module_start) + 0x9171,
+ reinterpret_cast<const char*>(g_module_end) + 1231,
+ NULL);
+ }
+
+ static void SetNotOnStack() {
+ SetStack(reinterpret_cast<const char*>(g_module_start) - 0x11283,
+ reinterpret_cast<const char*>(g_module_start) - 0x278361,
+ reinterpret_cast<const char*>(g_module_end) + 1231,
+ NULL);
+ }
+
+ // Populate stack array
+ static void SetStack(const void* p, ...) {
+ va_list vl;
+ va_start(vl, p);
+ g_stack_entries = 0;
+ for (; p; ++g_stack_entries) {
+ CHECK(g_stack_entries < arraysize(g_stack));
+ g_stack[g_stack_entries] = p;
+ p = va_arg(vl, const void*);
+ }
+ }
+
+ static void SetSEHChain(const void* p, ...) {
+ va_list vl;
+ va_start(vl, p);
+ int i = 0;
+ for (; p; ++i) {
+ CHECK(i + 1 < arraysize(g_seh_chain));
+ g_seh_chain[i].Handler = const_cast<void*>(p);
+ g_seh_chain[i].Next = &g_seh_chain[i + 1];
+ p = va_arg(vl, const void*);
+ }
+
+ g_seh_chain[i].Next = EXCEPTION_CHAIN_END;
+ }
+
+ static EXCEPTION_REGISTRATION_RECORD g_seh_chain[25];
+ static const void* g_stack[VectoredHandlerBase::max_back_trace];
+ static WORD g_stack_entries;
+ static bool g_dump_made;
+};
+
+EXCEPTION_REGISTRATION_RECORD EMock::g_seh_chain[25];
+const void* EMock::g_stack[VectoredHandlerBase::max_back_trace];
+WORD EMock::g_stack_entries;
+bool EMock::g_dump_made;
+
+typedef VectoredHandlerT<EMock> VectoredHandlerMock;
+
+class ExPtrsHelper : public _EXCEPTION_POINTERS {
+ public:
+ ExPtrsHelper() {
+ ExceptionRecord = &er_;
+ ContextRecord = &ctx_;
+ ZeroMemory(&er_, sizeof(er_));
+ ZeroMemory(&ctx_, sizeof(ctx_));
+ }
+
+ void Set(DWORD code, void* address, DWORD flags) {
+ er_.ExceptionCode = code;
+ er_.ExceptionAddress = address;
+ er_.ExceptionFlags = flags;
+ }
+
+ EXCEPTION_RECORD er_;
+ CONTEXT ctx_;
+};
+
+
+TEST(ChromeFrame, ExceptionReport) {
+ char* s = reinterpret_cast<char*>(0x30000000);
+ char* e = s + 0x10000;
+ void* handler = VectoredHandlerMock::Register(s, e);
+ char* our_code = s + 0x1111;
+ char* not_our_code = s - 0x5555;
+ char* not_our_code2 = e + 0x5555;
+
+ ExPtrsHelper ex;
+ // Exception in our code, but we have SEH filter
+ ex.Set(STATUS_ACCESS_VIOLATION, our_code, 0);
+ EMock::SetHaveSEHFilter();
+ EMock::SetOnStack();
+ EXPECT_EQ(ExceptionContinueSearch, VectoredHandlerMock::VectoredHandler(&ex));
+ EXPECT_EQ(1, VectoredHandlerMock::g_exceptions_seen);
+ EXPECT_FALSE(EMock::g_dump_made);
+
+ // RPC_E_DISCONNECTED (0x80010108) is "The object invoked has disconnected
+ // from its clients", shall not be caught since it's a warning only.
+ ex.Set(RPC_E_DISCONNECTED, our_code, 0);
+ EMock::SetHaveSEHFilter();
+ EMock::SetOnStack();
+ EXPECT_EQ(ExceptionContinueSearch, VectoredHandlerMock::VectoredHandler(&ex));
+ EXPECT_EQ(1, VectoredHandlerMock::g_exceptions_seen);
+ EXPECT_FALSE(EMock::g_dump_made);
+
+
+ // Exception, not in our code, we do not have SEH and we are not in stack.
+ ex.Set(STATUS_INTEGER_DIVIDE_BY_ZERO, not_our_code, 0);
+ EMock::SetNoSEHFilter();
+ EMock::SetNotOnStack();
+ EXPECT_EQ(ExceptionContinueSearch, VectoredHandlerMock::VectoredHandler(&ex));
+ EXPECT_EQ(2, VectoredHandlerMock::g_exceptions_seen);
+ EXPECT_FALSE(EMock::g_dump_made);
+
+ // Exception, not in our code, no SEH, but we are on the stack.
+ ex.Set(STATUS_INTEGER_DIVIDE_BY_ZERO, not_our_code2, 0);
+ EMock::SetNoSEHFilter();
+ EMock::SetOnStack();
+ EXPECT_EQ(ExceptionContinueSearch, VectoredHandlerMock::VectoredHandler(&ex));
+ EXPECT_EQ(3, VectoredHandlerMock::g_exceptions_seen);
+ EXPECT_TRUE(EMock::g_dump_made);
+ EMock::g_dump_made = false;
+
+
+ // Exception, in our code, no SEH, not on stack (assume FPO screwed us)
+ ex.Set(STATUS_INTEGER_DIVIDE_BY_ZERO, our_code, 0);
+ EMock::SetNoSEHFilter();
+ EMock::SetNotOnStack();
+ EXPECT_EQ(ExceptionContinueSearch, VectoredHandlerMock::VectoredHandler(&ex));
+ EXPECT_EQ(4, VectoredHandlerMock::g_exceptions_seen);
+ EXPECT_TRUE(EMock::g_dump_made);
+ EMock::g_dump_made = false;
+
+ VectoredHandlerMock::Unregister();
+}
+
+const wchar_t kInitializeHiddenPage[] = L"files/initialize_hidden.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_InitializeHidden) {
+ SimpleBrowserTest(IE, kInitializeHiddenPage, L"InitializeHidden");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_InitializeHidden) {
+ SimpleBrowserTest(FIREFOX, kInitializeHiddenPage, L"InitializeHidden");
+}
+
+// Disabled due to a problem with Opera.
+// http://b/issue?id=1708275
+TEST_F(ChromeFrameTestWithWebServer, DISABLED_WidgetModeOpera_InitializeHidden) {
+ OptionalBrowserTest(OPERA, kInitializeHiddenPage, L"InitializeHidden");
+}
+
+const wchar_t kInHeadPage[] = L"files/in_head.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_InHead) {
+ SimpleBrowserTest(FIREFOX, kInHeadPage, L"InHead");
+}
+
+const wchar_t kVersionPage[] = L"files/version.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_Version) {
+ VersionTest(IE, kVersionPage, L"version");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_Version) {
+ VersionTest(FIREFOX, kVersionPage, L"version");
+}
+
+const wchar_t kEventListenerPage[] = L"files/event_listener.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_EventListener) {
+ SimpleBrowserTest(IE, kEventListenerPage, L"EventListener");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_EventListener) {
+ SimpleBrowserTest(FIREFOX, kEventListenerPage, L"EventListener");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_EventListener) {
+ OptionalBrowserTest(OPERA, kEventListenerPage, L"EventListener");
+}
+
+const wchar_t kPrivilegedApisPage[] = L"files/privileged_apis_host.html";
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeIE_PrivilegedApis) {
+ SimpleBrowserTest(IE, kPrivilegedApisPage, L"PrivilegedApis");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeFF_PrivilegedApis) {
+ SimpleBrowserTest(FIREFOX, kPrivilegedApisPage, L"PrivilegedApis");
+}
+
+TEST_F(ChromeFrameTestWithWebServer, WidgetModeOpera_PrivilegedApis) {
+ OptionalBrowserTest(OPERA, kPrivilegedApisPage, L"PrivilegedApis");
+}
+
+class ChromeFrameTestEnvironment: public testing::Environment {
+ public:
+ ~ChromeFrameTestEnvironment() {
+ }
+
+ void SetUp() {
+ ScopedChromeFrameRegistrar::RegisterDefaults();
+ }
+
+ void TearDown() {
+ }
+};
+
+::testing::Environment* const chrome_frame_env =
+ ::testing::AddGlobalTestEnvironment(new ChromeFrameTestEnvironment);
+
+// TODO(stoyan): - Move everything below in separate file(s).
+struct LaunchDelegateMock : public ProxyFactory::LaunchDelegate {
+ MOCK_METHOD2(LaunchComplete, void(ChromeFrameAutomationProxy*,
+ AutomationLaunchResult));
+};
+
+TEST(ProxyFactoryTest, CreateDestroy) {
+ ProxyFactory f;
+ LaunchDelegateMock d;
+ EXPECT_CALL(d, LaunchComplete(testing::NotNull(), testing::_)).Times(1);
+ void* id = f.GetAutomationServer(0, L"Adam.N.Epilinter", L"", false, &d);
+ f.ReleaseAutomationServer(id);
+}
+
+TEST(ProxyFactoryTest, CreateSameProfile) {
+ ProxyFactory f;
+ LaunchDelegateMock d;
+ EXPECT_CALL(d, LaunchComplete(testing::NotNull(), testing::_)).Times(2);
+ void* i1 = f.GetAutomationServer(0, L"Dr. Gratiano Forbeson", L"", false, &d);
+ void* i2 = f.GetAutomationServer(0, L"Dr. Gratiano Forbeson", L"", false, &d);
+ EXPECT_EQ(i1, i2);
+ f.ReleaseAutomationServer(i2);
+ f.ReleaseAutomationServer(i1);
+}
+
+TEST(ProxyFactoryTest, CreateDifferentProfiles) {
+ ProxyFactory f;
+ LaunchDelegateMock d;
+ EXPECT_CALL(d, LaunchComplete(testing::NotNull(), testing::_)).Times(2);
+ void* i1 = f.GetAutomationServer(0, L"Adam.N.Epilinter", L"", false, &d);
+ void* i2 = f.GetAutomationServer(0, L"Dr. Gratiano Forbeson", L"", false, &d);
+ EXPECT_NE(i1, i2);
+ f.ReleaseAutomationServer(i2);
+ f.ReleaseAutomationServer(i1);
+}
+
+// ChromeFrameAutomationClient [CFAC] tests.
+struct MockCFDelegate : public ChromeFrameDelegateImpl {
+ MOCK_CONST_METHOD0(GetWindow, WindowType());
+ MOCK_METHOD1(GetBounds, void(RECT* bounds));
+ MOCK_METHOD0(GetDocumentUrl, std::string());
+ MOCK_METHOD2(ExecuteScript, bool(const std::string& script,
+ std::string* result));
+ MOCK_METHOD0(OnAutomationServerReady, void());
+ MOCK_METHOD2(OnAutomationServerLaunchFailed, void(
+ AutomationLaunchResult reason, const std::string& server_version));
+ // This remains in interface since we call it if Navigate()
+ // returns immediate error.
+ MOCK_METHOD2(OnLoadFailed, void(int error_code, const std::string& url));
+
+ // Do not mock this method. :) Use it as message demuxer and dispatcher
+ // to the following methods (which we mock)
+ // MOCK_METHOD1(OnMessageReceived, void(const IPC::Message&));
+
+
+ MOCK_METHOD2(OnNavigationStateChanged, void(int tab_handle, int flags));
+ MOCK_METHOD2(OnUpdateTargetUrl, void(int tab_handle,
+ const std::wstring& new_target_url));
+ MOCK_METHOD2(OnAcceleratorPressed, void(int tab_handle,
+ const MSG& accel_message));
+ MOCK_METHOD2(OnTabbedOut, void(int tab_handle, bool reverse));
+ MOCK_METHOD3(OnOpenURL, void(int tab_handle, const GURL& url,
+ int open_disposition));
+ MOCK_METHOD2(OnDidNavigate, void(int tab_handle,
+ const IPC::NavigationInfo& navigation_info));
+ MOCK_METHOD3(OnNavigationFailed, void(int tab_handle, int error_code,
+ const GURL& gurl));
+ MOCK_METHOD2(OnLoad, void(int tab_handle, const GURL& url));
+ MOCK_METHOD4(OnMessageFromChromeFrame, void(int tab_handle,
+ const std::string& message,
+ const std::string& origin,
+ const std::string& target));
+ MOCK_METHOD5(OnHandleContextMenu, void(int tab_handle, HANDLE menu_handle,
+ int x_pos, int y_pos, int align_flags));
+ MOCK_METHOD3(OnRequestStart, void(int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request));
+ MOCK_METHOD3(OnRequestRead, void(int tab_handle, int request_id,
+ int bytes_to_read));
+ MOCK_METHOD3(OnRequestEnd, void(int tab_handle, int request_id,
+ const URLRequestStatus& status));
+ MOCK_METHOD3(OnSetCookieAsync, void(int tab_handle, const GURL& url,
+ const std::string& cookie));
+
+ // Use for sending network responses
+ void SetAutomationSender(IPC::Message::Sender* automation) {
+ automation_ = automation;
+ }
+
+ // Set-expectation helpers
+ void SetOnNavigationStateChanged(int tab_handle) {
+ EXPECT_CALL(*this,
+ OnNavigationStateChanged(testing::Eq(tab_handle), testing::_))
+ .Times(testing::AnyNumber());
+ }
+
+ // Response sender helpers
+ void ReplyStarted(const IPC::AutomationURLResponse* response,
+ int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request) {
+ automation_->Send(new AutomationMsg_RequestStarted(0, tab_handle,
+ request_id, *response));
+ }
+
+ void ReplyData(const std::string* data, int tab_handle, int request_id,
+ int bytes_to_read) {
+ automation_->Send(new AutomationMsg_RequestData(0, tab_handle,
+ request_id, *data));
+ }
+
+ void ReplyEOF(int tab_handle, int request_id) {
+ automation_->Send(new AutomationMsg_RequestEnd(0, tab_handle,
+ request_id, URLRequestStatus()));
+ }
+
+ void Reply404(int tab_handle, int request_id,
+ const IPC::AutomationURLRequest& request) {
+ const IPC::AutomationURLResponse notfound = {"", "HTTP/1.1 404\r\n\r\n"};
+ automation_->Send(new AutomationMsg_RequestStarted(0, tab_handle,
+ request_id, notfound));
+ automation_->Send(new AutomationMsg_RequestEnd(0, tab_handle,
+ request_id, URLRequestStatus()));
+ }
+
+ IPC::Message::Sender* automation_;
+};
+
+class MockProxyFactory : public ProxyFactory {
+ public:
+ MOCK_METHOD5(GetAutomationServer, void*(int, const std::wstring&,
+ const std::wstring& extra_argument, bool, ProxyFactory::LaunchDelegate*));
+ MOCK_METHOD1(ReleaseAutomationServer, bool(void* id));
+
+ MockProxyFactory() : thread_("mock factory worker") {
+ thread_.Start();
+ loop_ = thread_.message_loop();
+ }
+
+ // Fake implementation
+ void GetServerImpl(ChromeFrameAutomationProxy* pxy,
+ AutomationLaunchResult result,
+ int timeout,
+ ProxyFactory::LaunchDelegate* d) {
+ Task* task = NewRunnableMethod(d,
+ &ProxyFactory::LaunchDelegate::LaunchComplete, pxy, result);
+ loop_->PostDelayedTask(FROM_HERE, task, timeout/2);
+ }
+
+ base::Thread thread_;
+ MessageLoop* loop_;
+};
+
+class MockAutomationProxy : public ChromeFrameAutomationProxy {
+ public:
+ MOCK_METHOD1(Send, bool(IPC::Message*));
+ MOCK_METHOD3(SendAsAsync, void(IPC::SyncMessage* msg, void* callback,
+ void* key));
+ MOCK_METHOD1(CancelAsync, void(void* key));
+ MOCK_METHOD1(CreateTabProxy, scoped_refptr<TabProxy>(int handle));
+ MOCK_METHOD0(server_version, std::string(void));
+ MOCK_METHOD1(SendProxyConfig, void(const std::string&));
+ MOCK_METHOD1(SetEnableExtensionAutomation, void(bool enable));
+
+ ~MockAutomationProxy() {}
+};
+
+struct MockAutomationMessageSender : public AutomationMessageSender {
+ MOCK_METHOD1(Send, bool(IPC::Message*));
+ MOCK_METHOD3(SendWithTimeout, bool(IPC::Message* , int , bool*));
+
+ void ForwardTo(MockAutomationProxy *p) {
+ proxy_ = p;
+ ON_CALL(*this, Send(testing::_))
+ .WillByDefault(testing::Invoke(proxy_, &MockAutomationProxy::Send));
+ }
+
+ MockAutomationProxy* proxy_;
+};
+
+template <> struct RunnableMethodTraits<ProxyFactory::LaunchDelegate> {
+ static void RetainCallee(ProxyFactory::LaunchDelegate* obj) {}
+ static void ReleaseCallee(ProxyFactory::LaunchDelegate* obj) {}
+};
+
+template <> struct RunnableMethodTraits<MockProxyFactory> {
+ static void RetainCallee(MockProxyFactory* obj) {}
+ static void ReleaseCallee(MockProxyFactory* obj) {}
+};
+
+template <> struct RunnableMethodTraits<ChromeFrameAutomationClient> {
+ static void RetainCallee(ChromeFrameAutomationClient* obj) {}
+ static void ReleaseCallee(ChromeFrameAutomationClient* obj) {}
+};
+
+// MessageLoopForUI wrapper that runs only for a limited time.
+// We need a UI message loop in the main thread.
+struct TimedMsgLoop {
+ public:
+ void RunFor(int seconds) {
+ loop_.PostDelayedTask(FROM_HERE, new MessageLoop::QuitTask, 1000 * seconds);
+ loop_.MessageLoop::Run();
+ }
+
+ void Quit() {
+ loop_.PostTask(FROM_HERE, new MessageLoop::QuitTask);
+ }
+
+ MessageLoopForUI loop_;
+};
+
+template <> struct RunnableMethodTraits<TimedMsgLoop> {
+ static void RetainCallee(TimedMsgLoop* obj) {}
+ static void ReleaseCallee(TimedMsgLoop* obj) {}
+};
+
+// Saves typing. It's somewhat hard to create a wrapper around
+// testing::InvokeWithoutArgs since it returns a
+// non-public (testing::internal) type.
+#define QUIT_LOOP(loop) testing::InvokeWithoutArgs(TaskHolder(\
+ NewRunnableMethod(&loop, &TimedMsgLoop::Quit)))
+
+// We mock ChromeFrameDelegate only. The rest is with real AutomationProxy
+TEST(CFACWithChrome, CreateTooFast) {
+ MockCFDelegate cfd;
+ TimedMsgLoop loop;
+ int timeout = 0; // Chrome cannot send Hello message so fast.
+ const std::wstring profile = L"Adam.N.Epilinter";
+
+ scoped_ptr<ChromeFrameAutomationClient> client;
+ client.reset(new ChromeFrameAutomationClient());
+
+ EXPECT_CALL(cfd, OnAutomationServerLaunchFailed(AUTOMATION_TIMEOUT,
+ testing::_))
+ .Times(1)
+ .WillOnce(QUIT_LOOP(loop));
+
+ EXPECT_TRUE(client->Initialize(&cfd, timeout, false, profile, L"", false));
+ loop.RunFor(10);
+ client->Uninitialize();
+}
+
+// This test may fail if Chrome take more that 10 seconds (timeout var) to
+// launch. In this case GMock shall print something like "unexpected call to
+// OnAutomationServerLaunchFailed". I'm yet to find out how to specify
+// that this is an unexpected call, and still to execute and action.
+TEST(CFACWithChrome, CreateNotSoFast) {
+ MockCFDelegate cfd;
+ TimedMsgLoop loop;
+ const std::wstring profile = L"Adam.N.Epilinter";
+ int timeout = 10000;
+
+ scoped_ptr<ChromeFrameAutomationClient> client;
+ client.reset(new ChromeFrameAutomationClient);
+
+ EXPECT_CALL(cfd, OnAutomationServerReady())
+ .Times(1)
+ .WillOnce(QUIT_LOOP(loop));
+
+ EXPECT_CALL(cfd, OnAutomationServerLaunchFailed(testing::_, testing::_))
+ .Times(0);
+
+ EXPECT_TRUE(client->Initialize(&cfd, timeout, false, profile, L"", false));
+
+ loop.RunFor(11);
+ client->Uninitialize();
+ client.reset(NULL);
+}
+
+MATCHER_P(MsgType, msg_type, "IPC::Message::type()") {
+ const IPC::Message& m = arg;
+ return (m.type() == msg_type);
+}
+
+MATCHER_P(EqNavigationInfoUrl, url, "IPC::NavigationInfo matcher") {
+ if (url.is_valid() && url != arg.url)
+ return false;
+ // TODO: other members
+ return true;
+}
+
+TEST(CFACWithChrome, NavigateOk) {
+ MockCFDelegate cfd;
+ TimedMsgLoop loop;
+ const std::wstring profile = L"Adam.N.Epilinter";
+ const std::string url = "about:version";
+ int timeout = 10000;
+
+ scoped_ptr<ChromeFrameAutomationClient> client;
+ client.reset(new ChromeFrameAutomationClient);
+
+ EXPECT_CALL(cfd, OnAutomationServerReady())
+ .WillOnce(testing::InvokeWithoutArgs(TaskHolder(NewRunnableMethod(
+ client.get(), &ChromeFrameAutomationClient::InitiateNavigation,
+ url))));
+
+// cfd.SetOnNavigationStateChanged();
+ EXPECT_CALL(cfd,
+ OnNavigationStateChanged(testing::_, testing::_))
+ .Times(testing::AnyNumber());
+
+ {
+ testing::InSequence s;
+
+ EXPECT_CALL(cfd, OnDidNavigate(testing::_, EqNavigationInfoUrl(GURL())))
+ .Times(1);
+
+ EXPECT_CALL(cfd, OnUpdateTargetUrl(testing::_, testing::_)).Times(1);
+
+ EXPECT_CALL(cfd, OnLoad(testing::_, testing::_))
+ .Times(1)
+ .WillOnce(QUIT_LOOP(loop));
+ }
+
+ EXPECT_TRUE(client->Initialize(&cfd, timeout, false, profile, L"", false));
+ loop.RunFor(10);
+ client->Uninitialize();
+ client.reset(NULL);
+}
+
+// Bug: http://b/issue?id=2033644
+TEST(CFACWithChrome, DISABLED_NavigateFailed) {
+ MockCFDelegate cfd;
+ TimedMsgLoop loop;
+ const std::wstring profile = L"Adam.N.Epilinter";
+ const std::string url = "http://127.0.0.3:65412/";
+ int timeout = 10000;
+
+ scoped_ptr<ChromeFrameAutomationClient> client;
+ client.reset(new ChromeFrameAutomationClient);
+
+ EXPECT_CALL(cfd, OnAutomationServerReady())
+ .WillOnce(testing::InvokeWithoutArgs(TaskHolder(NewRunnableMethod(
+ client.get(), &ChromeFrameAutomationClient::InitiateNavigation,
+ url))));
+
+ EXPECT_CALL(cfd,
+ OnNavigationStateChanged(testing::_, testing::_))
+ .Times(testing::AnyNumber());
+
+ EXPECT_CALL(cfd, OnNavigationFailed(testing::_, testing::_, testing::_))
+ .Times(1);
+
+ EXPECT_CALL(cfd, OnUpdateTargetUrl(testing::_, testing::_))
+ .Times(testing::AnyNumber());
+
+ EXPECT_CALL(cfd, OnLoad(testing::_, testing::_))
+ .Times(0);
+
+ EXPECT_TRUE(client->Initialize(&cfd, timeout, false, profile, L"", false));
+
+ loop.RunFor(10);
+ client->Uninitialize();
+ client.reset(NULL);
+}
+
+MATCHER_P(EqURLRequest, x, "IPC::AutomationURLRequest matcher") {
+ if (arg.url != x.url)
+ return false;
+ if (arg.method != x.method)
+ return false;
+ if (arg.referrer != x.referrer)
+ return false;
+ if (arg.extra_request_headers != x.extra_request_headers)
+ return false;
+ // TODO: uploaddata member
+ return true;
+}
+
+MATCHER_P(EqUrlGet, url, "Quick URL matcher for 'HTTP GET' request") {
+ if (arg.url != url)
+ return false;
+ if (arg.method != "GET")
+ return false;
+ return true;
+}
+
+TEST(CFACWithChrome, UseHostNetworkStack) {
+ MockCFDelegate cfd;
+ TimedMsgLoop loop;
+ const std::wstring profile = L"Adam.N.Epilinter";
+ const std::string url = "http://bongo.com";
+ int timeout = 10000;
+
+ scoped_ptr<ChromeFrameAutomationClient> client;
+ client.reset(new ChromeFrameAutomationClient);
+ client->set_use_chrome_network(false);
+ cfd.SetAutomationSender(client.get());
+
+ EXPECT_CALL(cfd, OnAutomationServerReady())
+ .WillOnce(testing::InvokeWithoutArgs(TaskHolder(NewRunnableMethod(
+ client.get(), &ChromeFrameAutomationClient::InitiateNavigation,
+ url))));
+
+ EXPECT_CALL(cfd, OnNavigationStateChanged(testing::_, testing::_))
+ .Times(testing::AnyNumber());
+
+ EXPECT_CALL(cfd, GetBounds(testing::_))
+ .Times(testing::AtMost(1));
+
+ EXPECT_CALL(cfd, OnUpdateTargetUrl(testing::_, testing::_))
+ .Times(testing::AnyNumber());
+
+ // Note slash appending to the url string, because of GURL inside chrome
+ const IPC::AutomationURLResponse found = {"", "HTTP/0.9 200\r\n\r\n\r\n", };
+
+ // Hard coded tab and request ids
+ static const int tab_id = 1;
+ int request_id = 2;
+
+ EXPECT_CALL(cfd, OnRequestStart(tab_id, request_id, EqUrlGet(url + '/')))
+ .Times(1)
+ .WillOnce(testing::Invoke(CBF(&cfd, &MockCFDelegate::ReplyStarted,
+ &found)));
+
+ // Return some trivial page, that have a link to a "logo.gif" image
+ const std::string data = "<!DOCTYPE html><title>Hello</title>"
+ "<img src=\"logo.gif\">";
+ EXPECT_CALL(cfd, OnRequestRead(tab_id, request_id, testing::Ge(0)))
+ .Times(2)
+ .WillOnce(testing::Invoke(CBF(&cfd, &MockCFDelegate::ReplyData, &data)))
+ .WillOnce(testing::WithArgs<0, 1>(testing::Invoke(CBF(&cfd,
+ &MockCFDelegate::ReplyEOF))));
+
+ EXPECT_CALL(cfd, OnDidNavigate(tab_id, EqNavigationInfoUrl(GURL(url))))
+ .Times(1);
+ EXPECT_CALL(cfd, OnLoad(tab_id, GURL(url)))
+ .Times(1);
+
+ // Expect request for logo.gif
+ request_id++;
+ EXPECT_CALL(cfd,
+ OnRequestStart(tab_id, request_id, EqUrlGet(url + "/logo.gif")))
+ .Times(1)
+ .WillOnce(testing::Invoke(CBF(&cfd, &MockCFDelegate::Reply404)));
+
+ EXPECT_CALL(cfd, OnRequestRead(tab_id, request_id, testing::_))
+ .Times(testing::AtMost(1));
+
+ // Chrome makes a brave request for favicon.ico
+ request_id++;
+ EXPECT_CALL(cfd,
+ OnRequestStart(tab_id, request_id, EqUrlGet(url + "/favicon.ico")))
+ .Times(1)
+ .WillOnce(testing::Invoke(CBF(&cfd, &MockCFDelegate::Reply404)));
+
+ EXPECT_CALL(cfd, OnRequestRead(tab_id, request_id, testing::_))
+ .Times(testing::AtMost(1));
+
+ bool incognito = true;
+ EXPECT_TRUE(client->Initialize(&cfd, timeout, false, profile, L"",
+ incognito));
+
+ loop.RunFor(10);
+ client->Uninitialize();
+ client.reset(NULL);
+}
+
+
+// [CFAC] -- uses a ProxyFactory for creation of ChromeFrameAutomationProxy
+// -- uses ChromeFrameAutomationProxy
+// -- uses TabProxy obtained from ChromeFrameAutomationProxy
+// -- uses ChromeFrameDelegate as outgoing interface
+//
+// We mock ProxyFactory to return mock object (MockAutomationProxy) implementing
+// ChromeFrameAutomationProxy interface.
+// Since CFAC uses TabProxy for few calls and TabProxy is not easy mockable,
+// we create 'real' TabProxy but with fake AutomationSender (the one responsible
+// for sending messages over channel).
+// Additionally we have mock implementation ChromeFrameDelagate interface -
+// MockCFDelegate.
+
+// Test fixture, saves typing all of it's members.
+class CFACMockTest : public testing::Test {
+ public:
+ MockProxyFactory factory_;
+ MockCFDelegate cfd_;
+ TimedMsgLoop loop_;
+ MockAutomationProxy proxy_;
+ scoped_ptr<AutomationHandleTracker> tracker_;
+ MockAutomationMessageSender dummy_sender_;
+ scoped_refptr<TabProxy> tab_;
+ scoped_ptr<ChromeFrameAutomationClient> client_; // the victim of all tests
+
+ std::wstring profile_;
+ int timeout_;
+ void* id_; // Automation server id we are going to return
+ int tab_handle_; // Tab handle. Any non-zero value is Ok.
+
+ inline ChromeFrameAutomationProxy* get_proxy() {
+ return static_cast<ChromeFrameAutomationProxy*>(&proxy_);
+ }
+
+ inline void CreateTab() {
+ ASSERT_EQ(NULL, tab_.get());
+ tab_ = new TabProxy(&dummy_sender_, tracker_.get(), tab_handle_);
+ }
+
+ // Easy methods to set expectations.
+ void SetAutomationServerOk() {
+ EXPECT_CALL(factory_, GetAutomationServer(testing::Eq(timeout_),
+ testing::StrEq(profile_),
+ testing::_,
+ testing::_,
+ testing::NotNull()))
+ .Times(1)
+ .WillOnce(testing::DoAll(
+ testing::WithArgs<0, 4>(
+ testing::Invoke(CBF(&factory_, &MockProxyFactory::GetServerImpl,
+ get_proxy(), AUTOMATION_SUCCESS))),
+ testing::Return(id_)));
+
+ EXPECT_CALL(factory_, ReleaseAutomationServer(testing::Eq(id_))).Times(1);
+ }
+
+ void Set_CFD_LaunchFailed(AutomationLaunchResult result) {
+ EXPECT_CALL(cfd_, OnAutomationServerLaunchFailed(
+ testing::Eq(result), testing::_))
+ .Times(1)
+ .WillOnce(QUIT_LOOP(loop_));
+ }
+
+ protected:
+ CFACMockTest() : tracker_(NULL), timeout_(500),
+ profile_(L"Adam.N.Epilinter") {
+ id_ = reinterpret_cast<void*>(5);
+ tab_handle_ = 3;
+ }
+
+ virtual void SetUp() {
+ dummy_sender_.ForwardTo(&proxy_);
+ tracker_.reset(new AutomationHandleTracker(&dummy_sender_));
+
+ client_.reset(new ChromeFrameAutomationClient);
+ client_->set_proxy_factory(&factory_);
+ }
+};
+
+// Could be implemented as MockAutomationProxy member (we have WithArgs<>!)
+ACTION_P3(HandleCreateTab, tab_handle, external_tab_container, tab_wnd) {
+ // arg0 - message
+ // arg1 - callback
+ // arg2 - key
+ CallbackRunner<Tuple3<HWND, HWND, int> >* c =
+ reinterpret_cast<CallbackRunner<Tuple3<HWND, HWND, int> >*>(arg1);
+ c->Run(external_tab_container, tab_wnd, tab_handle);
+ delete c;
+ delete arg0;
+}
+
+TEST_F(CFACMockTest, MockedCreateTabOk) {
+ int timeout = 500;
+ CreateTab();
+ SetAutomationServerOk();
+
+ EXPECT_CALL(proxy_, server_version()).Times(testing::AnyNumber())
+ .WillRepeatedly(testing::Return(""));
+
+ // We need some valid HWNDs, when responding to CreateExternalTab
+ HWND h1 = ::GetDesktopWindow();
+ HWND h2 = ::GetDesktopWindow();
+ EXPECT_CALL(proxy_, SendAsAsync(testing::Property(&IPC::SyncMessage::type,
+ AutomationMsg_CreateExternalTab__ID),
+ testing::NotNull(), testing::_))
+ .Times(1)
+ .WillOnce(HandleCreateTab(tab_handle_, h1, h2));
+
+ EXPECT_CALL(proxy_, CreateTabProxy(testing::Eq(tab_handle_)))
+ .WillOnce(testing::Return(tab_));
+
+ EXPECT_CALL(cfd_, OnAutomationServerReady())
+ .WillOnce(QUIT_LOOP(loop_));
+
+ // Here we go!
+ EXPECT_TRUE(client_->Initialize(&cfd_, timeout, false, profile_, L"", false));
+ loop_.RunFor(10);
+ client_->Uninitialize();
+}
+
+TEST_F(CFACMockTest, MockedCreateTabFailed) {
+ HWND null_wnd = NULL;
+ SetAutomationServerOk();
+
+ EXPECT_CALL(proxy_, server_version()).Times(testing::AnyNumber())
+ .WillRepeatedly(testing::Return(""));
+
+ EXPECT_CALL(proxy_, SendAsAsync(testing::Property(&IPC::SyncMessage::type,
+ AutomationMsg_CreateExternalTab__ID),
+ testing::NotNull(), testing::_))
+ .Times(1)
+ .WillOnce(HandleCreateTab(tab_handle_, null_wnd, null_wnd));
+
+ EXPECT_CALL(proxy_, CreateTabProxy(testing::_)).Times(0);
+
+ Set_CFD_LaunchFailed(AUTOMATION_CREATE_TAB_FAILED);
+
+ // Here we go!
+ EXPECT_TRUE(client_->Initialize(&cfd_, timeout_, false, profile_, L"",
+ false));
+ loop_.RunFor(4);
+ client_->Uninitialize();
+}
+
+const wchar_t kMetaTagPage[] = L"files/meta_tag.html";
+TEST_F(ChromeFrameTestWithWebServer, FullTabModeIE_MetaTag) {
+ SimpleBrowserTest(IE, kMetaTagPage, L"meta_tag");
+}
+
+const wchar_t kCFProtocolPage[] = L"files/chrome_frame_protocol.html";
+TEST_F(ChromeFrameTestWithWebServer, FullTabModeIE_CFProtocol) {
+ SimpleBrowserTest(IE, kCFProtocolPage, L"chrome_frame_protocol");
+}
+
+const wchar_t kPersistentCookieTest[] =
+ L"files/persistent_cookie_test_page.html";
+TEST_F(ChromeFrameTestWithWebServer, FullTabModeIE_PersistentCookieTest) {
+ SimpleBrowserTest(IE, kPersistentCookieTest, L"PersistentCookieTest");
+}
+
+const wchar_t kNavigateOutPage[] = L"files/navigate_out.html";
+TEST_F(ChromeFrameTestWithWebServer, FullTabModeIE_NavigateOut) {
+ SimpleBrowserTest(IE, kNavigateOutPage, L"navigate_out");
+}
+
+HRESULT LaunchIEAsComServer(IWebBrowser2** web_browser) {
+ if (!web_browser)
+ return E_INVALIDARG;
+
+ ScopedComPtr<IWebBrowser2> web_browser2;
+ HRESULT hr = CoCreateInstance(
+ CLSID_InternetExplorer, NULL, CLSCTX_LOCAL_SERVER, IID_IWebBrowser2,
+ reinterpret_cast<void**>(web_browser2.Receive()));
+
+ if (SUCCEEDED(hr)) {
+ *web_browser = web_browser2.Detach();
+ }
+
+ return hr;
+}
+
+TEST(ChromeFrameTest, FullTabModeIE_DisallowedUrls) {
+ int major_version = 0;
+ int minor_version = 0;
+ int bugfix_version = 0;
+
+ base::SysInfo::OperatingSystemVersionNumbers(&major_version, &minor_version,
+ &bugfix_version);
+ if (major_version > 5) {
+ DLOG(INFO) << __FUNCTION__ << " Not running test on Windows version: "
+ << major_version;
+ return;
+ }
+
+ IEVersion ie_version = GetIEVersion();
+ if (ie_version == IE_8) {
+ DLOG(INFO) << __FUNCTION__ << " Not running test on IE8";
+ return;
+ }
+
+ HRESULT hr = CoInitialize(NULL);
+ bool should_uninit = SUCCEEDED(hr);
+
+ ScopedComPtr<IWebBrowser2> web_browser2;
+ EXPECT_TRUE(S_OK == LaunchIEAsComServer(web_browser2.Receive()));
+ web_browser2->put_Visible(VARIANT_TRUE);
+
+ CComObject<WebBrowserEventSink>* web_browser_sink = NULL;
+ CComObject<WebBrowserEventSink>::CreateInstance(&web_browser_sink);
+
+ // Pass the main thread id to the browser sink so that it can notify
+ // us about test completion.
+ web_browser_sink->set_main_thread_id(GetCurrentThreadId());
+
+ hr = web_browser_sink->DispEventAdvise(web_browser2,
+ &DIID_DWebBrowserEvents2);
+ EXPECT_TRUE(hr == S_OK);
+
+ VARIANT empty = ScopedVariant::kEmptyVariant;
+ ScopedVariant url;
+ url.Set(L"cf:file:///C:/");
+
+ TimedMsgLoop loop;
+
+ hr = web_browser2->Navigate2(url.AsInput(), &empty, &empty, &empty, &empty);
+ EXPECT_TRUE(hr == S_OK);
+
+ loop.RunFor(10);
+
+ EXPECT_TRUE(web_browser_sink->navigation_failed());
+
+ hr = web_browser_sink->DispEventUnadvise(web_browser2);
+ EXPECT_TRUE(hr == S_OK);
+
+ web_browser2.Release();
+ chrome_frame_test::CloseAllIEWindows();
+
+ if (should_uninit) {
+ CoUninitialize();
+ }
+}
+
diff --git a/chrome_frame/test/chrome_frame_unittests.h b/chrome_frame/test/chrome_frame_unittests.h
new file mode 100644
index 0000000..98b5985
--- /dev/null
+++ b/chrome_frame/test/chrome_frame_unittests.h
@@ -0,0 +1,164 @@
+// Copyright (c) 2006-2008 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.
+#ifndef CHROME_FRAME_TEST_CHROME_FRAME_UNITTESTS_H_
+#define CHROME_FRAME_TEST_CHROME_FRAME_UNITTESTS_H_
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <string>
+#include <exdisp.h>
+#include <exdispid.h>
+
+#include "base/ref_counted.h"
+#include "base/scoped_handle_win.h"
+#include "googleurl/src/gurl.h"
+#include "chrome_frame/test/http_server.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+// Class that:
+// 1) Starts the local webserver,
+// 2) Supports launching browsers - Internet Explorer and Firefox with local url
+// 3) Wait the webserver to finish. It is supposed the test webpage to shutdown
+// the server by navigating to "kill" page
+// 4) Supports read the posted results from the test webpage to the "dump"
+// webserver directory
+class ChromeFrameTestWithWebServer: public testing::Test {
+ protected:
+ enum BrowserKind { INVALID, IE, FIREFOX, OPERA, SAFARI, CHROME };
+
+ bool LaunchBrowser(BrowserKind browser, const wchar_t* url);
+ bool WaitForTestToComplete(int milliseconds);
+ // Waits for the page to notify us of the window.onload event firing.
+ // Note that the milliseconds value is only approximate.
+ bool WaitForOnLoad(int milliseconds);
+ bool ReadResultFile(const std::wstring& file_name, std::string* data);
+
+ // Launches the specified browser and waits for the test to complete
+ // (see WaitForTestToComplete). Then checks that the outcome is OK.
+ // This function uses EXPECT_TRUE and ASSERT_TRUE for all steps performed
+ // hence no return value.
+ void SimpleBrowserTest(BrowserKind browser, const wchar_t* page,
+ const wchar_t* result_file_to_check);
+
+ // Same as SimpleBrowserTest but if the browser isn't installed (LaunchBrowser
+ // fails), the function will print out a warning but not treat the test
+ // as failed.
+ // Currently this is how we run Opera tests.
+ void OptionalBrowserTest(BrowserKind browser, const wchar_t* page,
+ const wchar_t* result_file_to_check);
+
+ // Test if chrome frame correctly reports its version.
+ void VersionTest(BrowserKind browser, const wchar_t* page,
+ const wchar_t* result_file_to_check);
+
+ void CloseBrowser();
+
+ // Ensures (well, at least tries to ensure) that the browser window has focus.
+ bool BringBrowserToTop();
+
+ // Returns true iff the specified result file contains 'expected result'.
+ bool CheckResultFile(const std::wstring& file_name,
+ const std::string& expected_result);
+
+ virtual void SetUp();
+ virtual void TearDown();
+
+ // Important: kind means "sheep" in Icelandic. ?:-o
+ const char* ToString(BrowserKind kind) {
+ switch (kind) {
+ case IE:
+ return "IE";
+ case FIREFOX:
+ return "Firefox";
+ case OPERA:
+ return "Opera";
+ case CHROME:
+ return "Chrome";
+ case SAFARI:
+ return "Safari";
+ default:
+ NOTREACHED();
+ break;
+ }
+ return "";
+ }
+
+ BrowserKind browser_;
+ std::wstring results_dir_;
+ ScopedHandle browser_handle_;
+ ChromeFrameHTTPServer server_;
+};
+
+// This class sets up event sinks to the IWebBrowser interface. Currently it
+// subscribes to the following events:-
+// 1. DISPID_BEFORENAVIGATE2
+// 2. DISPID_NAVIGATEERROR
+// 3. DISPID_NAVIGATECOMPLETE2
+// Other events can be subscribed to on an if needed basis.
+class WebBrowserEventSink
+ : public CComObjectRootEx<CComSingleThreadModel>,
+ public IDispEventSimpleImpl<0, WebBrowserEventSink,
+ &DIID_DWebBrowserEvents2> {
+ public:
+ WebBrowserEventSink()
+ : navigation_failed_(false),
+ main_thread_id_(0) {
+ }
+
+BEGIN_COM_MAP(WebBrowserEventSink)
+END_COM_MAP()
+
+BEGIN_SINK_MAP(WebBrowserEventSink)
+ SINK_ENTRY_INFO(0, DIID_DWebBrowserEvents2, DISPID_BEFORENAVIGATE2,
+ OnBeforeNavigate2, &kBeforeNavigate2Info)
+ SINK_ENTRY_INFO(0, DIID_DWebBrowserEvents2, DISPID_NAVIGATECOMPLETE2,
+ OnNavigateComplete2, &kNavigateComplete2Info)
+ SINK_ENTRY_INFO(0, DIID_DWebBrowserEvents2, DISPID_NAVIGATEERROR,
+ OnNavigateError, &kNavigateErrorInfo)
+END_SINK_MAP()
+
+ STDMETHOD_(void, OnNavigateError)(IDispatch* dispatch, VARIANT* url,
+ VARIANT* frame_name, VARIANT* status_code,
+ VARIANT* cancel) {
+ navigation_failed_ = true;
+ }
+
+ STDMETHOD(OnBeforeNavigate2)(IDispatch* dispatch, VARIANT* url, VARIANT*
+ flags, VARIANT* target_frame_name,
+ VARIANT* post_data, VARIANT* headers,
+ VARIANT_BOOL* cancel) {
+ DLOG(INFO) << __FUNCTION__;
+ // If a navigation fails then IE issues a navigation to an interstitial
+ // page. Catch this to track navigation errors as the NavigateError
+ // notification does not seem to fire reliably.
+ GURL crack_url(url->bstrVal);
+ if (crack_url.scheme() == "res") {
+ navigation_failed_ = true;
+ }
+ return S_OK;
+ }
+
+ STDMETHOD_(void, OnNavigateComplete2)(IDispatch* dispatch, VARIANT* url) {
+ DLOG(INFO) << __FUNCTION__;
+ }
+
+ bool navigation_failed() const {
+ return navigation_failed_;
+ }
+
+ void set_main_thread_id(DWORD thread_id) {
+ main_thread_id_ = thread_id;
+ }
+
+ protected:
+ bool navigation_failed_;
+
+ static _ATL_FUNC_INFO kBeforeNavigate2Info;
+ static _ATL_FUNC_INFO kNavigateComplete2Info;
+ static _ATL_FUNC_INFO kNavigateErrorInfo;
+ DWORD main_thread_id_;
+};
+
+#endif // CHROME_FRAME_TEST_CHROME_FRAME_UNITTESTS_H_
+
diff --git a/chrome_frame/test/chrometab_unittests.vsprops b/chrome_frame/test/chrometab_unittests.vsprops
new file mode 100644
index 0000000..d5375c9
--- /dev/null
+++ b/chrome_frame/test/chrometab_unittests.vsprops
@@ -0,0 +1,14 @@
+<?xml version="1.0" encoding="Windows-1252"?>
+<VisualStudioPropertySheet
+ ProjectType="Visual C++"
+ Version="8.00"
+ Name="chrometab_unittests"
+ InheritedPropertySheets="$(SolutionDir)..\tools\grit\build\using_generated_resources.vsprops"
+ >
+ <Tool
+ Name="VCCLCompilerTool"
+ PreprocessorDefinitions="_ATL_APARTMENT_THREADED;_ATL_CSTRING_EXPLICIT_CONSTRUCTORS"
+ AdditionalIncludeDirectories=""
+ />
+
+</VisualStudioPropertySheet>
diff --git a/chrome_frame/test/com_message_event_unittest.cc b/chrome_frame/test/com_message_event_unittest.cc
new file mode 100644
index 0000000..f850c20
--- /dev/null
+++ b/chrome_frame/test/com_message_event_unittest.cc
@@ -0,0 +1,325 @@
+// Copyright (c) 2009 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 "chrome_frame/com_message_event.h"
+
+#include <atlbase.h>
+#include <atlcom.h>
+
+#include "testing/gtest/include/gtest/gtest.h"
+
+// To allow the unit test read-only access to check protected member variables.
+class FriendlyComMessageEvent : public ComMessageEvent {
+ public:
+ inline IHTMLEventObj* basic_event() { return basic_event_; }
+};
+
+class ATL_NO_VTABLE MockDumbContainer :
+ public CComObjectRoot,
+ public IOleContainer {
+ public:
+ DECLARE_NOT_AGGREGATABLE(MockDumbContainer)
+ BEGIN_COM_MAP(MockDumbContainer)
+ COM_INTERFACE_ENTRY(IParseDisplayName)
+ COM_INTERFACE_ENTRY(IOleContainer)
+ END_COM_MAP()
+
+ STDMETHOD(ParseDisplayName)(IBindCtx*, LPOLESTR, ULONG*, IMoniker**) {
+ NOTREACHED();
+ return E_NOTIMPL;
+ }
+ STDMETHOD(EnumObjects)(DWORD, IEnumUnknown**) {
+ NOTREACHED();
+ return E_NOTIMPL;
+ }
+ STDMETHOD(LockContainer)(BOOL) {
+ NOTREACHED();
+ return E_NOTIMPL;
+ }
+};
+
+TEST(ComMessageEvent, WithDumbContainer) {
+ CComObject<MockDumbContainer>* container_obj = NULL;
+ CComObject<MockDumbContainer>::CreateInstance(&container_obj);
+ ScopedComPtr<IOleContainer> container(container_obj);
+ EXPECT_FALSE(!container);
+
+ CComObject<FriendlyComMessageEvent>* event_obj = NULL;
+ CComObject<FriendlyComMessageEvent>::CreateInstance(&event_obj);
+ ScopedComPtr<IUnknown> event_ref(event_obj);
+
+ bool result = event_obj->Initialize(container, "hi", "http://www.foo.com/",
+ "message");
+ EXPECT_TRUE(result);
+ EXPECT_TRUE(!event_obj->basic_event());
+}
+
+// Mock object to mimic a "smart" container, e.g. IE, that will
+// be able to return an IHTMLDocument2 and 4, and from which you
+// can get an IHTMLEventObj implementation. Doubles as a mock
+// IHTMLEventObj implementation.
+class ATL_NO_VTABLE MockSmartContainer :
+ public CComObjectRoot,
+ public IOleContainer,
+ public IHTMLDocument2,
+ public IHTMLDocument4,
+ public IHTMLEventObj {
+ public:
+ DECLARE_NOT_AGGREGATABLE(MockSmartContainer)
+ BEGIN_COM_MAP(MockSmartContainer)
+ COM_INTERFACE_ENTRY_IID(IID_IDispatch, IHTMLEventObj)
+ COM_INTERFACE_ENTRY(IParseDisplayName)
+ COM_INTERFACE_ENTRY(IOleContainer)
+ COM_INTERFACE_ENTRY(IHTMLDocument)
+ COM_INTERFACE_ENTRY(IHTMLDocument2)
+ COM_INTERFACE_ENTRY(IHTMLDocument4)
+ COM_INTERFACE_ENTRY(IHTMLEventObj)
+ END_COM_MAP()
+
+ static const DISPID kDispId = 424242;
+ static const long kResultValue = 42;
+
+ // Only method we actually implement from IHTMLDocument4, to give
+ // out the mock IHTMLEventObj.
+ STDMETHOD(createEventObject)(VARIANT*, IHTMLEventObj** event_obj) {
+ return GetUnknown()->QueryInterface(event_obj);
+ }
+
+ // Dummy IDispatch implementation for unit testing, to validate
+ // passthrough semantics.
+ STDMETHOD(GetIDsOfNames)(REFIID iid, LPOLESTR* names, UINT num_names,
+ LCID lcid, DISPID* disp_ids) {
+ DCHECK(num_names == 1);
+ disp_ids[0] = kDispId;
+ return S_OK;
+ }
+
+ STDMETHOD(Invoke)(DISPID id, REFIID iid, LCID lcid, WORD flags,
+ DISPPARAMS* disp_params, VARIANT* var_result,
+ EXCEPINFO* excep_info, UINT* arg_error) {
+ var_result->vt = VT_I4;
+ var_result->lVal = kResultValue;
+ return S_OK;
+ }
+
+
+ // Do-nothing implementation of the rest of the interface methods.
+ // To make this less verbose, define a macro here and undefine it
+ // at the end of the list.
+#define STDMETHODNOTIMP(method, parameters) \
+ STDMETHOD(method) parameters { \
+ NOTREACHED(); \
+ return E_NOTIMPL; \
+ }
+
+ // IDispatch
+ STDMETHODNOTIMP(GetTypeInfoCount, (UINT*));
+ STDMETHODNOTIMP(GetTypeInfo, (UINT, LCID, ITypeInfo**));
+
+ // IParseDisplayName
+ STDMETHODNOTIMP(ParseDisplayName, (IBindCtx*, LPOLESTR, ULONG*, IMoniker**));
+ // IOleContainer
+ STDMETHODNOTIMP(EnumObjects, (DWORD, IEnumUnknown**));
+ STDMETHODNOTIMP(LockContainer, (BOOL));
+ // IHTMLDocument
+ STDMETHODNOTIMP(get_Script, (IDispatch**));
+ // IHTMLDocument2
+ STDMETHODNOTIMP(get_all, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(get_body, (IHTMLElement**));
+ STDMETHODNOTIMP(get_activeElement, (IHTMLElement**));
+ STDMETHODNOTIMP(get_images, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(get_applets, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(get_links, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(get_forms, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(get_anchors, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(put_title, (BSTR));
+ STDMETHODNOTIMP(get_title, (BSTR*));
+ STDMETHODNOTIMP(get_scripts, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(put_designMode, (BSTR));
+ STDMETHODNOTIMP(get_designMode, (BSTR*));
+ STDMETHODNOTIMP(get_selection, (IHTMLSelectionObject**));
+ STDMETHODNOTIMP(get_readyState, (BSTR*));
+ STDMETHODNOTIMP(get_frames, (IHTMLFramesCollection2**));
+ STDMETHODNOTIMP(get_embeds, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(get_plugins, (IHTMLElementCollection**));
+ STDMETHODNOTIMP(put_alinkColor, (VARIANT));
+ STDMETHODNOTIMP(get_alinkColor, (VARIANT*));
+ STDMETHODNOTIMP(put_bgColor, (VARIANT));
+ STDMETHODNOTIMP(get_bgColor, (VARIANT*));
+ STDMETHODNOTIMP(put_fgColor, (VARIANT));
+ STDMETHODNOTIMP(get_fgColor, (VARIANT*));
+ STDMETHODNOTIMP(put_linkColor, (VARIANT));
+ STDMETHODNOTIMP(get_linkColor, (VARIANT*));
+ STDMETHODNOTIMP(put_vlinkColor, (VARIANT));
+ STDMETHODNOTIMP(get_vlinkColor, (VARIANT*));
+ STDMETHODNOTIMP(get_referrer, (BSTR*));
+ STDMETHODNOTIMP(get_location, (IHTMLLocation**));
+ STDMETHODNOTIMP(get_lastModified, (BSTR*));
+ STDMETHODNOTIMP(put_URL, (BSTR));
+ STDMETHODNOTIMP(get_URL, (BSTR*));
+ STDMETHODNOTIMP(put_domain, (BSTR));
+ STDMETHODNOTIMP(get_domain, (BSTR*));
+ STDMETHODNOTIMP(put_cookie, (BSTR));
+ STDMETHODNOTIMP(get_cookie, (BSTR*));
+ STDMETHODNOTIMP(put_expando, (VARIANT_BOOL));
+ STDMETHODNOTIMP(get_expando, (VARIANT_BOOL*));
+ STDMETHODNOTIMP(put_charset, (BSTR));
+ STDMETHODNOTIMP(get_charset, (BSTR*));
+ STDMETHODNOTIMP(put_defaultCharset, (BSTR));
+ STDMETHODNOTIMP(get_defaultCharset, (BSTR*));
+ STDMETHODNOTIMP(get_mimeType, (BSTR*));
+ STDMETHODNOTIMP(get_fileSize, (BSTR*));
+ STDMETHODNOTIMP(get_fileCreatedDate, (BSTR*));
+ STDMETHODNOTIMP(get_fileModifiedDate, (BSTR*));
+ STDMETHODNOTIMP(get_fileUpdatedDate, (BSTR*));
+ STDMETHODNOTIMP(get_security, (BSTR*));
+ STDMETHODNOTIMP(get_protocol, (BSTR*));
+ STDMETHODNOTIMP(get_nameProp, (BSTR*));
+ STDMETHODNOTIMP(write, (SAFEARRAY*));
+ STDMETHODNOTIMP(writeln, (SAFEARRAY*));
+ STDMETHODNOTIMP(open, (BSTR, VARIANT, VARIANT, VARIANT, IDispatch**));
+ STDMETHODNOTIMP(close, ());
+ STDMETHODNOTIMP(clear, ());
+ STDMETHODNOTIMP(queryCommandSupported, (BSTR, VARIANT_BOOL*));
+ STDMETHODNOTIMP(queryCommandEnabled, (BSTR, VARIANT_BOOL*));
+ STDMETHODNOTIMP(queryCommandState, (BSTR, VARIANT_BOOL*));
+ STDMETHODNOTIMP(queryCommandIndeterm, (BSTR, VARIANT_BOOL*));
+ STDMETHODNOTIMP(queryCommandText, (BSTR, BSTR*));
+ STDMETHODNOTIMP(queryCommandValue, (BSTR, VARIANT*));
+ STDMETHODNOTIMP(execCommand, (BSTR, VARIANT_BOOL, VARIANT, VARIANT_BOOL*));
+ STDMETHODNOTIMP(execCommandShowHelp, (BSTR, VARIANT_BOOL*));
+ STDMETHODNOTIMP(createElement, (BSTR, IHTMLElement**));
+ STDMETHODNOTIMP(put_onhelp, (VARIANT));
+ STDMETHODNOTIMP(get_onhelp, (VARIANT*));
+ STDMETHODNOTIMP(put_onclick, (VARIANT));
+ STDMETHODNOTIMP(get_onclick, (VARIANT*));
+ STDMETHODNOTIMP(put_ondblclick, (VARIANT));
+ STDMETHODNOTIMP(get_ondblclick, (VARIANT*));
+ STDMETHODNOTIMP(put_onkeyup, (VARIANT));
+ STDMETHODNOTIMP(get_onkeyup, (VARIANT*));
+ STDMETHODNOTIMP(put_onkeydown, (VARIANT));
+ STDMETHODNOTIMP(get_onkeydown, (VARIANT*));
+ STDMETHODNOTIMP(put_onkeypress, (VARIANT));
+ STDMETHODNOTIMP(get_onkeypress, (VARIANT*));
+ STDMETHODNOTIMP(put_onmouseup, (VARIANT));
+ STDMETHODNOTIMP(get_onmouseup, (VARIANT*));
+ STDMETHODNOTIMP(put_onmousedown, (VARIANT));
+ STDMETHODNOTIMP(get_onmousedown, (VARIANT*));
+ STDMETHODNOTIMP(put_onmousemove, (VARIANT));
+ STDMETHODNOTIMP(get_onmousemove, (VARIANT*));
+ STDMETHODNOTIMP(put_onmouseout, (VARIANT));
+ STDMETHODNOTIMP(get_onmouseout, (VARIANT*));
+ STDMETHODNOTIMP(put_onmouseover, (VARIANT));
+ STDMETHODNOTIMP(get_onmouseover, (VARIANT*));
+ STDMETHODNOTIMP(put_onreadystatechange, (VARIANT));
+ STDMETHODNOTIMP(get_onreadystatechange, (VARIANT*));
+ STDMETHODNOTIMP(put_onafterupdate, (VARIANT));
+ STDMETHODNOTIMP(get_onafterupdate, (VARIANT*));
+ STDMETHODNOTIMP(put_onrowexit, (VARIANT));
+ STDMETHODNOTIMP(get_onrowexit, (VARIANT*));
+ STDMETHODNOTIMP(put_onrowenter, (VARIANT));
+ STDMETHODNOTIMP(get_onrowenter, (VARIANT*));
+ STDMETHODNOTIMP(put_ondragstart, (VARIANT));
+ STDMETHODNOTIMP(get_ondragstart, (VARIANT*));
+ STDMETHODNOTIMP(put_onselectstart, (VARIANT));
+ STDMETHODNOTIMP(get_onselectstart, (VARIANT*));
+ STDMETHODNOTIMP(elementFromPoint, (long, long, IHTMLElement**));
+ STDMETHODNOTIMP(get_parentWindow, (IHTMLWindow2**));
+ STDMETHODNOTIMP(get_styleSheets, (IHTMLStyleSheetsCollection**));
+ STDMETHODNOTIMP(put_onbeforeupdate, (VARIANT));
+ STDMETHODNOTIMP(get_onbeforeupdate, (VARIANT*));
+ STDMETHODNOTIMP(put_onerrorupdate, (VARIANT));
+ STDMETHODNOTIMP(get_onerrorupdate, (VARIANT*));
+ STDMETHODNOTIMP(toString, (BSTR*));
+ STDMETHODNOTIMP(createStyleSheet, (BSTR, long, IHTMLStyleSheet**));
+ // IHTMLDocument4
+ STDMETHODNOTIMP(focus, ());
+ STDMETHODNOTIMP(hasFocus, (VARIANT_BOOL*));
+ STDMETHODNOTIMP(put_onselectionchange, (VARIANT));
+ STDMETHODNOTIMP(get_onselectionchange, (VARIANT*));
+ STDMETHODNOTIMP(get_namespaces, (IDispatch**));
+ STDMETHODNOTIMP(createDocumentFromUrl, (BSTR, BSTR, IHTMLDocument2**));
+ STDMETHODNOTIMP(put_media, (BSTR));
+ STDMETHODNOTIMP(get_media, (BSTR*));
+ STDMETHODNOTIMP(fireEvent, (BSTR, VARIANT*, VARIANT_BOOL*));
+ STDMETHODNOTIMP(createRenderStyle, (BSTR, IHTMLRenderStyle**));
+ STDMETHODNOTIMP(put_oncontrolselect, (VARIANT));
+ STDMETHODNOTIMP(get_oncontrolselect, (VARIANT*));
+ STDMETHODNOTIMP(get_URLUnencoded, (BSTR*));
+ // IHTMLEventObj
+ STDMETHODNOTIMP(get_srcElement, (IHTMLElement**))
+ STDMETHODNOTIMP(get_altKey, (VARIANT_BOOL*));
+ STDMETHODNOTIMP(get_ctrlKey, (VARIANT_BOOL*));
+ STDMETHODNOTIMP(get_shiftKey, (VARIANT_BOOL*));
+ STDMETHODNOTIMP(put_returnValue, (VARIANT));
+ STDMETHODNOTIMP(get_returnValue, (VARIANT*));
+ STDMETHODNOTIMP(put_cancelBubble, (VARIANT_BOOL));
+ STDMETHODNOTIMP(get_cancelBubble, (VARIANT_BOOL*));
+ STDMETHODNOTIMP(get_fromElement, (IHTMLElement**));
+ STDMETHODNOTIMP(get_toElement, (IHTMLElement**));
+ STDMETHODNOTIMP(put_keyCode, (long));
+ STDMETHODNOTIMP(get_keyCode, (long*));
+ STDMETHODNOTIMP(get_button, (long*));
+ STDMETHODNOTIMP(get_type, (BSTR*));
+ STDMETHODNOTIMP(get_qualifier, (BSTR*));
+ STDMETHODNOTIMP(get_reason, (long*));
+ STDMETHODNOTIMP(get_x, (long*));
+ STDMETHODNOTIMP(get_y, (long*));
+ STDMETHODNOTIMP(get_clientX, (long*));
+ STDMETHODNOTIMP(get_clientY, (long*));
+ STDMETHODNOTIMP(get_offsetX, (long*));
+ STDMETHODNOTIMP(get_offsetY, (long*));
+ STDMETHODNOTIMP(get_screenX, (long*));
+ STDMETHODNOTIMP(get_screenY, (long*));
+ STDMETHODNOTIMP(get_srcFilter, (IDispatch**));
+#undef STDMETHODNOTIMP
+};
+
+TEST(ComMessageEvent, WithSmartContainer) {
+ CComObject<MockSmartContainer>* container_obj = NULL;
+ CComObject<MockSmartContainer>::CreateInstance(&container_obj);
+ ScopedComPtr<IOleContainer> container(container_obj);
+ EXPECT_FALSE(!container);
+
+ CComObject<FriendlyComMessageEvent>* event_obj = NULL;
+ CComObject<FriendlyComMessageEvent>::CreateInstance(&event_obj);
+ ScopedComPtr<IUnknown> event_ref(event_obj);
+
+ bool succeeded = event_obj->Initialize(container, "hi",
+ "http://www.foo.com/", "message");
+ EXPECT_TRUE(succeeded);
+ EXPECT_FALSE(!event_obj->basic_event());
+
+ // Name handled natively by CF's ComMessageEvent.
+ DISPID dispid = -1;
+ LPOLESTR name = L"data";
+ HRESULT hr = event_obj->GetIDsOfNames(IID_IDispatch, &name, 1,
+ LOCALE_USER_DEFAULT, &dispid);
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ EXPECT_EQ(dispid, ComMessageEvent::DISPID_MESSAGE_EVENT_DATA);
+
+ // Name not handled by CF's ComMessageEvent.
+ dispid = -1;
+ name = L"nothandledatallbyanyone";
+ hr = event_obj->GetIDsOfNames(IID_IDispatch, &name, 1,
+ LOCALE_USER_DEFAULT, &dispid);
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ EXPECT_EQ(dispid, MockSmartContainer::kDispId);
+
+ // Invoke function handled by ComMessageEvent.
+ CComDispatchDriver dispatcher(event_obj);
+ CComVariant result;
+ hr = dispatcher.GetProperty(ComMessageEvent::DISPID_MESSAGE_EVENT_DATA,
+ &result);
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ EXPECT_EQ(result.vt, VT_BSTR);
+ EXPECT_EQ(wcscmp(result.bstrVal, L"hi"), 0);
+
+ // And now check passthrough.
+ result.Clear();
+ hr = dispatcher.GetProperty(MockSmartContainer::kDispId, &result);
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ EXPECT_EQ(result.vt, VT_I4);
+ EXPECT_EQ(result.lVal, MockSmartContainer::kResultValue);
+}
diff --git a/chrome_frame/test/data/CFInstance_basic_frame.html b/chrome_frame/test/data/CFInstance_basic_frame.html
new file mode 100644
index 0000000..11938ee
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_basic_frame.html
@@ -0,0 +1,8 @@
+<html>
+ <head>
+ <title></title>
+ </head>
+ <body>
+ <h1>do nothing</h1>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/CFInstance_basic_host.html b/chrome_frame/test/data/CFInstance_basic_host.html
new file mode 100644
index 0000000..6bad61c
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_basic_host.html
@@ -0,0 +1,55 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceBasic";
+ (function(){
+ try{
+ var cf = new CFInstance({
+ src: "CFInstance_basic_frame.html",
+ node: "toBeReplaced"
+ });
+
+ if (document.getElementById("parent") != cf.parentNode ) {
+ onFailure(testName, 1, "parent node mismatch");
+ return;
+ }
+
+ if (document.getElementById("prev").nextSibling != cf) {
+ onFailure(testName, 1, "sibling node mismatch");
+ return;
+ }
+
+ if (document.getElementById("after").previousSibling != cf) {
+ onFailure(testName, 1, "sibling node mismatch");
+ return;
+ }
+
+ onSuccess(testName, 1);
+
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ })();
+ </script>
+ <p>Tests ChromeFrame Navigation</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_default_ctor_host.html b/chrome_frame/test/data/CFInstance_default_ctor_host.html
new file mode 100644
index 0000000..744c6de
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_default_ctor_host.html
@@ -0,0 +1,45 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceDefaultCtor";
+ (function(){
+ try{
+ var cf = new CFInstance();
+ cf.src = "CFInstance_basic_frame.html";
+ var node = document.getElementById("toBeReplaced");
+ node.parentNode.replaceChild(cf, node);
+ var timer = setTimeout(function() {
+ onFailure(testName, 1, "CFInstance navigation timeout");
+ }, 15000);
+ cf.listen("load", function() {
+ clearTimeout(timer);
+ onSuccess(testName, 1);
+ });
+ cf.src = "CFInstance_basic_frame.html";
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ })();
+ </script>
+ <p>Tests Chrome Frame constructor without arguments</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_delay_host.html b/chrome_frame/test/data/CFInstance_delay_host.html
new file mode 100644
index 0000000..f6d9e02
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_delay_host.html
@@ -0,0 +1,47 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceDelay";
+ (function(){
+ try{
+ var cf = new CFInstance({
+ onload: function() {
+ onSuccess(testName, 1);
+ },
+ src: "CFInstance_basic_frame.html"
+ });
+
+ setTimeout(function() {
+ var replNode = document.getElementById("toBeReplaced");
+ // impedence matching between new and old CFInstance.js
+ var node = cf["plugin"] ? cf.plugin : cf;
+ replNode.parentNode.replaceChild(node, replNode);
+ }, 100);
+
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ })();
+ </script>
+ <p>Tests ChromeFrame Navigation when placed in the document on a delay</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_fallback_host.html b/chrome_frame/test/data/CFInstance_fallback_host.html
new file mode 100644
index 0000000..5939180
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_fallback_host.html
@@ -0,0 +1,44 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceFallback";
+ (function() {
+ try {
+ var cf = new CFInstance({
+ node: "toBeReplaced",
+ src: "CFInstance_basic_frame.html",
+ requirements: []
+ });
+
+ if (cf.tagName.toLowerCase() == "iframe") {
+ onSuccess(testName, 1);
+ } else {
+ onFailure(testName, 1, "expected tagName mismatch");
+ }
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ })();
+ </script>
+ <p>Tests ChromeFrame fallback</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_iframe_onload_host.html b/chrome_frame/test/data/CFInstance_iframe_onload_host.html
new file mode 100644
index 0000000..913c462
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_iframe_onload_host.html
@@ -0,0 +1,41 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceIfrOnload";
+ (function() {
+ try {
+ var cf = new CFInstance({
+ node: "toBeReplaced",
+ src: "CFInstance_basic_frame.html",
+ onload: function() {
+ onSuccess(testName, 1);
+ },
+ requirements: [] // always use an iframe
+ });
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ })();
+ </script>
+ <p>Tests CFInstance event handling on iframes</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_iframe_post_host.html b/chrome_frame/test/data/CFInstance_iframe_post_host.html
new file mode 100644
index 0000000..f503522
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_iframe_post_host.html
@@ -0,0 +1,50 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceIfrPost";
+ (function() {
+ if (!CFInstance.contentTests.postMessage()) {
+ onSuccess(testName, 1);
+ } else {
+ try {
+ var cf = new CFInstance({
+ node: "toBeReplaced",
+ src: "CFInstance_post_frame.html",
+ onload: function() {
+ cf.postMessage("howdy!");
+ },
+ onmessage: function(evt) {
+ if (evt.data == 'hola!') {
+ onSuccess(testName, 1);
+ }
+ },
+ requirements: [] // always use an iframe
+ });
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ }
+ })();
+ </script>
+ <p>Tests CFInstance event handling on iframes</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_no_src_host.html b/chrome_frame/test/data/CFInstance_no_src_host.html
new file mode 100644
index 0000000..379e26d
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_no_src_host.html
@@ -0,0 +1,43 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceNoSrc";
+ (function() {
+ try {
+ var cf = new CFInstance({
+ src: "",
+ node: "toBeReplaced"
+ });
+
+ // check that we got "src" set to "about:blank" by CFInstance
+ if (cf.src == "about:blank") {
+ onSuccess(testName, 1);
+ } else {
+ onFailure(testName, 1, "blank URL mismatch");
+ }
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ })();
+ </script>
+ <p>Tests ChromeFrame with blank src</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/CFInstance_post_frame.html b/chrome_frame/test/data/CFInstance_post_frame.html
new file mode 100644
index 0000000..8055cb2
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_post_frame.html
@@ -0,0 +1,26 @@
+<!-- saved from url=(0014)about:internet -->
+<html>
+<!-- This page is meant to load inside ChromeFrame -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ <script>
+ var cf = CFInstance;
+
+ cf.listen("load", function() {
+ cf.postMessage("hola!");
+ })
+ </script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <p>ChromeFrame + CFInstance PostMessage Test
+ <br>Test for PostMessage from the host browser to iframe and back</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/CFInstance_post_host.html b/chrome_frame/test/data/CFInstance_post_host.html
new file mode 100644
index 0000000..09402ac
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_post_host.html
@@ -0,0 +1,47 @@
+<!DOCTYPE html>
+
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstancePost";
+ (function() {
+ try {
+ var cf = new CFInstance({
+ node: "toBeReplaced",
+ src: "CFInstance_post_frame.html",
+ onload: function() {
+ cf.postMessage("howdy!");
+ },
+ onmessage: function(evt) {
+ if (evt.data == "hola!") {
+ onSuccess(testName, 1);
+ }
+ }
+ });
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: " + e);
+ }
+ })();
+ </script>
+ <p>Tests CFInstance event handling on iframes</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_rpc_frame.html b/chrome_frame/test/data/CFInstance_rpc_frame.html
new file mode 100644
index 0000000..a7dbfd7
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_rpc_frame.html
@@ -0,0 +1,27 @@
+<!DOCTYPE html>
+
+<html>
+<!-- This page is meant to load inside ChromeFrame -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ <script>
+ var cf = CFInstance;
+ cf.rpc.expose("rpcHandler", function(arg) {
+ cf.rpc.callRemote("handleCallback", ["hola!"]);
+ });
+ cf.rpc.init();
+ </script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <p>ChromeFrame + CFInstance PostMessage Test
+ <br>Test for PostMessage from the host browser to iframe and back</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/CFInstance_rpc_host.html b/chrome_frame/test/data/CFInstance_rpc_host.html
new file mode 100644
index 0000000..60680cf
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_rpc_host.html
@@ -0,0 +1,51 @@
+<!DOCTYPE html>
+
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceRPC";
+ (function() {
+ try {
+ var cf = new CFInstance({
+ node: "toBeReplaced",
+ src: "CFInstance_rpc_frame.html"
+ });
+
+ var handleCallback = function(arg) {
+ // alert(arg);
+ if (arg == "hola!") {
+ onSuccess(testName, 1);
+ }
+ };
+
+ cf.rpc.expose("handleCallback", handleCallback);
+ cf.rpc.init();
+
+ cf.rpc.callRemote("rpcHandler", ["whatcho talkin 'bout, willis!?"]);
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: " + e);
+ }
+ })();
+ </script>
+ <p>Tests CFInstance event handling on iframes</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_rpc_internal_frame.html b/chrome_frame/test/data/CFInstance_rpc_internal_frame.html
new file mode 100644
index 0000000..8208269
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_rpc_internal_frame.html
@@ -0,0 +1,28 @@
+<!DOCTYPE html>
+
+<html>
+<!-- This page is meant to load inside ChromeFrame -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ <script>
+ setTimeout(rpcCall, 10000);
+ function rpcCall() {
+ var cf = CFInstance;
+ cf.rpc.callRemote("callback");
+ cf.rpc.init();
+ }
+ </script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <p>ChromeFrame + CFInstance PostMessage Test
+ <br>Test for PostMessage from the host browser to iframe and back</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/CFInstance_rpc_internal_host.html b/chrome_frame/test/data/CFInstance_rpc_internal_host.html
new file mode 100644
index 0000000..f350871
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_rpc_internal_host.html
@@ -0,0 +1,45 @@
+<!DOCTYPE html>
+
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceRPCInternal";
+ (function() {
+ try {
+ var cf = new CFInstance({
+ node: "toBeReplaced",
+ src: "CFInstance_rpc_internal_frame.html"
+ });
+
+
+ cf.rpc.expose("callback", function(arg) {
+ onSuccess(testName, 1);
+ });
+ cf.rpc.init();
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: " + e);
+ }
+ })();
+ </script>
+ <p>Tests CFInstance event handling on iframes</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_singleton_frame.html b/chrome_frame/test/data/CFInstance_singleton_frame.html
new file mode 100644
index 0000000..1ab0302
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_singleton_frame.html
@@ -0,0 +1,20 @@
+<html>
+ <head>
+ <title>talk to me...</title>
+ <script type="text/javascript" src="CFInstance.js"></script>
+ <script type="text/javascript">
+ CFInstance.listen("load", function() {
+ document.body.innerHTML = "sending 'foo'";
+ CFInstance.postMessage("foo");
+ document.body.innerHTML = "...waiting...";
+ });
+ CFInstance.listen("message", function(evt) {
+ document.body.innerHTML = "sending 'baz'";
+ CFInstance.postMessage("baz");
+ });
+ </script>
+ </head>
+ <body>
+ <h1>sends a message to the parent</h1>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/CFInstance_singleton_host.html b/chrome_frame/test/data/CFInstance_singleton_host.html
new file mode 100644
index 0000000..3eda108
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_singleton_host.html
@@ -0,0 +1,44 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript" src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="toBeReplaced">
+ fallback content goes here
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceSingleton";
+ (function() {
+ try{
+ var cf = new CFInstance({
+ src: "CFInstance_singleton_frame.html",
+ node: "toBeReplaced"
+ });
+
+ // test a call/response set of actions driven by the CF content
+ cf.listen("message", function(evt) {
+ if (evt.data == "foo") {
+ cf.postMessage("bar");
+ } else if(evt.data == "baz") {
+ onSuccess(testName, 1);
+ }
+ });
+
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: " + e);
+ }
+ })();
+ </script>
+ <p>Tests ChromeFrame Navigation</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/CFInstance_zero_size_host.html b/chrome_frame/test/data/CFInstance_zero_size_host.html
new file mode 100644
index 0000000..94b35c0
--- /dev/null
+++ b/chrome_frame/test/data/CFInstance_zero_size_host.html
@@ -0,0 +1,41 @@
+<html>
+ <!-- This page is meant to loaded inside the host browser (IE, FF, etc.) -->
+ <head>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+ <script type="text/javascript"
+ src="CFInstance.js"></script>
+ </head>
+
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <div id="parent">
+ <div id="prev">before</div><div id="toBeReplaced">
+ fallback content goes here
+ </div><div id="after">after</div>
+ </div>
+ <script type="text/javascript">
+ var testName = "CFInstanceZeroSize";
+ (function() {
+ try {
+ var cf = new CFInstance({
+ node: "toBeReplaced",
+ src: "CFInstance_basic_frame.html",
+ cssText: "width: 0px; height: 0px;",
+ onload: function() {
+ onSuccess(testName, 1);
+ }
+ });
+ } catch (e) {
+ onFailure(testName, 1,
+ "CFInstance constructor failed with error: "+e);
+ }
+ })();
+ </script>
+ <p>Tests CFInstance event handling on iframes</p>
+ </body>
+</html>
+
diff --git a/chrome_frame/test/data/back_to_ie.html b/chrome_frame/test/data/back_to_ie.html
new file mode 100644
index 0000000..ad4b61d
--- /dev/null
+++ b/chrome_frame/test/data/back_to_ie.html
@@ -0,0 +1,21 @@
+<html>
+ <head><title>Back to IE</title>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+
+ <script type="text/javascript">
+ function test() {
+ var test_name = 'navigate_out';
+ if (isRunningInMSIE()) {
+ onSuccess(test_name, 1);
+ } else {
+ onFailure(test_name, 1, 'Failed');
+ }
+ }
+ </script>
+ </head>
+ <body onLoad="test();">
+ <h2>Redirected!</h2>
+ <p>This page should have loaded in IE</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/cf_protocol.html b/chrome_frame/test/data/cf_protocol.html
new file mode 100644
index 0000000..0036555
--- /dev/null
+++ b/chrome_frame/test/data/cf_protocol.html
@@ -0,0 +1,20 @@
+<html>
+ <head><title>cf: protocol test</title>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+
+ <script type="text/javascript">
+ function test() {
+ if (isRunningInMSIE()) {
+ reloadUsingCFProtocol();
+ } else {
+ onSuccess("chrome_frame_protocol", 1);
+ }
+ }
+ </script>
+ </head>
+ <body onLoad="setTimeout(test, 100);">
+ <h2>Prepare to be redirected!</h2>
+ <p>Redirects the same page to its 'cf:' version</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/chrome_frame_mime_filter_test.html b/chrome_frame/test/data/chrome_frame_mime_filter_test.html
new file mode 100644
index 0000000..7520758
--- /dev/null
+++ b/chrome_frame/test/data/chrome_frame_mime_filter_test.html
@@ -0,0 +1,32 @@
+<!-- saved from url=(0014)about:internet -->
+<!-- Note that for the above Mark of the Web to work, the comment must
+ be followed by a CR/LF ending, so please do not change the line endings. -->
+<html>
+<!-- This page is meant to load inside a host browser like IE/FF -->
+<head>
+<meta http-equiv="X-UA-Compatible" content="chrome=1"/>
+<script type="text/javascript" src="chrome_frame_tester_helpers.js"></script>
+<script type="text/javascript">
+
+function TestIfRunningInChrome() {
+ var is_chrome = /chrome/.test(navigator.userAgent.toLowerCase());
+ if (is_chrome) {
+ onSuccess("MIMEFilter", "MIME filter worked!");
+ } else {
+ onFailure("MIMEFilter", "MIME filter failed :-(",
+ "User agent = " + navigator.userAgent.toLowerCase());
+ }
+}
+
+</script>
+</head>
+
+<body onload="TestIfRunningInChrome();">
+<div id="statusPanel" style="border: 1px solid red; width: 100%">
+Test running....
+</div>
+
+<p>
+
+</body>
+</html> \ No newline at end of file
diff --git a/chrome_frame/test/data/chrome_frame_resize.html b/chrome_frame/test/data/chrome_frame_resize.html
new file mode 100644
index 0000000..afba53b
--- /dev/null
+++ b/chrome_frame/test/data/chrome_frame_resize.html
@@ -0,0 +1,138 @@
+<!-- saved from url=(0014)about:internet -->
+<!-- Please preserve the CR/LF at the end of the previous line. -->
+<html>
+<!-- This page is meant to load inside the host browser like IE/FF -->
+<head>
+<script type="text/javascript" src="chrome_frame_tester_helpers.js"></script>
+<script type="text/javascript">
+function onLoad() {
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.onmessage = OnChromeFrameResize;
+ setTimeout(NavigateToURL, 100);
+}
+
+function NavigateToURL() {
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.src = "chrome_frame_resize_hosted.html";
+ setTimeout(CheckIfNavigationFailed, 15000);
+}
+
+var navigation_success = false;
+
+function CheckIfNavigationFailed() {
+ if (!navigation_success) {
+ onFailure("Resize", 1, "ChromeFrame Navigation failed");
+ }
+}
+
+function OnNavigationSucceeded() {
+ navigation_success = true;
+ appendStatus("ChromeFrame hosted page loaded, beginning test...");
+ setTimeout(ResizeChromeFrame, 100);
+}
+
+var resize_step = 0;
+
+function ResizeChromeFrame() {
+ var chromeFrame = GetChromeFrame();
+
+ if (resize_step == 0) {
+ appendStatus("Setting chromeFrame to 100x100");
+ resize_step = 1;
+ chromeFrame.width = 100;
+ setTimeout("OnResizeFailure(0)", 2000);
+ } else if (resize_step == 1) {
+ resize_step = 2;
+ chromeFrame.height = 100;
+ setTimeout("OnResizeFailure(1)", 2000);
+ } else if (resize_step == 2) {
+ appendStatus("Setting chromeFrame to 10x10");
+ resize_step = 3;
+ chromeFrame.width = 10;
+ setTimeout("OnResizeFailure(0)", 2000);
+ } else if (resize_step == 3) {
+ resize_step = 4;
+ chromeFrame.height = 10;
+ setTimeout("OnResizeFailure(1)", 2000);
+ }
+
+ // Note that setting the ChromeFrame window to 0x0 (or < 2x2 if we have the
+ // WS_BORDER style defined on our window) currently results
+ // in a check failure from the child chrome.exe process.
+ // TODO(robertshield): Figure out why and fix it.
+}
+
+var resize_step_received = 0;
+
+function OnChromeFrameResize(evt) {
+ resize_step_received++;
+ appendStatus("ChromeFrame resized: " + evt.data + "step=" +
+ resize_step_received);
+
+ if (resize_step == 4) {
+ onSuccess("Resize", 1);
+ } else {
+ setTimeout(ResizeChromeFrame, 100);
+ }
+}
+
+function OnResizeFailure(step) {
+ // It turns out that the hosted page gets two calls to onresize()
+ // every time a single size parameter (i.e. width OR height) is changed.
+ // As such this check doesn't quite guarantee success, but if it fails,
+ // then we should fail the unit test.
+ if (step >= resize_step_received) {
+ onFailure("Resize", 1, "Did not receive resize reply back from frame.");
+ }
+}
+
+function GetChromeFrame() {
+ return window.document.ChromeFrame;
+}
+
+var debug_counter = 0;
+
+function DebugResizeChromeFrame(delta) {
+ var chromeFrame = GetChromeFrame();
+ var newWidth = chromeFrame.clientWidth + delta;
+ var newHeight = chromeFrame.clientHeight + delta;
+
+ appendStatus(debug_counter + ". DEBUG resizing CF to (" + newWidth + "," +
+ newHeight + ")");
+
+ debug_counter++;
+
+ chromeFrame.width = newWidth;
+ chromeFrame.height = newHeight;
+}
+
+</script>
+</head>
+
+<body onload="onLoad();">
+<div id="description" style="border: 2px solid black; width: 100%">
+ Test for resizing the chrome frame control.
+</div>
+<div id="statusPanel" style="border: 1px solid red; width: 100%">
+Test running....
+</div>
+
+<object id="ChromeFrame" codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A"
+ style="border: 1px solid blue">
+ <param name="onload" value="return OnNavigationSucceeded();" />
+ <embed id="ChromeFramePlugin" name="ChromeFrame"
+ onload="return OnNavigationSucceeded();"
+ type="application/chromeframe"
+ style="border: 1px solid green">
+ </embed>
+</object>
+<br />
+<br />
+
+<button onclick="javascript:DebugResizeChromeFrame(20);">Bigger</button>
+<button onclick="javascript:DebugResizeChromeFrame(-20);">Smaller</button>
+
+</body>
+</html>
+
diff --git a/chrome_frame/test/data/chrome_frame_resize_hosted.html b/chrome_frame/test/data/chrome_frame_resize_hosted.html
new file mode 100644
index 0000000..95528ec
--- /dev/null
+++ b/chrome_frame/test/data/chrome_frame_resize_hosted.html
@@ -0,0 +1,48 @@
+<html>
+<!-- This page is meant to load inside ChromeFrame -->
+<head>
+
+<script type="text/javascript" src="chrome_frame_tester_helpers.js"></script>
+<script type="text/javascript">
+
+function onLoad() {
+ var host = window.externalHost;
+ if (host) {
+ host.postMessage(
+ "ChromeFrame navigated to: " + window.location);
+ } else {
+ appendStatus("Running in non-hosted mode");
+ }
+}
+
+var resize_event_counter = 0;
+
+function OnResizeEvent() {
+ width = window.innerWidth;
+ height = window.innerHeight;
+
+ appendStatus(resize_event_counter + ". Resized to (" + width +
+ "," + height + ")");
+
+ var host = window.externalHost;
+ if (host) {
+ host.postMessage(
+ resize_event_counter + ":(" + width + "," + height + ")");
+ } else {
+ appendStatus("window.externalHost is null!");
+ }
+}
+</script>
+</head>
+
+<body onload="onLoad();" bgcolor="#999999" onresize="OnResizeEvent();">
+<div id="description" style="border: 2px solid black; width: 100%">
+ Hosted resize test component.
+</div>
+
+<div id="statusPanel" style="border: 1px solid red; width: 100%">
+Test running....
+</div>
+
+</body>
+</html>
diff --git a/chrome_frame/test/data/chrome_frame_tester_helpers.js b/chrome_frame/test/data/chrome_frame_tester_helpers.js
new file mode 100644
index 0000000..1c914ee
--- /dev/null
+++ b/chrome_frame/test/data/chrome_frame_tester_helpers.js
@@ -0,0 +1,142 @@
+//
+// This script provides some mechanics for testing ChromeFrame
+//
+function onSuccess(name, id) {
+ onFinished(name, id, "OK");
+}
+
+function onFailure(name, id, status) {
+ onFinished(name, id, status);
+}
+
+function getXHRObject(){
+ var XMLHTTP_PROGIDS = ['Msxml2.XMLHTTP', 'Microsoft.XMLHTTP',
+ 'Msxml2.XMLHTTP.4.0'];
+ var http = null;
+ try {
+ http = new XMLHttpRequest();
+ } catch(e) {
+ }
+
+ if (http)
+ return http;
+
+ for (var i = 0; i < 3; ++i) {
+ var progid = XMLHTTP_PROGIDS[i];
+ try {
+ http = new ActiveXObject(progid);
+ } catch(e) {
+ }
+
+ if (http)
+ break;
+ }
+ return http;
+}
+
+var reportURL = "/writefile/";
+
+function shutdownServer() {
+ var xhr = getXHRObject();
+ if(!xhr)
+ return;
+
+ xhr.open("POST", "/kill", false);
+ try {
+ xhr.send(null);
+ } catch(e) {
+ appendStatus("XHR send failed. Error: " + e.description);
+ }
+}
+
+// Optionally send the server a notification that onload was fired.
+// To be called from within window.onload.
+function sendOnLoadEvent() {
+ writeToServer("OnLoadEvent", "loaded");
+}
+
+function writeToServer(name, result) {
+ var xhr = getXHRObject();
+ if(!xhr)
+ return;
+
+ // synchronously POST the results
+ xhr.open("POST", reportURL + name, false);
+ xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest");
+ try {
+ xhr.send(result);
+ } catch(e) {
+ appendStatus("XHR send failed. Error: " + e.description);
+ }
+}
+
+function postResult(name, result) {
+ writeToServer(name, result);
+ // NOTE:
+ // not watching for failure or return status issues. What should we do here?
+ shutdownServer();
+}
+
+// Finish running a test by setting the status
+// and the cookie.
+function onFinished(name, id, result) {
+ appendStatus(result);
+
+ // set a cookie to report the results...
+ var cookie = name + "." + id + ".status=" + result + "; path=/";
+ document.cookie = cookie;
+
+ // ...and POST the status back to the server
+ postResult(name, result);
+}
+
+function appendStatus(message) {
+ var statusPanel = document.getElementById("statusPanel");
+ if (statusPanel) {
+ statusPanel.innerHTML += '<BR>' + message;
+ }
+}
+
+function readCookie(name) {
+ var cookie_name = name + "=";
+ var ca = document.cookie.split(';');
+
+ for(var i = 0 ; i < ca.length ; i++) {
+ var c = ca[i];
+ while (c.charAt(0) == ' ') {
+ c = c.substring(1,c.length);
+ }
+ if (c.indexOf(cookie_name) == 0) {
+ return c.substring(cookie_name.length, c.length);
+ }
+ }
+ return null;
+}
+
+function createCookie(name,value,days) {
+ var expires = "";
+ if (days) {
+ var date = new Date();
+ date.setTime(date.getTime() + (days * 24 * 60 * 60 * 1000));
+ expires = "; expires=" + date.toGMTString();
+ }
+ document.cookie = name+"="+value+expires+"; path=/";
+}
+
+function eraseCookie(name) {
+ createCookie(name, "", -1);
+}
+
+function isRunningInMSIE() {
+ if (/MSIE (\d+\.\d+);/.test(navigator.userAgent))
+ return true;
+
+ return false;
+}
+
+function reloadUsingCFProtocol() {
+ var redirect_location = "cf:";
+ redirect_location += window.location;
+ window.location = redirect_location;
+}
+
diff --git a/chrome_frame/test/data/event_listener.html b/chrome_frame/test/data/event_listener.html
new file mode 100644
index 0000000..6fbd158
--- /dev/null
+++ b/chrome_frame/test/data/event_listener.html
@@ -0,0 +1,42 @@
+<html>
+<!-- This page is meant to load inside the host browser like IE/FF -->
+<head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js"></script>
+ <script language="javascript">
+ var g_test_name = 'EventListener';
+
+ function onChromeFrameLoaded() {
+ appendStatus('Chrome frame loaded.');
+ onSuccess(g_test_name, 1);
+ }
+
+ function onEventNotFired() {
+ onFailure(g_test_name, 1, 'Did not receive onload event');
+ }
+
+ function onDocumentLoad() {
+ appendStatus('document loaded');
+ var cf = getCf();
+ cf.addEventListener("load", onChromeFrameLoaded, false);
+ setTimeout(onEventNotFired, 10000)
+ cf.src = "CFInstance_basic_frame.html";
+ }
+
+ function getCf() {
+ return window.document.ChromeFrame;
+ }
+ </script>
+</head>
+<body onload="onDocumentLoad();">
+ <object id="ChromeFrame" width="500" height ="300"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <embed id="ChromeFramePlugin" name="ChromeFrame" width="500"
+ height="500" type="application/chromeframe">
+ </embed>
+ </object>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+</body>
+</html>
diff --git a/chrome_frame/test/data/iframe_basic_host.html b/chrome_frame/test/data/iframe_basic_host.html
new file mode 100644
index 0000000..f9a4c0c
--- /dev/null
+++ b/chrome_frame/test/data/iframe_basic_host.html
@@ -0,0 +1,13 @@
+<!-- saved from url=(0014)about:internet -->
+<html>
+<!-- This page is meant to load inside the host browser like IE/FF -->
+<head></head>
+<body>
+ <iframe src ="postmessage_basic_host.html" width="50%" height="600">
+ </iframe>
+
+ <br>
+ <br>
+ Test for ChromeFrame inside iframe
+</body>
+</html>
diff --git a/chrome_frame/test/data/in_head.html b/chrome_frame/test/data/in_head.html
new file mode 100644
index 0000000..a093c71
--- /dev/null
+++ b/chrome_frame/test/data/in_head.html
@@ -0,0 +1,62 @@
+<html>
+ <!-- This page is meant to load inside the host browser like IE/FF -->
+ <head><title>Initialize hidden chrome frame</title>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ var g_failure_timeout = null;
+ var g_test_id = 1;
+ var g_test_name = 'InHead';
+ var g_cf3_loaded = false;
+
+ function OnNavigationFailed() {
+ onFailure(g_test_name, g_test_id, 'ChromeFrame Navigation failed');
+ }
+
+ function OnObjectFocusFailed() {
+ appendStatus('chrome frame focus failed');
+ onFailure(g_test_name, g_test_id, 'Embed in head test failed');
+ }
+
+ function OnFrameMessage(evt) {
+ appendStatus('Chrome frame visible and focused');
+ if (evt.data == 'btnOnFocus') {
+ window.clearTimeout(g_failure_timeout);
+ g_failure_timeout = null;
+ appendStatus('Chrome frame visible and focused');
+
+ onSuccess(g_test_name, g_test_id);
+ }
+ }
+
+ function OnFrameLoad() {
+ document.ChromeFrame.focus();
+ }
+
+ function OnLoad() {
+ g_failure_timeout = window.setTimeout(OnObjectFocusFailed, 10000);
+ }
+ </script>
+ <object id="ChromeFrame" width="300" height="80" tabindex="0"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="src" value="simple_object_focus_cf.html">
+ <param name="onload" value="OnFrameLoad();">
+ <param name="onloaderror" value="OnNavigationFailed();">
+ <param name="onmessage" value="OnFrameMessage(arguments[0]);">
+ <embed width="300" height="80" name="ChromeFrame"
+ type="application/chromeframe"
+ src="simple_object_focus_cf.html"
+ onload="OnFrameLoad();"
+ onloaderror="OnNavigationFailed();"
+ onmessage="OnFrameMessage(arguments[0]);">
+ </embed>
+ </object>
+ </head>
+ <body onload = "OnLoad();">
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+ <div id = "dummy"> </div>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/initialize_hidden.html b/chrome_frame/test/data/initialize_hidden.html
new file mode 100644
index 0000000..2da0917
--- /dev/null
+++ b/chrome_frame/test/data/initialize_hidden.html
@@ -0,0 +1,120 @@
+<html>
+ <!-- This page is meant to load inside the host browser like IE/FF -->
+ <head><title>Initialize hidden chrome frame</title>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ var g_failure_timeout = null;
+ var g_test_id = 1;
+ var g_test_name = 'InitializeHidden';
+ var g_cf3_loaded = false;
+
+ function OnNavigationFailed() {
+ onFailure(g_test_name, g_test_id, 'ChromeFrame Navigation failed');
+ }
+
+ function OnObjectFocusFailed() {
+ appendStatus('chrome frame focus failed');
+ onFailure(g_test_name, g_test_id, 'Visibility test failed');
+ }
+
+ function OnCF1Loaded() {
+ appendStatus('Chrome frame 1 loaded, not visible yet.');
+ try {
+ // Make chrome frame visible
+ var cf1 = document.getElementById('CFDiv1');
+ cf1.style.visibility = 'visible';
+ appendStatus('Chrome frame 1 visibility - ' + cf1.style.visibility);
+ // Set focus to chrome frame. This should set focus to the
+ // first element inside the frame, which a script inside the
+ // page will detect and notify us back by sending us a message.
+ document.ChromeFrame1.focus();
+ g_failure_timeout = window.setTimeout(OnObjectFocusFailed, 10000);
+ } catch(e) {
+ appendStatus('Error setting focus to CF1. Error: ' + e.description);
+ onFailure(g_test_name, g_test_id, 'CF1 focus() error');
+ }
+ }
+
+ function OnCF1Message(evt) {
+ if (evt.data == 'btnOnFocus') {
+ window.clearTimeout(g_failure_timeout);
+ g_failure_timeout = null;
+ appendStatus('CF1 visible and focused');
+
+ // Now make second chrome frame instance visible
+ document.getElementById('CFDiv2').style.display = 'block';
+ appendStatus('Chrome frame 2 visible, should start loading now');
+ g_failure_timeout = window.setTimeout(OnObjectFocusFailed, 10000);
+ }
+ }
+
+ function OnCF2Loaded() {
+ appendStatus('Chrome frame 2 loaded');
+ try {
+ // Set focus to chrome frame. This should set focus to the
+ // first element inside the frame, which a script inside the
+ // page will detect and notify us back by sending us a message.
+ // We set focus on a timeout as on IE it takes a while for the window
+ // to become visible.
+ setTimeout(SetFocusToCF2, 100);
+ } catch(e) {
+ appendStatus('Error setting focus to CF2. Error: ' + e.description);
+ onFailure(g_test_name, g_test_id, 'CF2 focus() error');
+ }
+ }
+
+ function SetFocusToCF2() {
+ document.ChromeFrame2.focus();
+ }
+
+ function OnCF2Message(evt) {
+ if (evt.data == 'btnOnFocus') {
+ window.clearTimeout(g_failure_timeout);
+ g_failure_timeout = null;
+ appendStatus('CF2 visible and focused');
+ onSuccess(g_test_name, g_test_id);
+ }
+ }
+ </script>
+ </head>
+ <body>
+ <div id="CFDiv1" style="visibility: hidden;">
+ <object id="ChromeFrame1" width="300" height="80" tabindex="0"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="src" value="simple_object_focus_cf.html">
+ <param name="onload" value="OnCF1Loaded();">
+ <param name="onloaderror" value="OnNavigationFailed();">
+ <param name="onmessage" value="OnCF1Message(arguments[0]);">
+ <embed width="300" height="80" name="ChromeFrame1"
+ type="application/chromeframe"
+ src="simple_object_focus_cf.html"
+ onload="OnCF1Loaded();"
+ onloaderror="OnNavigationFailed();"
+ onmessage="OnCF1Message(arguments[0]);">
+ </embed>
+ </object>
+ </div>
+ <div id="CFDiv2" style="display: none;">
+ <object id="ChromeFrame2" width="300" height="80" tabindex="1"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="src" value="simple_object_focus_cf.html">
+ <param name="onload" value="OnCF2Loaded();">
+ <param name="onloaderror" value="OnNavigationFailed();">
+ <param name="onmessage" value="OnCF2Message(arguments[0]);">
+ <embed width="300" height="80" name="ChromeFrame2"
+ type="application/chromeframe"
+ src="simple_object_focus_cf.html"
+ onload="OnCF2Loaded();"
+ onloaderror="OnNavigationFailed();"
+ onmessage="OnCF2Message(arguments[0]);">
+ </embed>
+ </object>
+ </div>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/meta_tag.html b/chrome_frame/test/data/meta_tag.html
new file mode 100644
index 0000000..9b17406
--- /dev/null
+++ b/chrome_frame/test/data/meta_tag.html
@@ -0,0 +1,21 @@
+<html>
+ <head>
+ <meta http-equiv="x-ua-compatible" content="chrome=1" />
+ <title>Load chrome frame using meta tag</title>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+
+ <script type="text/javascript">
+ function test() {
+ if (isRunningInMSIE()) {
+ onFailure("meta_tag", 1, "Failed");
+ } else {
+ onSuccess("meta_tag", 1);
+ }
+ }
+ </script>
+ </head>
+ <body onLoad="setTimeout(test, 100);">
+ chrome trame in tab mode
+ </body>
+</html>
diff --git a/chrome_frame/test/data/navigate_out.html b/chrome_frame/test/data/navigate_out.html
new file mode 100644
index 0000000..7b910b4
--- /dev/null
+++ b/chrome_frame/test/data/navigate_out.html
@@ -0,0 +1,20 @@
+<html>
+ <head><title>Test to make sure that navigations sent back to IE</title>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+
+ <script type="text/javascript">
+ function test() {
+ if (isRunningInMSIE()) {
+ reloadUsingCFProtocol();
+ } else {
+ window.location = "back_to_ie.html";
+ }
+ }
+ </script>
+ </head>
+ <body onLoad="setTimeout(test, 100);">
+ <h2>Prepare to be redirected!</h2>
+ <p>Redirects the same page to its 'cf:' version and </p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/navigateurl_absolute_host.html b/chrome_frame/test/data/navigateurl_absolute_host.html
new file mode 100644
index 0000000..03e1de2
--- /dev/null
+++ b/chrome_frame/test/data/navigateurl_absolute_host.html
@@ -0,0 +1,64 @@
+<html>
+<!-- This page is meant to load inside the host browser like IE/FF -->
+ <head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ function onLoad() {
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.onloaderror = OnNavigationFailed;
+ setTimeout(NavigateToURL, 100);
+ }
+
+ function NavigateToURL() {
+ var frame_location = new String(window.location);
+ frame_location = frame_location.replace(
+ /navigateurl_absolute_host.html/, "navigateurl_basic_frame.html");
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.src = frame_location;
+ setTimeout(OnNavigationTimeout, 10000);
+ }
+
+ var navigation_success = 0;
+
+ function OnNavigationFailed(msg) {
+ if (!navigation_success) {
+ onFailure("NavigateURL", 1, 'ChromeFrame Navigation failed: ' + msg);
+ }
+ }
+
+ function OnNavigationTimeout() {
+ OnNavigationFailed('TIMEOUT');
+ }
+
+ function OnChromeFrameLoaded() {
+ navigation_success = 1;
+ onSuccess("NavigateURL", 1);
+ }
+
+ function GetChromeFrame() {
+ return window.document.ChromeFrame;
+ }
+ </script>
+ </head>
+
+ <body onload="onLoad();">
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+ <object id="ChromeFrame" width="500" height="500"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="onload" value="return OnChromeFrameLoaded();">
+ <embed id="ChromeFramePlugin" width="500" height="500"
+ name="ChromeFrame" onload="return OnChromeFrameLoaded();"
+ type="application/chromeframe">
+ </embed>
+ </OBJECT>
+ <br />
+ <br />
+
+ <p>Tests ChromeFrame Navigation</p>
+
+ </body>
+</html>
diff --git a/chrome_frame/test/data/navigateurl_basic_frame.html b/chrome_frame/test/data/navigateurl_basic_frame.html
new file mode 100644
index 0000000..4d99b43
--- /dev/null
+++ b/chrome_frame/test/data/navigateurl_basic_frame.html
@@ -0,0 +1,12 @@
+<html>
+<!-- This page is meant to load inside ChromeFrame -->
+
+<body>
+<div id="statusPanel" style="border: 1px solid red; width: 100%">
+Test running....
+</div>
+
+<p>ChromeFrame NavigateURL Test<br>
+Tests ChromeFrame Navigation</p>
+</body>
+</html>
diff --git a/chrome_frame/test/data/navigateurl_relative_host.html b/chrome_frame/test/data/navigateurl_relative_host.html
new file mode 100644
index 0000000..06ec63e
--- /dev/null
+++ b/chrome_frame/test/data/navigateurl_relative_host.html
@@ -0,0 +1,60 @@
+<html>
+ <!-- This page is meant to load inside the host browser like IE/FF -->
+ <head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ function onLoad() {
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.onloaderror = OnNavigationFailed;
+ setTimeout(NavigateToURL, 100);
+ }
+
+ function NavigateToURL() {
+ var chromeFrame = GetChromeFrame();
+ chromeFrame.src = "navigateurl_basic_frame.html";
+ setTimeout(OnNavigationTimeout, 10000);
+ }
+
+ var navigation_complete = 0;
+
+ function OnNavigationFailed(msg) {
+ if (!navigation_complete) {
+ onFailure("NavigateURL", 1, 'ChromeFrame Navigation failed: ' + msg);
+ }
+ }
+
+ function OnNavigationTimeout() {
+ OnNavigationFailed('TIMEOUT');
+ }
+
+ function OnChromeFrameLoaded() {
+ navigation_success = 1;
+ onSuccess("NavigateURL", 1);
+ }
+
+ function GetChromeFrame() {
+ return window.document.ChromeFrame;
+ }
+ </script>
+ </head>
+
+ <body onload="onLoad();">
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+ <object id="ChromeFrame" width="500" height="500"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="onload" value="return OnChromeFrameLoaded();">
+ <embed id="ChromeFramePlugin" width="500" height="500"
+ name="ChromeFrame" onload="return OnChromeFrameLoaded();"
+ type="application/chromeframe">
+ </embed>
+ </OBJECT>
+ <br />
+ <br />
+
+ <p>Tests ChromeFrame Navigation</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/persistent_cookie_test_page.html b/chrome_frame/test/data/persistent_cookie_test_page.html
new file mode 100644
index 0000000..ea56262
--- /dev/null
+++ b/chrome_frame/test/data/persistent_cookie_test_page.html
@@ -0,0 +1,36 @@
+<html>
+ <head><title>Persistent host browser chrome frame cookie test</title>
+ <script type="text/javascript"
+ src="chrome_frame_tester_helpers.js"></script>
+
+ <script type="text/javascript">
+ function validatePersistentCookie() {
+ if (readCookie("PersistentCookie1") != "Cookie1" ||
+ readCookie("PersistentCookie2") != "Cookie2") {
+ onFailure("PersistentCookieTest", 1, "Failed");
+ } else {
+ onSuccess("PersistentCookieTest", 1);
+ }
+ eraseCookie("PersistentCookie1");
+ eraseCookie("PersistentCookie2");
+ }
+
+ function setPersistentCookieAndRedirect() {
+ if (isRunningInMSIE()) {
+ eraseCookie("PersistentCookie1");
+ eraseCookie("PersistentCookie2");
+ createCookie("PersistentCookie1", "Cookie1", 365);
+ createCookie("PersistentCookie2", "Cookie2", 365);
+ reloadUsingCFProtocol();
+ } else {
+ validatePersistentCookie();
+ }
+ }
+ </script>
+ </head>
+ <body onLoad="setTimeout(setPersistentCookieAndRedirect, 100);">
+ <h2>Prepare to be redirected!</h2>
+ <p>Sets two persistent cookies in the host and redirects ChromeFrame <br />
+ to the same page </p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/postmessage_basic_frame.html b/chrome_frame/test/data/postmessage_basic_frame.html
new file mode 100644
index 0000000..76f8cb3
--- /dev/null
+++ b/chrome_frame/test/data/postmessage_basic_frame.html
@@ -0,0 +1,27 @@
+<!-- saved from url=(0014)about:internet -->
+<html>
+<!-- This page is meant to load inside ChromeFrame -->
+ <head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ function OnLoad() {
+ externalHost.onmessage = OnHostMessage;
+ }
+
+ function OnHostMessage(evt) {
+ appendStatus('Host message: ' + evt.data);
+ externalHost.postMessage("Hello from ChromeFrame");
+ }
+ </script>
+ </head>
+
+ <body onload="OnLoad();">
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <p>ChromeFrame PostMessage Test
+ <br>Test for PostMessage from the host browser to ChromeFrame and back</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/postmessage_basic_host.html b/chrome_frame/test/data/postmessage_basic_host.html
new file mode 100644
index 0000000..e5ecef9
--- /dev/null
+++ b/chrome_frame/test/data/postmessage_basic_host.html
@@ -0,0 +1,69 @@
+<!-- saved from url=(0014)about:internet -->
+<html>
+<!-- This page is meant to load inside the host browser like IE/FF -->
+ <head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ var post_message_reply_received = 0;
+
+ function onChromeFrameLoaded() {
+ appendStatus('Chrome frame loaded...');
+ document.ChromeFrame.postMessage('Hello from host');
+ setTimeout(onPostMessageFailure, 10000);
+ }
+
+ function onNavigationFailed(msg) {
+ onFailure('PostMessage', 1, 'ChromeFrame Navigation failed: ' + msg);
+ }
+
+ function onChromeFrameMessage(evt) {
+ try {
+ var d = new String(evt.data);
+ appendStatus('Message: ' + d);
+ if (d == 'Hello from ChromeFrame') {
+ post_message_reply_received = 1;
+ onSuccess('PostMessage', 1);
+ } else {
+ onFailure('PostMessage', 1, 'unexpected data');
+ }
+ } catch (e) {
+ onFailure('PostMessage', 1, 'exception in onChromeFrameMessage');
+ }
+ }
+
+ function onPostMessageFailure() {
+ if (!post_message_reply_received) {
+ onFailure('PostMessage', 1, 'Did not receive reply back from frame');
+ }
+ }
+ </script>
+ </head>
+
+ <body>
+ <object id="ChromeFrame" width="500" height ="300"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="src" value="postmessage_basic_frame.html">
+ <param name="onload" value="onChromeFrameLoaded();">
+ <param name="onloaderror" value="onNavigationFailed();">
+ <param name="onmessage" value="onChromeFrameMessage(arguments[0]);">
+ <embed id="ChromeFramePlugin" name="ChromeFrame"
+ width="500" height="500"
+ src="postmessage_basic_frame.html"
+ type="application/chromeframe"
+ onload="onChromeFrameLoaded();"
+ onloaderror="onNavigationFailed();"
+ onmessage="onChromeFrameMessage(arguments[0]);">
+ </embed>
+ </object>
+ <br>
+ <br>
+ <p>Test for PostMessage from the host browser to ChromeFrame and back</p>
+ <button onclick="document.ChromeFrame.postMessage('Message from button');">
+ Send message to frame</button>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/privileged_apis_frame.html b/chrome_frame/test/data/privileged_apis_frame.html
new file mode 100644
index 0000000..9e51152
--- /dev/null
+++ b/chrome_frame/test/data/privileged_apis_frame.html
@@ -0,0 +1,33 @@
+<!-- saved from url=(0014)about:internet -->
+<html>
+<!-- This page is meant to load inside ChromeFrame -->
+ <head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ function OnLoad() {
+ externalHost.onmessage = OnHostMessage;
+ }
+
+ function OnHostMessage(evt) {
+ // Any time we receive a message, we reflect it back both
+ // with a nonsensical target, and with "*".
+ appendStatus('Host message: ' + evt.data);
+ externalHost.postMessage(evt.data,
+ "privileged_target");
+ appendStatus('After postMessage(' + evt.data + ', "privileged_target)"');
+ externalHost.postMessage(evt.data);
+ appendStatus('After postMessage(' + evt.data + '")');
+ }
+ </script>
+ </head>
+
+ <body onload="OnLoad();">
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <p>ChromeFrame PrivilegeApis Test
+ <br>Tests that private messaging is not available to regular web pages</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/privileged_apis_host.html b/chrome_frame/test/data/privileged_apis_host.html
new file mode 100644
index 0000000..7261704
--- /dev/null
+++ b/chrome_frame/test/data/privileged_apis_host.html
@@ -0,0 +1,85 @@
+<html>
+ <head><title>Privileged Apis test</title>
+ <script type='text/javascript' src='chrome_frame_tester_helpers.js'>
+ </script>
+ <script type='text/javascript'>
+ var testName = 'PrivilegedApis';
+ function OnNavigationFailed(msg) {
+ onFailure(testName, 1, 'ChromeFrame Navigation failed: ' + msg);
+ }
+
+ function OnPrivateMessage() {
+ onFailure(testName, 1, 'OnPrivateMessage should not execute');
+ }
+
+ function OnChromeFrameMessage(evt) {
+ try {
+ var d = new String(evt.data);
+ appendStatus('Message: ' + d);
+ if (d == 'succeed') {
+ onSuccess(testName, 1);
+ } else {
+ onFailure(testName, 1, 'unexpected data');
+ }
+ } catch (e) {
+ onFailure(testName, 1, 'exception in OnChromeFrameMessage');
+ }
+ }
+
+ function OnChromeFrameLoaded(url) {
+ var cf = GetChromeFrame();
+
+ try {
+ // Any message received by this listener is a failure.
+ // This succeeds in FF, but throws an exception in IE.
+ cf.addEventListener('onprivatemessage', OnPrivateMessage, false);
+ } catch(e) {
+ cf.onprivatemessage =
+ appendStatus('addEventListener onprivatemessage threw exception')
+ }
+
+ // If this invocation succeeds, then 'fail' is reflected by the frame
+ // and we fail in the OnChromeFrameMessage handler above.
+ try {
+ cf.postPrivateMessage('fail', String(document.location), '*');
+ onFailure(testName, 1, 'postPrivateMessage should throw');
+ } catch(e) {
+ }
+ appendStatus('After postPrivateMessage')
+ // The frame reflects this twice, first to a bogus target
+ // and again to the default target '*'. We succeed if we
+ // get the reflected message to OnChromeFrameMessage and not to
+ // OnPrivateMessage.
+ cf.postMessage('succeed');
+ }
+
+ function GetChromeFrame() {
+ return window.document.ChromeFrame;
+ }
+ </script>
+ </head>
+ <body>
+ <div id='statusPanel' style='border: 1px solid red; width: 100%'>
+ Test running....
+ </div>
+
+ <!-- TODO(siggi): Test setting onprivatemessage in these params -->
+ <object id='ChromeFrame' width='500' height='500'
+ codebase='http://www.google.com'
+ classid='CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A'>
+ <param name='src' value='privileged_apis_frame.html'>
+ <param name='onload' value='OnChromeFrameLoaded(arguments[0]);'>
+ <param name='onloaderror' value='OnNavigationFailed();'>
+ <param name='onmessage' value='OnChromeFrameMessage(arguments[0]);'>
+ <embed id='ChromeFramePlugin' width='500' height='500' name='ChromeFrame'
+ src='privileged_apis_frame.html'
+ type='application/chromeframe'
+ onload='OnChromeFrameLoaded(arguments[0]);'
+ onloaderror='OnNavigationFailed();'
+ onmessage='return OnChromeFrameMessage(arguments[0]);'
+ privileged_mode='1'
+ </embed>
+ </object>
+ <p>Tests that privileged apis are unavailable from regular pages</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/simple_object_focus.html b/chrome_frame/test/data/simple_object_focus.html
new file mode 100644
index 0000000..138ffa5
--- /dev/null
+++ b/chrome_frame/test/data/simple_object_focus.html
@@ -0,0 +1,95 @@
+<!-- saved from url=(0014)about:internet -->
+<html>
+<!-- This page is meant to load inside the host browser like IE/FF -->
+<head>
+<script type="text/javascript" src="chrome_frame_tester_helpers.js"></script>
+<script type="text/javascript">
+var g_failure_timeout = null;
+var g_test_id = 1;
+var g_test_name = "ObjectFocus";
+
+function onLoad() {
+ status("onload");
+
+ try {
+ var cf = getCf();
+ cf.onmessage = OnChromeFrameMessage;
+ window.setTimeout(NavigateToURL, 100);
+ } catch(e) {
+ status("error: onload");
+ onFailure(g_test_name, g_test_id, "error in onload");
+ }
+
+ sendOnLoadEvent();
+}
+
+function NavigateToURL() {
+ try {
+ status("Navigate to URL");
+ var cf = getCf();
+ cf.src = "simple_object_focus_cf.html";
+ g_failure_timeout = window.setTimeout(OnObjectFocusFailed, 10000);
+ } catch(e) {
+ status("error: NavigateToURL");
+ onFailure(g_test_name, g_test_id, "NavigateToURL error");
+ }
+}
+
+function OnObjectFocusFailed() {
+ status("OnNavigationFailed");
+ onFailure(g_test_name, g_test_id, "focus test failed");
+}
+
+function OnChromeFrameLoaded() {
+ status("OnChromeFrameLoaded");
+ try {
+ // Set focus to chrome frame. This should set focus to the first element
+ // inside the frame, which a script inside the page will detect and notify
+ // us back by sending us a message.
+ getCf().focus();
+ } catch(e) {
+ status("error: can't set focus");
+ onFailure(g_test_name, g_test_id, "focus() error");
+ }
+}
+
+function OnChromeFrameMessage(evt) {
+ if (evt.data != "btnOnFocus") {
+ status("unexpected message: " + evt.data + " from " + evt.origin);
+ } else {
+ window.clearTimeout(g_failure_timeout);
+ g_failure_timeout = null;
+ status("success");
+ }
+ onSuccess(g_test_name, g_test_id);
+}
+
+function getCf() {
+ // Fetching chrome frame with getElementById doesn't work in Firefox.
+ // Most likely due to object vs embed.
+ return document.ChromeFrame;
+}
+
+// Useful while writing and debugging the unit test.
+function status(s) {
+ var panel = document.getElementById("status_panel");
+ panel.innerHTML = s;
+}
+
+</script>
+</head>
+<body onload="onLoad();">
+<div id="status_panel" style="border: 1px solid red; width: 100%">
+Test running....
+</div>
+<object id="ChromeFrame" width="300" height="60" tabindex="0"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="onload" value="return OnChromeFrameLoaded();">
+ <embed width="300" height="60" name="ChromeFrame"
+ onload="return OnChromeFrameLoaded();"
+ type="application/chromeframe">
+ </embed>
+</object>
+</body>
+</html>
diff --git a/chrome_frame/test/data/simple_object_focus_cf.html b/chrome_frame/test/data/simple_object_focus_cf.html
new file mode 100644
index 0000000..9b06711
--- /dev/null
+++ b/chrome_frame/test/data/simple_object_focus_cf.html
@@ -0,0 +1,10 @@
+<html>
+<!-- This page is meant to load inside Chrome Frame -->
+ <body>
+ <button onfocus="externalHost.postMessage('btnOnFocus');">
+ hello world</button>
+ <div id="statusPanel" style="border: 1px solid green; width: 100%">
+ Inside Chrome Frame....
+ </div>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/src_property_frame1.html b/chrome_frame/test/data/src_property_frame1.html
new file mode 100644
index 0000000..1eaa3cf
--- /dev/null
+++ b/chrome_frame/test/data/src_property_frame1.html
@@ -0,0 +1,13 @@
+<html>
+ <head><title>src property test - page 1</title>
+ <script type="text/javascript">
+ function redirect(){
+ window.location = "src_property_frame2.html";
+ }
+ </script>
+ </head>
+ <body onLoad="setTimeout(redirect, 100);">
+ <h2>Prepare to be redirected!</h2>
+ <p>Redirecting to a new page within frame...</p>
+ </body>
+</html> \ No newline at end of file
diff --git a/chrome_frame/test/data/src_property_frame2.html b/chrome_frame/test/data/src_property_frame2.html
new file mode 100644
index 0000000..c5c0364
--- /dev/null
+++ b/chrome_frame/test/data/src_property_frame2.html
@@ -0,0 +1,8 @@
+<html>
+ <head><title>src property test - page 2</title>
+ </head>
+ <body>
+ <h2>Redirected!</h2>
+ <p>All finished.</p>
+ </body>
+</html> \ No newline at end of file
diff --git a/chrome_frame/test/data/src_property_host.html b/chrome_frame/test/data/src_property_host.html
new file mode 100644
index 0000000..7b7b358
--- /dev/null
+++ b/chrome_frame/test/data/src_property_host.html
@@ -0,0 +1,65 @@
+<html>
+ <head><title>src property test</title>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ function OnNavigationFailed() {
+ onFailure("ChromeFrame_SrcTest", 1, "ChromeFrame Navigation failed");
+ }
+
+ var load_count = 2;
+
+ function OnChromeFrameLoaded(url) {
+ url = url.data;
+
+ var chromeFrame = GetChromeFrame();
+ var frame_url = chromeFrame.src;
+
+ appendStatus("Loaded URL: " + url + " Frame url: " + frame_url);
+ load_count--;
+
+ if (load_count) {
+ // For the first load, the URLs should match.
+ if (frame_url != url) {
+ onFailure("SrcProperty", 1, "Url: " + url);
+ }
+ } else {
+ // Previous versions changed the frame URL when internal navigation
+ // was performed. This does not match how iframes behave, and so we
+ // report success only in the case that they continue to match, even
+ // though the "internal" URL is different (and not visible) to the
+ // external host.
+ if (frame_url == url) {
+ onFailure("SrcProperty", 1, "Url: " + url);
+ } else {
+ onSuccess("SrcProperty", 1);
+ }
+ }
+ }
+
+ function GetChromeFrame() {
+ return window.document.ChromeFrame;
+ }
+ </script>
+ </head>
+ <body>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+
+ <object id="ChromeFrame" width="500" height="500"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <param name="src" value="src_property_frame1.html">
+ <param name="onload" value="return OnChromeFrameLoaded(arguments[0]);">
+ <param name="onloaderror" value="return OnNavigationFailed(arguments[0]);">
+ <embed id="ChromeFramePlugin" width="500" height="500" name="ChromeFrame"
+ src="src_property_frame1.html"
+ type="application/chromeframe"
+ onload="return OnChromeFrameLoaded(arguments[0]);"
+ onloaderror="return OnNavigationFailed(arguments[0]);">
+ </embed>
+ </object>
+ <p>Tests ChromeFrame Navigation</p>
+ </body>
+</html>
diff --git a/chrome_frame/test/data/version.html b/chrome_frame/test/data/version.html
new file mode 100644
index 0000000..b585a6d
--- /dev/null
+++ b/chrome_frame/test/data/version.html
@@ -0,0 +1,29 @@
+<html>
+<!-- This page is meant to load inside the host browser like IE/FF -->
+ <head>
+ <script type="text/javascript" src="chrome_frame_tester_helpers.js">
+ </script>
+ <script type="text/javascript">
+ function onLoad() {
+ appendStatus('Chrome frame version: ' + document.ChromeFrame.version);
+ onFinished('version', 1, document.ChromeFrame.version);
+ }
+ </script>
+ </head>
+
+ <body onload="onLoad();">
+ <object id="ChromeFrame"
+ codebase="http://www.google.com"
+ classid="CLSID:E0A900DF-9611-4446-86BD-4B1D47E7DB2A">
+ <embed id="ChromeFramePlugin" name="ChromeFrame"
+ type="application/chromeframe"
+ </embed>
+ </object>
+ <br>
+ <br>
+ <p>Test for Chrome frame version</p>
+ <div id="statusPanel" style="border: 1px solid red; width: 100%">
+ Test running....
+ </div>
+ </body>
+</html>
diff --git a/chrome_frame/test/function_stub_unittest.cc b/chrome_frame/test/function_stub_unittest.cc
new file mode 100644
index 0000000..6ef6f36
--- /dev/null
+++ b/chrome_frame/test/function_stub_unittest.cc
@@ -0,0 +1,66 @@
+// Copyright (c) 2009 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 "chrome_frame/function_stub.h"
+
+#include "chrome_frame/test/chrome_frame_unittests.h"
+
+#define NO_INLINE __declspec(noinline)
+
+namespace {
+
+typedef int (__stdcall* FooPrototype)();
+
+NO_INLINE int __stdcall Foo() {
+ return 1;
+}
+
+NO_INLINE int __stdcall PatchedFoo(FooPrototype original) {
+ return original() + 1;
+}
+
+} // end namespace
+
+TEST(PatchTests, FunctionStub) {
+ EXPECT_EQ(Foo(), 1);
+ // Create a function stub that calls PatchedFoo and supplies it with
+ // a pointer to Foo.
+ FunctionStub* stub = FunctionStub::Create(reinterpret_cast<uintptr_t>(&Foo),
+ &PatchedFoo);
+ EXPECT_TRUE(stub != NULL);
+ // Call the stub as it were Foo(). The call should get forwarded to Foo().
+ FooPrototype patch = reinterpret_cast<FooPrototype>(stub->code());
+ EXPECT_EQ(patch(), 2);
+ // Now neutralize the stub so that it calls Foo() directly without touching
+ // PatchedFoo().
+ // stub->BypassStub(&Foo);
+ stub->BypassStub(reinterpret_cast<void*>(stub->argument()));
+ EXPECT_EQ(patch(), 1);
+ // We're done with the stub.
+ FunctionStub::Destroy(stub);
+}
+
+// Basic tests to check the validity of a stub.
+TEST(PatchTests, FunctionStubCompare) {
+ EXPECT_EQ(Foo(), 1);
+
+ // Detect the absence of a stub
+ FunctionStub* stub = reinterpret_cast<FunctionStub*>(&Foo);
+ EXPECT_FALSE(stub->is_valid());
+
+ stub = FunctionStub::Create(reinterpret_cast<uintptr_t>(&Foo), &PatchedFoo);
+ EXPECT_TRUE(stub != NULL);
+ EXPECT_TRUE(stub->is_valid());
+
+ FooPrototype patch = reinterpret_cast<FooPrototype>(stub->code());
+ EXPECT_EQ(patch(), 2);
+
+ // See if we can get the correct absolute pointer to the hook function
+ // back from the stub.
+ EXPECT_EQ(stub->absolute_target(), reinterpret_cast<uintptr_t>(&PatchedFoo));
+
+ // Also verify that the argument being passed to the hook function is indeed
+ // the pointer to the original function (again, absolute not relative).
+ EXPECT_EQ(stub->argument(), reinterpret_cast<uintptr_t>(&Foo));
+}
diff --git a/chrome_frame/test/helper_gmock.h b/chrome_frame/test/helper_gmock.h
new file mode 100644
index 0000000..7f6d0a7
--- /dev/null
+++ b/chrome_frame/test/helper_gmock.h
@@ -0,0 +1,597 @@
+// Copyright (c) 2006-2009 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.
+#ifndef CHROME_FRAME_TEST_HELPER_GMOCK_H_
+#define CHROME_FRAME_TEST_HELPER_GMOCK_H_
+// This intention of this file is to make possible gmock WithArgs<> in
+// Chromium code base.
+// MutantImpl is like CallbackImpl, but also has prebound arguments (like Task)
+// There is also functor wrapper around it that should be used with
+// testing::Invoke, for example:
+// testing::WithArgs<0, 2>(
+// testing::Invoke(CBF(&mock_object, &MockObject::Something, &tmp_obj, 12)));
+// This will invoke MockObject::Something(tmp_obj, 12, arg_0, arg_2)
+
+// DispatchToMethod supporting two sets of arguments -
+// prebound (P) and calltime (C)
+// 1 - 1
+template <class ObjT, class Method, class P1, class C1>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple1<P1>& p,
+ const Tuple1<C1>& c) {
+ (obj->*method)(p.a, c.a);
+}
+// 2 - 1
+template <class ObjT, class Method, class P1, class P2, class C1>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple2<P1, P2>& p,
+ const Tuple1<C1>& c) {
+ (obj->*method)(p.a, p.b, c.a);
+}
+// 3 - 1
+template <class ObjT, class Method, class P1, class P2, class P3, class C1>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple3<P1, P2, P3>& p,
+ const Tuple1<C1>& c) {
+ (obj->*method)(p.a, p.b, p.c, c.a);
+}
+// 4 - 1
+template <class ObjT, class Method, class P1, class P2, class P3,
+ class P4, class C1>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple4<P1, P2, P3, P4>& p,
+ const Tuple1<C1>& c) {
+ (obj->*method)(p.a, p.b, p.c, p.d, c.a);
+}
+
+// 1 - 2
+template <class ObjT, class Method, class P1, class C1, class C2>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple1<P1>& p,
+ const Tuple2<C1, C2>& c) {
+ (obj->*method)(p.a, c.a, c.b);
+}
+
+// 2 - 2
+template <class ObjT, class Method, class P1, class P2, class C1, class C2>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple2<P1, P2>& p,
+ const Tuple2<C1, C2>& c) {
+ (obj->*method)(p.a, p.b, c.a, c.b);
+}
+
+// 3 - 2
+template <class ObjT, class Method, class P1, class P2, class P3, class C1,
+ class C2>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple3<P1, P2, P3>& p,
+ const Tuple2<C1, C2>& c) {
+ (obj->*method)(p.a, p.b, p.c, c.a, c.b);
+}
+
+// 4 - 2
+template <class ObjT, class Method, class P1, class P2, class P3, class P4,
+ class C1, class C2>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple4<P1, P2, P3, P4>& p,
+ const Tuple2<C1, C2>& c) {
+ (obj->*method)(p.a, p.b, p.c, p.d, c.a, c.b);
+}
+
+// 1 - 3
+template <class ObjT, class Method, class P1, class C1, class C2, class C3>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple1<P1>& p,
+ const Tuple3<C1, C2, C3>& c) {
+ (obj->*method)(p.a, c.a, c.b, c.c);
+}
+
+// 2 - 3
+template <class ObjT, class Method, class P1, class P2, class C1, class C2,
+ class C3>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple2<P1, P2>& p,
+ const Tuple3<C1, C2, C3>& c) {
+ (obj->*method)(p.a, p.b, c.a, c.b, c.c);
+}
+
+// 3 - 3
+template <class ObjT, class Method, class P1, class P2, class P3, class C1,
+ class C2, class C3>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple3<P1, P2, P3>& p,
+ const Tuple3<C1, C2, C3>& c) {
+ (obj->*method)(p.a, p.b, p.c, c.a, c.b, c.c);
+}
+
+// 4 - 3
+template <class ObjT, class Method, class P1, class P2, class P3, class P4,
+ class C1, class C2, class C3>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple4<P1, P2, P3, P4>& p,
+ const Tuple3<C1, C2, C3>& c) {
+ (obj->*method)(p.a, p.b, p.c, p.d, c.a, c.b, c.c);
+}
+
+// 1 - 4
+template <class ObjT, class Method, class P1, class C1, class C2, class C3,
+ class C4>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple1<P1>& p,
+ const Tuple4<C1, C2, C3, C4>& c) {
+ (obj->*method)(p.a, c.a, c.b, c.c, c.d);
+}
+
+// 2 - 4
+template <class ObjT, class Method, class P1, class P2, class C1, class C2,
+ class C3, class C4>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple2<P1, P2>& p,
+ const Tuple4<C1, C2, C3, C4>& c) {
+ (obj->*method)(p.a, p.b, c.a, c.b, c.c, c.d);
+}
+
+// 3 - 4
+template <class ObjT, class Method, class P1, class P2, class P3,
+ class C1, class C2, class C3, class C4>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple3<P1, P2, P3>& p,
+ const Tuple4<C1, C2, C3, C4>& c) {
+ (obj->*method)(p.a, p.b, p.c, c.a, c.b, c.c, c.d);
+}
+
+// 4 - 4
+template <class ObjT, class Method, class P1, class P2, class P3, class P4,
+ class C1, class C2, class C3, class C4>
+inline void DispatchToMethod(ObjT* obj, Method method,
+ const Tuple4<P1, P2, P3, P4>& p,
+ const Tuple4<C1, C2, C3, C4>& c) {
+ (obj->*method)(p.a, p.b, p.c, p.d, c.a, c.b, c.c, c.d);
+}
+
+
+////////////////////////////////////////////////////////////////////////////////
+
+// Like CallbackImpl but has prebound arguments (like Task)
+template <class T, typename Method, typename PreBound, typename Params>
+class MutantImpl : public CallbackStorage<T, Method>,
+ public CallbackRunner<Params> {
+ public:
+ MutantImpl(T* obj, Method meth, const PreBound& pb)
+ : CallbackStorage<T, Method>(obj, meth),
+ pb_(pb) {
+ }
+
+ virtual void RunWithParams(const Params& params) {
+ // use "this->" to force C++ to look inside our templatized base class; see
+ // Effective C++, 3rd Ed, item 43, p210 for details.
+ DispatchToMethod(this->obj_, this->meth_, pb_, params);
+ }
+
+ PreBound pb_;
+};
+
+////////////////////////////////////////////////////////////////////////////////
+// Mutant creation simplification
+// 1 - 1
+template <class T, typename P1, typename A1>
+inline typename Callback1<A1>::Type* NewMutant(T* obj,
+ void (T::*method)(P1, A1),
+ P1 p1) {
+ return new MutantImpl<T, void (T::*)(P1, A1), P1, A1>(obj, method,
+ MakeTuple(p1));
+}
+
+// 1 - 2
+template <class T, typename P1, typename A1, typename A2>
+inline typename Callback2<A1, A2>::Type* NewMutant(T* obj,
+ void (T::*method)(P1, A1, A2),
+ P1 p1) {
+ return new MutantImpl<T, void (T::*)(P1, A1, A2), Tuple1<P1>, Tuple2<A1, A2> >
+ (obj, method, MakeTuple(p1));
+}
+
+// 1 - 3
+template <class T, typename P1, typename A1, typename A2, typename A3>
+inline typename Callback3<A1, A2, A3>::Type*
+NewMutant(T* obj, void (T::*method)(P1, A1, A2, A3), P1 p1) {
+ return new MutantImpl<T, void (T::*)(P1, A1, A2, A3), Tuple1<P1>,
+ Tuple3<A1, A2, A3> >(obj, method, MakeTuple(p1));
+}
+
+// 1 - 4
+template <class T, typename P1, typename A1, typename A2, typename A3,
+ typename A4>
+inline typename Callback4<A1, A2, A3, A4>::Type*
+NewMutant(T* obj, void (T::*method)(P1, A1, A2, A3, A4), P1 p1) {
+ return new MutantImpl<T, void (T::*)(P1, A1, A2, A3, A4), Tuple1<P1>,
+ Tuple4<A1, A2, A3, A4> >(obj, method, MakeTuple(p1));
+}
+
+
+// 2 - 1
+template <class T, typename P1, typename P2, typename A1>
+inline typename Callback1<A1>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, A1), P1 p1, P2 p2) {
+ return new MutantImpl<T, void (T::*)(P1, P2, A1), Tuple2<P1, P2>,
+ Tuple1<A1> >(obj, method, MakeTuple(p1, p2));
+}
+
+// 2 - 2
+template <class T, typename P1, typename P2, typename A1, typename A2>
+inline typename Callback2<A1, A2>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, A1, A2), P1 p1, P2 p2) {
+ return new MutantImpl<T, void (T::*)(P1, P2, A1, A2), Tuple2<P1, P2>,
+ Tuple2<A1, A2> >(obj, method, MakeTuple(p1, p2));
+}
+
+// 2 - 3
+template <class T, typename P1, typename P2, typename A1, typename A2,
+ typename A3>
+inline typename Callback3<A1, A2, A3>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, A1, A2, A3), P1 p1, P2 p2) {
+ return new MutantImpl<T, void (T::*)(P1, P2, A1, A2, A3), Tuple2<P1, P2>,
+ Tuple3<A1, A2, A3> >(obj, method, MakeTuple(p1, p2));
+}
+
+// 2 - 4
+template <class T, typename P1, typename P2, typename A1, typename A2,
+ typename A3, typename A4>
+inline typename Callback4<A1, A2, A3, A4>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, A1, A2, A3, A4), P1 p1, P2 p2) {
+ return new MutantImpl<T, void (T::*)(P1, P2, A1, A2, A3, A4), Tuple2<P1, P2>,
+ Tuple3<A1, A2, A3, A4> >(obj, method, MakeTuple(p1, p2));
+}
+
+// 3 - 1
+template <class T, typename P1, typename P2, typename P3, typename A1>
+inline typename Callback1<A1>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, A1), P1 p1, P2 p2, P3 p3) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, A1), Tuple3<P1, P2, P3>,
+ Tuple1<A1> >(obj, method, MakeTuple(p1, p2, p3));
+}
+
+// 3 - 2
+template <class T, typename P1, typename P2, typename P3, typename A1,
+ typename A2>
+inline typename Callback2<A1, A2>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, A1, A2), P1 p1, P2 p2, P3 p3) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, A1, A2), Tuple3<P1, P2, P3>,
+ Tuple2<A1, A2> >(obj, method, MakeTuple(p1, p2, p3));
+}
+
+// 3 - 3
+template <class T, typename P1, typename P2, typename P3, typename A1,
+ typename A2, typename A3>
+inline typename Callback3<A1, A2, A3>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, A1, A2, A3), P1 p1, P2 p2,
+ P3 p3) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, A1, A2, A3),
+ Tuple3<P1, P2, P3>, Tuple3<A1, A2, A3> >(obj, method,
+ MakeTuple(p1, p2, p3));
+}
+
+// 3 - 4
+template <class T, typename P1, typename P2, typename P3, typename A1,
+ typename A2, typename A3, typename A4>
+inline typename Callback4<A1, A2, A3, A4>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, A1, A2, A3, A4), P1 p1, P2 p2,
+ P3 p3) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, A1, A2, A3, A4),
+ Tuple3<P1, P2, P3>, Tuple3<A1, A2, A3, A4> >(obj, method,
+ MakeTuple(p1, p2, p3));
+}
+
+// 4 - 1
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1>
+inline typename Callback1<A1>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, P4, A1), P1 p1, P2 p2, P3 p3,
+ P4 p4) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1),
+ Tuple4<P1, P2, P3, P4>, Tuple1<A1> >(obj, method,
+ MakeTuple(p1, p2, p3, p4));
+}
+
+// 4 - 2
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1, typename A2>
+inline typename Callback2<A1, A2>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, P4, A1, A2), P1 p1, P2 p2,
+ P3 p3, P4 p4) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1, A2),
+ Tuple4<P1, P2, P3, P4>, Tuple2<A1, A2> >(obj, method,
+ MakeTuple(p1, p2, p3, p4));
+}
+
+// 4 - 3
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1, typename A2, typename A3>
+inline typename Callback3<A1, A2, A3>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, P4, A1, A2, A3), P1 p1, P2 p2,
+ P3 p3, P4 p4) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1, A2, A3),
+ Tuple4<P1, P2, P3, P4>, Tuple3<A1, A2, A3> >(obj, method,
+ MakeTuple(p1, p2, p3, p4));
+}
+
+// 4 - 4
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1, typename A2, typename A3, typename A4>
+inline typename Callback4<A1, A2, A3, A4>::Type*
+NewMutant(T* obj, void (T::*method)(P1, P2, P3, P4, A1, A2, A3, A4),
+ P1 p1, P2 p2, P3 p3, P4 p4) {
+ return new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1, A2, A3, A4),
+ Tuple4<P1, P2, P3, P4>, Tuple3<A1, A2, A3, A4> >(obj, method,
+ MakeTuple(p1, p2, p3, p4));
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// Simple callback wrapper acting as a functor.
+// Redirects operator() to CallbackRunner<Params>::Run
+// We cannot delete the inner impl_ in object's destructor because
+// this object is copied few times inside from GMock machinery.
+template <typename Params>
+struct CallbackFunctor {
+ explicit CallbackFunctor(CallbackRunner<Params>* cb) : impl_(cb) {}
+
+ template <typename Arg1>
+ inline void operator()(const Arg1& a) {
+ impl_->Run(a);
+ delete impl_;
+ impl_ = NULL;
+ }
+
+ template <typename Arg1, typename Arg2>
+ inline void operator()(const Arg1& a, const Arg2& b) {
+ impl_->Run(a, b);
+ delete impl_;
+ impl_ = NULL;
+ }
+
+ template <typename Arg1, typename Arg2, typename Arg3>
+ inline void operator()(const Arg1& a, const Arg2& b, const Arg3& c) {
+ impl_->Run(a, b, c);
+ delete impl_;
+ impl_ = NULL;
+ }
+
+ template <typename Arg1, typename Arg2, typename Arg3, typename Arg4>
+ inline void operator()(const Arg1& a, const Arg2& b, const Arg3& c,
+ const Arg4& d) {
+ impl_->Run(a, b, c, d);
+ delete impl_;
+ impl_ = NULL;
+ }
+
+ private:
+ CallbackFunctor();
+ CallbackRunner<Params>* impl_;
+};
+
+
+///////////////////////////////////////////////////////////////////////////////
+// CallbackFunctors creation
+
+// 0 - 1
+template <class T, typename A1>
+inline CallbackFunctor<Tuple1<A1> >
+CBF(T* obj, void (T::*method)(A1)) {
+ return CallbackFunctor<Tuple1<A1> >(NewCallback(obj, method));
+}
+
+// 0 - 2
+template <class T, typename A1, typename A2>
+inline CallbackFunctor<Tuple2<A1, A2> >
+CBF(T* obj, void (T::*method)(A1, A2)) {
+ return CallbackFunctor<Tuple2<A1, A2> >(NewCallback(obj, method));
+}
+
+// 0 - 3
+template <class T, typename A1, typename A2, typename A3>
+inline CallbackFunctor<Tuple3<A1, A2, A3> >
+CBF(T* obj, void (T::*method)(A1, A2, A3)) {
+ return CallbackFunctor<Tuple3<A1, A2, A3> >(NewCallback(obj, method));
+}
+
+// 0 - 4
+template <class T, typename A1, typename A2, typename A3, typename A4>
+inline CallbackFunctor<Tuple4<A1, A2, A3, A4> >
+CBF(T* obj, void (T::*method)(A1, A2, A3, A4)) {
+ return CallbackFunctor<Tuple4<A1, A2, A3, A4> >(NewCallback(obj, method));
+}
+
+// 1 - 1
+template <class T, typename P1, typename A1>
+inline CallbackFunctor<Tuple1<A1> >
+CBF(T* obj, void (T::*method)(P1, A1), P1 p1) {
+ Callback1<A1>::Type* t = new MutantImpl<T, void (T::*)(P1, A1), Tuple1<P1>,
+ Tuple1<A1> >(obj, method, MakeTuple(p1));
+ return CallbackFunctor<Tuple1<A1> >(t);
+}
+
+// 1 - 2
+template <class T, typename P1, typename A1, typename A2>
+inline CallbackFunctor<Tuple2<A1, A2> >
+CBF(T* obj, void (T::*method)(P1, A1, A2), P1 p1) {
+ Callback2<A1, A2>::Type* t = new MutantImpl<T, void (T::*)(P1, A1, A2),
+ Tuple1<P1>, Tuple2<A1, A2> >(obj, method, MakeTuple(p1));
+ return CallbackFunctor<Tuple2<A1, A2> >(t);
+}
+
+// 1 - 3
+template <class T, typename P1, typename A1, typename A2, typename A3>
+inline CallbackFunctor<Tuple3<A1, A2, A3> >
+CBF(T* obj, void (T::*method)(P1, A1, A2, A3), P1 p1) {
+ Callback3<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, A1, A2, A3), Tuple1<P1>,
+ Tuple3<A1, A2, A3> >(obj, method, MakeTuple(p1));
+ return CallbackFunctor<Tuple3<A1, A2, A3> >(t);
+}
+
+// 1 - 4
+template <class T, typename P1, typename A1, typename A2, typename A3,
+ typename A4>
+inline CallbackFunctor<Tuple4<A1, A2, A3, A4> >
+CBF(T* obj, void (T::*method)(P1, A1, A2, A3, A4), P1 p1) {
+ Callback4<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, A1, A2, A3, A4), Tuple1<P1>,
+ Tuple4<A1, A2, A3, A4> >(obj, method, MakeTuple(p1));
+ return CallbackFunctor<Tuple4<A1, A2, A3, A4> >(t);
+}
+
+// 2 - 1
+template <class T, typename P1, typename P2, typename A1>
+inline CallbackFunctor<Tuple1<A1> >
+CBF(T* obj, void (T::*method)(P1, P2, A1), P1 p1, P2 p2) {
+ Callback1<A1>::Type* t = new MutantImpl<T, void (T::*)(P1, P2, A1),
+ Tuple2<P1, P2>, Tuple1<A1> >(obj, method, MakeTuple(p1, p2));
+ return CallbackFunctor<Tuple1<A1> >(t);
+}
+
+// 2 - 2
+template <class T, typename P1, typename P2, typename A1, typename A2>
+inline CallbackFunctor<Tuple2<A1, A2> >
+CBF(T* obj, void (T::*method)(P1, P2, A1, A2), P1 p1, P2 p2) {
+ Callback2<A1, A2>::Type* t = new MutantImpl<T, void (T::*)(P1, P2, A1, A2),
+ Tuple2<P1, P2>, Tuple2<A1, A2> >(obj, method, MakeTuple(p1, p2));
+ return CallbackFunctor<Tuple2<A1, A2> >(t);
+}
+
+// 2 - 3
+template <class T, typename P1, typename P2, typename A1, typename A2,
+ typename A3> inline CallbackFunctor<Tuple3<A1, A2, A3> >
+CBF(T* obj, void (T::*method)(P1, P2, A1, A2, A3), P1 p1, P2 p2) {
+ Callback3<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, A1, A2, A3), Tuple2<P1, P2>,
+ Tuple3<A1, A2, A3> >(obj, method, MakeTuple(p1, p2));
+ return CallbackFunctor<Tuple3<A1, A2, A3> >(t);
+}
+
+// 2 - 4
+template <class T, typename P1, typename P2, typename A1, typename A2,
+ typename A3, typename A4>
+inline CallbackFunctor<Tuple4<A1, A2, A3, A4> >
+CBF(T* obj, void (T::*method)(P1, P2, A1, A2, A3, A4), P1 p1, P2 p2) {
+ Callback4<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, A1, A2, A3, A4), Tuple2<P1, P2>,
+ Tuple4<A1, A2, A3, A4> >(obj, method, MakeTuple(p1, p2));
+ return CallbackFunctor<Tuple4<A1, A2, A3, A4> >(t);
+}
+
+
+// 3 - 1
+template <class T, typename P1, typename P2, typename P3, typename A1>
+inline CallbackFunctor<Tuple1<A1> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, A1), P1 p1, P2 p2, P3 p3) {
+ Callback1<A1>::Type* t = new MutantImpl<T, void (T::*)(P1, P2, P3, A1),
+ Tuple3<P1, P2, P3>, Tuple1<A1> >(obj, method, MakeTuple(p1, p2, p3));
+ return CallbackFunctor<Tuple1<A1> >(t);
+}
+
+
+// 3 - 2
+template <class T, typename P1, typename P2, typename P3, typename A1,
+ typename A2> inline CallbackFunctor<Tuple2<A1, A2> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, A1, A2), P1 p1, P2 p2, P3 p3) {
+ Callback2<A1, A2>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, P3, A1, A2), Tuple3<P1, P2, P3>,
+ Tuple2<A1, A2> >(obj, method, MakeTuple(p1, p2, p3));
+ return CallbackFunctor<Tuple2<A1, A2> >(t);
+}
+
+// 3 - 3
+template <class T, typename P1, typename P2, typename P3, typename A1,
+ typename A2, typename A3>
+inline CallbackFunctor<Tuple3<A1, A2, A3> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, A1, A2, A3), P1 p1, P2 p2, P3 p3) {
+ Callback3<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, P3, A1, A2, A3), Tuple3<P1, P2, P3>,
+ Tuple3<A1, A2, A3> >(obj, method, MakeTuple(p1, p2, p3));
+ return CallbackFunctor<Tuple3<A1, A2, A3> >(t);
+}
+
+// 3 - 4
+template <class T, typename P1, typename P2, typename P3, typename A1,
+ typename A2, typename A3, typename A4>
+inline CallbackFunctor<Tuple4<A1, A2, A3, A4> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, A1, A2, A3, A4),
+ P1 p1, P2 p2, P3 p3) {
+ Callback4<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, P3, A1, A2, A3, A4),
+ Tuple3<P1, P2, P3>, Tuple4<A1, A2, A3, A4> >(obj, method,
+ MakeTuple(p1, p2, p3));
+ return CallbackFunctor<Tuple4<A1, A2, A3, A4> >(t);
+}
+
+
+
+// 4 - 1
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1>
+inline CallbackFunctor<Tuple1<A1> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, P4, A1), P1 p1, P2 p2, P3 p3, P4 p4) {
+ Callback1<A1>::Type* t = new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1),
+ Tuple4<P1, P2, P3, P4>, Tuple1<A1> >
+ (obj, method, MakeTuple(p1, p2, p3, p4));
+ return CallbackFunctor<Tuple1<A1> >(t);
+}
+
+
+// 4 - 2
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1, typename A2>
+inline CallbackFunctor<Tuple2<A1, A2> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, P4, A1, A2),
+ P1 p1, P2 p2, P3 p3, P4 p4) {
+ Callback2<A1, A2>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1, A2),
+ Tuple4<P1, P2, P3, P4>,
+ Tuple2<A1, A2> >(obj, method, MakeTuple(p1, p2, p3, p4));
+ return CallbackFunctor<Tuple2<A1, A2> >(t);
+}
+
+// 4 - 3
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1, typename A2, typename A3>
+inline CallbackFunctor<Tuple3<A1, A2, A3> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, P4, A1, A2, A3),
+ P1 p1, P2 p2, P3 p3, P4 p4) {
+ Callback3<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1, A2, A3),
+ Tuple4<P1, P2, P3, P4>, Tuple3<A1, A2, A3> >
+ (obj, method, MakeTuple(p1, p2, p3, p4));
+ return CallbackFunctor<Tuple3<A1, A2, A3> >(t);
+}
+
+// 4 - 4
+template <class T, typename P1, typename P2, typename P3, typename P4,
+ typename A1, typename A2, typename A3, typename A4>
+inline CallbackFunctor<Tuple4<A1, A2, A3, A4> >
+CBF(T* obj, void (T::*method)(P1, P2, P3, P4, A1, A2, A3, A4),
+ P1 p1, P2 p2, P3 p3, P4 p4) {
+ Callback4<A1, A2, A3>::Type* t =
+ new MutantImpl<T, void (T::*)(P1, P2, P3, P4, A1, A2, A3, A4),
+ Tuple4<P1, P2, P3, P4>, Tuple4<A1, A2, A3, A4> >(obj, method,
+ MakeTuple(p1, p2, p3, p4));
+ return CallbackFunctor<Tuple4<A1, A2, A3, A4> >(t);
+}
+
+
+// Simple task wrapper acting as a functor.
+// Redirects operator() to Task::Run. We cannot delete the inner impl_ object
+// in object's destructor because this object is copied few times inside
+// from GMock machinery.
+struct TaskHolder {
+ explicit TaskHolder(Task* impl) : impl_(impl) {}
+ void operator()() {
+ impl_->Run();
+ delete impl_;
+ impl_ = NULL;
+ }
+ private:
+ TaskHolder();
+ Task* impl_;
+};
+
+#endif // CHROME_FRAME_TEST_HELPER_GMOCK_H_
diff --git a/chrome_frame/test/html_util_test_data/basic_test.html b/chrome_frame/test/html_util_test_data/basic_test.html
new file mode 100644
index 0000000..f0cd17a
--- /dev/null
+++ b/chrome_frame/test/html_util_test_data/basic_test.html
@@ -0,0 +1,11 @@
+<HTML>
+
+ <HEAD>
+ <!-- Note the capitalization in CONtent to test the
+ case-insensitiveness -->
+ <META http-equiv="X-UA-Compatible" CONtent="chrome=1" />
+ </HEAD>
+ <BODY>
+
+ Wooo!
+ </BODY></HTML>
diff --git a/chrome_frame/test/html_util_test_data/degenerate_cases_test.html b/chrome_frame/test/html_util_test_data/degenerate_cases_test.html
new file mode 100644
index 0000000..d527496a
--- /dev/null
+++ b/chrome_frame/test/html_util_test_data/degenerate_cases_test.html
@@ -0,0 +1,7 @@
+<><foo ">
+</head>
+<"foo">
+<!-- Note that the meta tag shouldn't be picked up since we are still
+ inside a quote block. -->
+<META http-equiv="X-UA-Compatible" CONtent="chrome=1" />
+<fee>
diff --git a/chrome_frame/test/html_util_test_data/multiple_tags.html b/chrome_frame/test/html_util_test_data/multiple_tags.html
new file mode 100644
index 0000000..9bd5cea
--- /dev/null
+++ b/chrome_frame/test/html_util_test_data/multiple_tags.html
@@ -0,0 +1,17 @@
+<HTML><HEAD>
+
+ <META http-equiv="X-UA-Compatible" CONtent="chrome=1" />
+ <META http-equiv="X-UA-Compatible" content="chrome=1" />
+ <METAman http-equiv="X-UA-Compatible" CONtent="notchrome=1" />
+ <transMETA http-equiv="X-UA-Compatible" CONtent="notchrome=1" />
+ <IMETAGIRL http-equiv="X-UA-Compatible" CONtent="notchrome=1" />
+ <metA http-equiv="X-UA-Compatible" content="chrome=1" />
+ <!-- shouldn't pick up commented meta tags! -->
+ <!-- <metA http-equiv="X-UA-Compatible" content="chrome=1" /> -->
+
+ <!-- The following cases should also not be matched -->
+ <meta http-equiv="X-UA-Compatibleeee" content="chrome=1" />
+ <meta http-equiv="X-UA-Compatible!" content="chrome=1" />
+ <meta http-equiv="!X-UA-Compatible" content="chrome=1" />
+ <meta http-equiv="\"X-UA-Compatible\"" content="chrome=1" />
+<fee> \ No newline at end of file
diff --git a/chrome_frame/test/html_util_test_data/quotes_test.html b/chrome_frame/test/html_util_test_data/quotes_test.html
new file mode 100644
index 0000000..03ce96d
--- /dev/null
+++ b/chrome_frame/test/html_util_test_data/quotes_test.html
@@ -0,0 +1,10 @@
+<HTML>
+
+ <HEAD>
+ <DANGER red="herring>'" testing = "do'><><>quotes\"\\'work?">
+ <META http-equiv=X-UA-Compatible CONtent="chrome=1" />
+ </HEAD>
+ <BODY>
+
+ Wooo!
+ </BODY></HTML>
diff --git a/chrome_frame/test/html_util_unittests.cc b/chrome_frame/test/html_util_unittests.cc
new file mode 100644
index 0000000..131b185
--- /dev/null
+++ b/chrome_frame/test/html_util_unittests.cc
@@ -0,0 +1,215 @@
+// Copyright (c) 2006-2009 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 <windows.h>
+#include <atlsecurity.h>
+#include <shellapi.h>
+
+#include "base/basictypes.h"
+#include "base/file_util.h"
+#include "base/message_loop.h"
+#include "base/path_service.h"
+#include "base/process_util.h"
+#include "base/ref_counted.h"
+#include "base/scoped_handle.h"
+#include "base/task.h"
+#include "base/win_util.h"
+#include "net/base/net_util.h"
+
+#include "chrome_frame/test/chrome_frame_unittests.h"
+#include "chrome_frame/chrome_frame_automation.h"
+#include "chrome_frame/chrome_frame_delegate.h"
+#include "chrome_frame/html_utils.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+class HtmlUtilUnittest : public testing::Test {
+ protected:
+ // Constructor
+ HtmlUtilUnittest() {}
+
+ // Returns the test path given a test case.
+ virtual bool GetTestPath(const std::wstring& test_case, std::wstring* path) {
+ if (!path) {
+ NOTREACHED();
+ return false;
+ }
+
+ std::wstring test_path;
+ if (!PathService::Get(base::DIR_SOURCE_ROOT, &test_path)) {
+ NOTREACHED();
+ return false;
+ }
+
+ file_util::AppendToPath(&test_path, L"chrome_frame");
+ file_util::AppendToPath(&test_path, L"test");
+ file_util::AppendToPath(&test_path, L"html_util_test_data");
+ file_util::AppendToPath(&test_path, test_case);
+
+ *path = test_path;
+ return true;
+ }
+
+ virtual bool GetTestData(const std::wstring& test_case, std::wstring* data) {
+ if (!data) {
+ NOTREACHED();
+ return false;
+ }
+
+ std::wstring path;
+ if (!GetTestPath(test_case, &path)) {
+ NOTREACHED();
+ return false;
+ }
+
+ std::string raw_data;
+ file_util::ReadFileToString(path, &raw_data);
+
+ // Convert to wide using the "best effort" assurance described in
+ // string_util.h
+ data->assign(UTF8ToWide(raw_data));
+ return true;
+ }
+};
+
+TEST_F(HtmlUtilUnittest, BasicTest) {
+ std::wstring test_data;
+ GetTestData(L"basic_test.html", &test_data);
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Grab the meta tag from the document and ensure that we get exactly one.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ ASSERT_EQ(1, tag_list.size());
+
+ // Pull out the http-equiv attribute and check its value:
+ HTMLScanner::StringRange attribute_value;
+ EXPECT_TRUE(tag_list[0].GetTagAttribute(L"http-equiv", &attribute_value));
+ EXPECT_TRUE(attribute_value.Equals(L"X-UA-Compatible"));
+
+ // Pull out the content attribute and check its value:
+ EXPECT_TRUE(tag_list[0].GetTagAttribute(L"content", &attribute_value));
+ EXPECT_TRUE(attribute_value.Equals(L"chrome=1"));
+}
+
+TEST_F(HtmlUtilUnittest, QuotesTest) {
+ std::wstring test_data;
+ GetTestData(L"quotes_test.html", &test_data);
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Grab the meta tag from the document and ensure that we get exactly one.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ ASSERT_EQ(1, tag_list.size());
+
+ // Pull out the http-equiv attribute and check its value:
+ HTMLScanner::StringRange attribute_value;
+ EXPECT_TRUE(tag_list[0].GetTagAttribute(L"http-equiv", &attribute_value));
+ EXPECT_TRUE(attribute_value.Equals(L"X-UA-Compatible"));
+
+ // Pull out the content attribute and check its value:
+ EXPECT_TRUE(tag_list[0].GetTagAttribute(L"content", &attribute_value));
+ EXPECT_TRUE(attribute_value.Equals(L"chrome=1"));
+}
+
+TEST_F(HtmlUtilUnittest, DegenerateCasesTest) {
+ std::wstring test_data;
+ GetTestData(L"degenerate_cases_test.html", &test_data);
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Scan for meta tags in the document. We expect not to pick up the one
+ // that appears to be there since it is technically inside a quote block.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ EXPECT_TRUE(tag_list.empty());
+}
+
+TEST_F(HtmlUtilUnittest, MultipleTagsTest) {
+ std::wstring test_data;
+ GetTestData(L"multiple_tags.html", &test_data);
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Grab the meta tag from the document and ensure that we get exactly three.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ EXPECT_EQ(7, tag_list.size());
+
+ // Pull out the content attribute for each tag and check its value:
+ HTMLScanner::StringRange attribute_value;
+ HTMLScanner::StringRangeList::const_iterator tag_list_iter(
+ tag_list.begin());
+ int valid_tag_count = 0;
+ for (; tag_list_iter != tag_list.end(); tag_list_iter++) {
+ HTMLScanner::StringRange attribute_value;
+ if (tag_list_iter->GetTagAttribute(L"http-equiv", &attribute_value) &&
+ attribute_value.Equals(L"X-UA-Compatible")) {
+ EXPECT_TRUE(tag_list_iter->GetTagAttribute(L"content", &attribute_value));
+ EXPECT_TRUE(attribute_value.Equals(L"chrome=1"));
+ valid_tag_count++;
+ }
+ }
+ EXPECT_EQ(3, valid_tag_count);
+}
+
+TEST_F(HtmlUtilUnittest, ShortDegenerateTest1) {
+ std::wstring test_data(
+ L"<foo><META http-equiv=X-UA-Compatible content='chrome=1'");
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Scan for meta tags in the document. We expect not to pick up the one
+ // that is there since it is not properly closed.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ EXPECT_TRUE(tag_list.empty());
+}
+
+TEST_F(HtmlUtilUnittest, ShortDegenerateTest2) {
+ std::wstring test_data(
+ L"<foo <META http-equiv=X-UA-Compatible content='chrome=1'/>");
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Scan for meta tags in the document. We expect not to pick up the one
+ // that appears to be there since it is inside a non-closed tag.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ EXPECT_TRUE(tag_list.empty());
+}
+
+TEST_F(HtmlUtilUnittest, QuoteInsideHTMLCommentTest) {
+ std::wstring test_data(
+ L"<!-- comment' --><META http-equiv=X-UA-Compatible content='chrome=1'/>");
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Grab the meta tag from the document and ensure that we get exactly one.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ ASSERT_EQ(1, tag_list.size());
+
+ // Pull out the http-equiv attribute and check its value:
+ HTMLScanner::StringRange attribute_value;
+ EXPECT_TRUE(tag_list[0].GetTagAttribute(L"http-equiv", &attribute_value));
+ EXPECT_TRUE(attribute_value.Equals(L"X-UA-Compatible"));
+
+ // Pull out the content attribute and check its value:
+ EXPECT_TRUE(tag_list[0].GetTagAttribute(L"content", &attribute_value));
+ EXPECT_TRUE(attribute_value.Equals(L"chrome=1"));
+}
+
+TEST_F(HtmlUtilUnittest, CloseTagInsideHTMLCommentTest) {
+ std::wstring test_data(
+ L"<!-- comment> <META http-equiv=X-UA-Compatible content='chrome=1'/>-->");
+
+ HTMLScanner scanner(test_data.c_str());
+
+ // Grab the meta tag from the document and ensure that we get exactly one.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(L"meta", &tag_list, L"body");
+ ASSERT_TRUE(tag_list.empty());
+}
diff --git a/chrome_frame/test/http_server.cc b/chrome_frame/test/http_server.cc
new file mode 100644
index 0000000..f2cc333
--- /dev/null
+++ b/chrome_frame/test/http_server.cc
@@ -0,0 +1,56 @@
+// Copyright (c) 2006-2009 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 "chrome_frame/test/http_server.h"
+
+const wchar_t kDocRoot[] = L"chrome_frame\\test\\data";
+
+void ChromeFrameHTTPServer::SetUp() {
+ std::wstring document_root(kDocRoot);
+ server_ = HTTPTestServer::CreateServer(document_root, NULL, 30, 1000);
+ ASSERT_TRUE(server_ != NULL);
+
+ // copy CFInstance.js into the test directory
+ FilePath cf_source_path;
+ PathService::Get(base::DIR_SOURCE_ROOT, &cf_source_path);
+ cf_source_path = cf_source_path.Append(FILE_PATH_LITERAL("chrome_frame"));
+
+ file_util::CopyFile(cf_source_path.Append(FILE_PATH_LITERAL("CFInstance.js")),
+ cf_source_path.Append(
+ FILE_PATH_LITERAL("test")).Append(
+ FILE_PATH_LITERAL("data")).Append(
+ FILE_PATH_LITERAL("CFInstance.js"))); // NOLINT
+}
+
+void ChromeFrameHTTPServer::TearDown() {
+ if (server_) {
+ server_ = NULL;
+ }
+
+ // clobber CFInstance.js
+ FilePath cfi_path;
+ PathService::Get(base::DIR_SOURCE_ROOT, &cfi_path);
+ cfi_path = cfi_path
+ .Append(FILE_PATH_LITERAL("chrome_frame"))
+ .Append(FILE_PATH_LITERAL("test"))
+ .Append(FILE_PATH_LITERAL("data"))
+ .Append(FILE_PATH_LITERAL("CFInstance.js"));
+
+ file_util::Delete(cfi_path, false);
+}
+
+bool ChromeFrameHTTPServer::WaitToFinish(int milliseconds) {
+ if (!server_)
+ return true;
+
+ return server_->WaitToFinish(milliseconds);
+}
+
+GURL ChromeFrameHTTPServer::Resolve(const wchar_t* relative_url) {
+ return server_->TestServerPageW(relative_url);
+}
+
+std::wstring ChromeFrameHTTPServer::GetDataDir() {
+ return server_->GetDataDirectory().ToWStringHack();
+}
+
diff --git a/chrome_frame/test/http_server.h b/chrome_frame/test/http_server.h
new file mode 100644
index 0000000..acac5b5
--- /dev/null
+++ b/chrome_frame/test/http_server.h
@@ -0,0 +1,32 @@
+// Copyright (c) 2006-2009 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.
+#ifndef CHROME_FRAME_TEST_HTTP_SERVER_H_
+#define CHROME_FRAME_TEST_HTTP_SERVER_H_
+
+#include <windows.h>
+#include <string>
+
+#include "googleurl/src/gurl.h"
+#include "base/ref_counted.h"
+#include "net/url_request/url_request_unittest.h"
+
+// chrome frame specilization of http server from net.
+class ChromeFrameHTTPServer {
+ public:
+ void SetUp();
+ void TearDown();
+ bool WaitToFinish(int milliseconds);
+ GURL Resolve(const wchar_t* relative_url);
+ std::wstring GetDataDir();
+
+ HTTPTestServer* server() {
+ return server_;
+ }
+
+ protected:
+ scoped_refptr<HTTPTestServer> server_;
+};
+
+#endif // CHROME_FRAME_TEST_HTTP_SERVER_H_
+
diff --git a/chrome_frame/test/icu_stubs_unittests.cc b/chrome_frame/test/icu_stubs_unittests.cc
new file mode 100644
index 0000000..4da4a40
--- /dev/null
+++ b/chrome_frame/test/icu_stubs_unittests.cc
@@ -0,0 +1,73 @@
+// Copyright (c) 2006-2009 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 "chrome_frame/test/chrome_frame_unittests.h"
+
+// Need to include these first since they're included
+#include "base/logging.h"
+#include "base/string_util.h"
+#include "googleurl/src/url_canon.h"
+
+// Include the implementation of our stubs into a special namespace so that
+// we can separate them from Chrome's implementation.
+namespace icu_stubs {
+ // This struct is only to avoid build problems for the two googleurl stubs
+ // that currently are noops.
+ struct CanonOutputW { };
+
+ #include "chrome_frame/icu_stubs.cc"
+} // namespace icu_stubs
+
+// anonymous namespace for test data.
+namespace {
+
+ // Test strings borrowed from base/string_util_unittest.cc
+ static const wchar_t* const kConvertRoundtripCases[] = {
+ L"",
+ L"Google Vid¯ôfY »"
+ L"\x7f51\x9875\x0020\x56fe\x7247\x0020\x8d44\x8baf\x66f4\x591a\x0020\x00bb",
+ // " ±³ºÌü¹¿Â ÃÄÌÂ"
+ L"\x03a0\x03b1\x03b3\x03ba\x03cc\x03c3\x03bc\x03b9"
+ L"\x03bf\x03c2\x0020\x0399\x03c3\x03c4\x03cc\x03c2",
+ // ">8A: AB@0=8F =0 @CAA:><"
+ L"\x041f\x043e\x0438\x0441\x043a\x0020\x0441\x0442"
+ L"\x0440\x0430\x043d\x0438\x0446\x0020\x043d\x0430"
+ L"\x0020\x0440\x0443\x0441\x0441\x043a\x043e\x043c",
+ // "È´ÌÁD¾¤Â"
+ L"\xc804\xccb4\xc11c\xbe44\xc2a4",
+
+ // Test characters that take more than 16 bits. This will depend on whether
+ // wchar_t is 16 or 32 bits.
+ #if defined(WCHAR_T_IS_UTF16)
+ L"\xd800\xdf00",
+ // ????? (Mathematical Alphanumeric Symbols (U+011d40 - U+011d44 : A,B,C,D,E)
+ L"\xd807\xdd40\xd807\xdd41\xd807\xdd42\xd807\xdd43\xd807\xdd44",
+ #elif defined(WCHAR_T_IS_UTF32)
+ L"\x10300",
+ // ????? (Mathematical Alphanumeric Symbols (U+011d40 - U+011d44 : A,B,C,D,E)
+ L"\x11d40\x11d41\x11d42\x11d43\x11d44",
+ #endif
+ };
+
+} // namespace
+
+TEST(IcuStubsTests, UTF8AndWideStubTest) {
+ // Test code borrowed from ConvertUTF8AndWide in base/string_util_unittest.cc.
+
+ // The difference is that we want to make sure that our stubs work the same
+ // way as chrome's implementation of WideToUTF8 and UTF8ToWide.
+ for (size_t i = 0; i < arraysize(kConvertRoundtripCases); ++i) {
+ std::ostringstream utf8_base, utf8_stub;
+ utf8_base << WideToUTF8(kConvertRoundtripCases[i]);
+ utf8_stub << icu_stubs::WideToUTF8(kConvertRoundtripCases[i]);
+
+ EXPECT_EQ(utf8_base.str(), utf8_stub.str());
+
+ std::wostringstream wide_base, wide_stub;
+ wide_base << UTF8ToWide(utf8_base.str());
+ wide_stub << icu_stubs::UTF8ToWide(utf8_base.str());
+
+ EXPECT_EQ(wide_base.str(), wide_stub.str());
+ }
+}
diff --git a/chrome_frame/test/net/dialog_watchdog.cc b/chrome_frame/test/net/dialog_watchdog.cc
new file mode 100644
index 0000000..27a01a0
--- /dev/null
+++ b/chrome_frame/test/net/dialog_watchdog.cc
@@ -0,0 +1,146 @@
+// Copyright (c) 2009 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 <oleacc.h>
+
+#include "chrome_frame/test/net/dialog_watchdog.h"
+
+#include "base/logging.h"
+#include "base/scoped_comptr_win.h"
+#include "base/string_util.h"
+
+#include "chrome_frame/test/chrome_frame_test_utils.h"
+#include "chrome_frame/function_stub.h"
+
+namespace {
+// Uses the IAccessible interface for the window to set the focus.
+// This can be useful when you don't have control over the thread that
+// owns the window.
+// NOTE: this depends on oleacc.lib which the net tests already depend on
+// but other unit tests don't depend on oleacc so we can't just add the method
+// directly into chrome_frame_test_utils.cc (without adding a
+// #pragma comment(lib, "oleacc.lib")).
+bool SetFocusToAccessibleWindow(HWND hwnd) {
+ bool ret = false;
+ ScopedComPtr<IAccessible> acc;
+ AccessibleObjectFromWindow(hwnd, OBJID_WINDOW, IID_IAccessible,
+ reinterpret_cast<void**>(acc.Receive()));
+ if (acc) {
+ VARIANT self = { VT_I4 };
+ self.lVal = CHILDID_SELF;
+ ret = SUCCEEDED(acc->accSelect(SELFLAG_TAKEFOCUS, self));
+ }
+ return ret;
+}
+
+} // namespace
+
+SupplyProxyCredentials::SupplyProxyCredentials(const char* username,
+ const char* password)
+ : username_(username), password_(password) {
+}
+
+bool SupplyProxyCredentials::OnDialogDetected(HWND hwnd,
+ const std::string& caption) {
+ // IE's dialog caption (en-US).
+ if (caption.compare("Windows Security") != 0)
+ return false;
+
+ DialogProps props = {0};
+ ::EnumChildWindows(hwnd, EnumChildren, reinterpret_cast<LPARAM>(&props));
+ DCHECK(::IsWindow(props.username_));
+ DCHECK(::IsWindow(props.password_));
+
+ // We can't use SetWindowText to set the username/password, so simulate
+ // keyboard input instead.
+ chrome_frame_test::ForceSetForegroundWindow(hwnd);
+ CHECK(SetFocusToAccessibleWindow(props.username_));
+ chrome_frame_test::SendString(username_.c_str());
+ Sleep(100);
+
+ chrome_frame_test::SendVirtualKey(VK_TAB);
+ Sleep(100);
+ chrome_frame_test::SendString(password_.c_str());
+
+ Sleep(100);
+ chrome_frame_test::SendVirtualKey(VK_RETURN);
+
+ return true;
+}
+
+// static
+BOOL SupplyProxyCredentials::EnumChildren(HWND hwnd, LPARAM param) {
+ if (!::IsWindowVisible(hwnd))
+ return TRUE; // Ignore but continue to enumerate.
+
+ DialogProps* props = reinterpret_cast<DialogProps*>(param);
+
+ char class_name[MAX_PATH] = {0};
+ ::GetClassNameA(hwnd, class_name, arraysize(class_name));
+ if (lstrcmpiA(class_name, "Edit") == 0) {
+ if (props->username_ == NULL || props->username_ == hwnd) {
+ props->username_ = hwnd;
+ } else if (props->password_ == NULL) {
+ props->password_ = hwnd;
+ }
+ } else {
+ EnumChildWindows(hwnd, EnumChildren, param);
+ }
+
+ return TRUE;
+}
+
+DialogWatchdog::DialogWatchdog() : hook_(NULL), hook_stub_(NULL) {
+ Initialize();
+}
+
+DialogWatchdog::~DialogWatchdog() {
+ Uninitialize();
+}
+
+bool DialogWatchdog::Initialize() {
+ DCHECK(hook_ == NULL);
+ DCHECK(hook_stub_ == NULL);
+ hook_stub_ = FunctionStub::Create(reinterpret_cast<uintptr_t>(this),
+ WinEventHook);
+ hook_ = SetWinEventHook(EVENT_OBJECT_SHOW, EVENT_OBJECT_SHOW, NULL,
+ reinterpret_cast<WINEVENTPROC>(hook_stub_->code()), 0,
+ 0, WINEVENT_OUTOFCONTEXT | WINEVENT_SKIPOWNPROCESS);
+
+ return hook_ != NULL;
+}
+
+void DialogWatchdog::Uninitialize() {
+ if (hook_) {
+ ::UnhookWinEvent(hook_);
+ hook_ = NULL;
+ FunctionStub::Destroy(hook_stub_);
+ hook_stub_ = NULL;
+ }
+}
+
+// static
+void DialogWatchdog::WinEventHook(DialogWatchdog* me, HWINEVENTHOOK hook,
+ DWORD event, HWND hwnd, LONG object_id,
+ LONG child_id, DWORD event_thread_id,
+ DWORD event_time) {
+ // Check for a dialog class ("#32770") and notify observers if we find one.
+ char class_name[MAX_PATH] = {0};
+ ::GetClassNameA(hwnd, class_name, arraysize(class_name));
+ if (lstrcmpA(class_name, "#32770") == 0) {
+ int len = ::GetWindowTextLength(hwnd);
+ std::string text;
+ ::GetWindowTextA(hwnd, WriteInto(&text, len + 1), len + 1);
+ me->OnDialogFound(hwnd, text);
+ }
+}
+
+void DialogWatchdog::OnDialogFound(HWND hwnd, const std::string& caption) {
+ std::vector<DialogWatchdogObserver*>::iterator it = observers_.begin();
+ while (it != observers_.end()) {
+ if ((*it)->OnDialogDetected(hwnd, caption))
+ break;
+ it++;
+ }
+}
diff --git a/chrome_frame/test/net/dialog_watchdog.h b/chrome_frame/test/net/dialog_watchdog.h
new file mode 100644
index 0000000..dfb8989
--- /dev/null
+++ b/chrome_frame/test/net/dialog_watchdog.h
@@ -0,0 +1,64 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_TEST_NET_DIALOG_WATCHDOG_H_
+#define CHROME_FRAME_TEST_NET_DIALOG_WATCHDOG_H_
+
+#include <windows.h>
+
+#include <string>
+#include <vector>
+
+struct FunctionStub;
+
+class DialogWatchdogObserver { // NOLINT
+ public:
+ // returns true if this observer handled the dialog.
+ virtual bool OnDialogDetected(HWND hwnd, const std::string& caption) = 0;
+};
+
+class SupplyProxyCredentials : public DialogWatchdogObserver {
+ public:
+ SupplyProxyCredentials(const char* username, const char* password);
+
+ protected:
+ struct DialogProps {
+ HWND username_;
+ HWND password_;
+ };
+
+ virtual bool OnDialogDetected(HWND hwnd, const std::string& caption);
+ static BOOL CALLBACK EnumChildren(HWND hwnd, LPARAM param);
+
+ protected:
+ std::string username_;
+ std::string password_;
+};
+
+class DialogWatchdog {
+ public:
+ DialogWatchdog();
+ ~DialogWatchdog();
+
+ inline void AddObserver(DialogWatchdogObserver* observer) {
+ observers_.push_back(observer);
+ }
+
+ bool Initialize();
+ void Uninitialize();
+
+ protected:
+ static void CALLBACK WinEventHook(DialogWatchdog* me, HWINEVENTHOOK hook,
+ DWORD event, HWND hwnd, LONG object_id, LONG child_id,
+ DWORD event_thread_id, DWORD event_time);
+
+ void OnDialogFound(HWND hwnd, const std::string& caption);
+
+ protected:
+ HWINEVENTHOOK hook_;
+ std::vector<DialogWatchdogObserver*> observers_;
+ FunctionStub* hook_stub_;
+};
+
+#endif // CHROME_FRAME_TEST_NET_DIALOG_WATCHDOG_H_
diff --git a/chrome_frame/test/net/fake_external_tab.cc b/chrome_frame/test/net/fake_external_tab.cc
new file mode 100644
index 0000000..eebd2d9b
--- /dev/null
+++ b/chrome_frame/test/net/fake_external_tab.cc
@@ -0,0 +1,391 @@
+// Copyright (c) 2009 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 "chrome_frame/test/net/fake_external_tab.h"
+
+#include <exdisp.h>
+
+#include "app/app_paths.h"
+#include "app/resource_bundle.h"
+#include "app/win_util.h"
+
+#include "base/command_line.h"
+#include "base/file_util.h"
+#include "base/icu_util.h"
+#include "base/path_service.h"
+#include "base/scoped_bstr_win.h"
+#include "base/scoped_comptr_win.h"
+#include "base/scoped_variant_win.h"
+
+#include "chrome/browser/browser_prefs.h"
+#include "chrome/browser/chrome_thread.h"
+#include "chrome/browser/process_singleton.h"
+#include "chrome/browser/profile_manager.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/common/chrome_paths.h"
+#include "chrome/common/chrome_switches.h"
+#include "chrome/common/pref_names.h"
+#include "chrome/browser/renderer_host/render_process_host.h"
+
+#include "chrome_frame/utils.h"
+#include "chrome_frame/test/chrome_frame_test_utils.h"
+#include "chrome_frame/test/net/dialog_watchdog.h"
+#include "chrome_frame/test/net/test_automation_resource_message_filter.h"
+
+namespace {
+
+// A special command line switch to allow developers to manually launch the
+// browser and debug CF inside the browser.
+const wchar_t kManualBrowserLaunch[] = L"manual-browser";
+
+// Pops up a message box after the test environment has been set up
+// and before tearing it down. Useful for when debugging tests and not
+// the test environment that's been set up.
+const wchar_t kPromptAfterSetup[] = L"prompt-after-setup";
+
+const int kTestServerPort = 4666;
+// The test HTML we use to initialize Chrome Frame.
+// Note that there's a little trick in there to avoid an extra URL request
+// that the browser will otherwise make for the site's favicon.
+// If we don't do this the browser will create a new URL request after
+// the CF page has been initialized and that URL request will confuse the
+// global URL instance counter in the unit tests and subsequently trip
+// some DCHECKs.
+const char kChromeFrameHtml[] = "<html><head>"
+ "<meta http-equiv=\"X-UA-Compatible\" content=\"chrome=1\" />"
+ "<link rel=\"shortcut icon\" href=\"file://c:\\favicon.ico\"/>"
+ "</head><body>Chrome Frame should now be loaded</body></html>";
+
+bool ShouldLaunchBrowser() {
+ return !CommandLine::ForCurrentProcess()->HasSwitch(kManualBrowserLaunch);
+}
+
+bool PromptAfterSetup() {
+ return CommandLine::ForCurrentProcess()->HasSwitch(kPromptAfterSetup);
+}
+
+} // end namespace
+
+FakeExternalTab::FakeExternalTab() {
+ PathService::Get(chrome::DIR_USER_DATA, &overridden_user_dir_);
+ user_data_dir_ = FilePath::FromWStringHack(GetProfilePath());
+ PathService::Override(chrome::DIR_USER_DATA, user_data_dir_);
+ process_singleton_.reset(new ProcessSingleton(user_data_dir_));
+}
+
+FakeExternalTab::~FakeExternalTab() {
+ if (!overridden_user_dir_.empty()) {
+ PathService::Override(chrome::DIR_USER_DATA, overridden_user_dir_);
+ }
+}
+
+std::wstring FakeExternalTab::GetProfileName() {
+ return L"iexplore";
+}
+
+std::wstring FakeExternalTab::GetProfilePath() {
+ std::wstring path;
+ GetUserProfileBaseDirectory(&path);
+ file_util::AppendToPath(&path, GetProfileName());
+ return path;
+}
+
+void FakeExternalTab::Initialize() {
+ DCHECK(g_browser_process == NULL);
+
+ // The gears plugin causes the PluginRequestInterceptor to kick in and it
+ // will cause problems when it tries to intercept URL requests.
+ PathService::Override(chrome::FILE_GEARS_PLUGIN, FilePath());
+
+ icu_util::Initialize();
+
+ chrome::RegisterPathProvider();
+ app::RegisterPathProvider();
+
+ ResourceBundle::InitSharedInstance(L"en-US");
+ ResourceBundle::GetSharedInstance().LoadThemeResources();
+
+ const CommandLine* cmd = CommandLine::ForCurrentProcess();
+ browser_process_.reset(new BrowserProcessImpl(*cmd));
+ RenderProcessHost::set_run_renderer_in_process(true);
+ // BrowserProcessImpl's constructor should set g_browser_process.
+ DCHECK(g_browser_process);
+
+ Profile* profile = g_browser_process->profile_manager()->
+ GetDefaultProfile(FilePath(user_data()));
+ PrefService* prefs = profile->GetPrefs();
+ PrefService* local_state = browser_process_->local_state();
+ local_state->RegisterStringPref(prefs::kApplicationLocale, L"");
+ local_state->RegisterBooleanPref(prefs::kMetricsReportingEnabled, false);
+
+ browser::RegisterAllPrefs(prefs, local_state);
+
+ // Override some settings to avoid hitting some preferences that have not
+ // been registered.
+ prefs->SetBoolean(prefs::kPasswordManagerEnabled, false);
+ prefs->SetBoolean(prefs::kAlternateErrorPagesEnabled, false);
+ prefs->SetBoolean(prefs::kSafeBrowsingEnabled, false);
+
+ profile->InitExtensions();
+}
+
+void FakeExternalTab::Shutdown() {
+ browser_process_.reset();
+ g_browser_process = NULL;
+ process_singleton_.reset();
+
+ ResourceBundle::CleanupSharedInstance();
+}
+
+CFUrlRequestUnittestRunner::CFUrlRequestUnittestRunner(int argc, char** argv)
+ : NetTestSuite(argc, argv),
+ chrome_frame_html_("/chrome_frame", kChromeFrameHtml) {
+ fake_chrome_.Initialize();
+ pss_subclass_.reset(new ProcessSingletonSubclass(this));
+ EXPECT_TRUE(pss_subclass_->Subclass(fake_chrome_.user_data()));
+ StartChromeFrameInHostBrowser();
+}
+
+CFUrlRequestUnittestRunner::~CFUrlRequestUnittestRunner() {
+ fake_chrome_.Shutdown();
+}
+
+DWORD WINAPI NavigateIE(void* param) {
+ return 0;
+ win_util::ScopedCOMInitializer com;
+ BSTR url = reinterpret_cast<BSTR>(param);
+
+ bool found = false;
+ int retries = 0;
+ const int kMaxRetries = 20;
+ while (!found && retries < kMaxRetries) {
+ ScopedComPtr<IShellWindows> windows;
+ HRESULT hr = ::CoCreateInstance(__uuidof(ShellWindows), NULL, CLSCTX_ALL,
+ IID_IShellWindows, reinterpret_cast<void**>(windows.Receive()));
+ DCHECK(SUCCEEDED(hr)) << "CoCreateInstance";
+
+ if (SUCCEEDED(hr)) {
+ hr = HRESULT_FROM_WIN32(ERROR_NOT_FOUND);
+ long count = 0; // NOLINT
+ windows->get_Count(&count);
+ VARIANT i = { VT_I4 };
+ for (i.lVal = 0; i.lVal < count; ++i.lVal) {
+ ScopedComPtr<IDispatch> folder;
+ windows->Item(i, folder.Receive());
+ if (folder != NULL) {
+ ScopedComPtr<IWebBrowser2> browser;
+ if (SUCCEEDED(browser.QueryFrom(folder))) {
+ found = true;
+ browser->Stop();
+ Sleep(1000);
+ VARIANT empty = ScopedVariant::kEmptyVariant;
+ hr = browser->Navigate(url, &empty, &empty, &empty, &empty);
+ DCHECK(SUCCEEDED(hr)) << "Failed to navigate";
+ break;
+ }
+ }
+ }
+ }
+ if (!found) {
+ DLOG(INFO) << "Waiting for browser to initialize...";
+ ::Sleep(100);
+ retries++;
+ }
+ }
+
+ DCHECK(retries < kMaxRetries);
+ DCHECK(found);
+
+ ::SysFreeString(url);
+
+ return 0;
+}
+
+void CFUrlRequestUnittestRunner::StartChromeFrameInHostBrowser() {
+ if (!ShouldLaunchBrowser())
+ return;
+
+ win_util::ScopedCOMInitializer com;
+ chrome_frame_test::CloseAllIEWindows();
+
+ test_http_server_.reset(new test_server::SimpleWebServer(kTestServerPort));
+ test_http_server_->AddResponse(&chrome_frame_html_);
+ std::wstring url(StringPrintf(L"http://localhost:%i/chrome_frame",
+ kTestServerPort).c_str());
+
+ // Launch IE. This launches IE correctly on Vista too.
+ ScopedHandle ie_process(chrome_frame_test::LaunchIE(url));
+ EXPECT_TRUE(ie_process.IsValid());
+
+ // NOTE: If you're running IE8 and CF is not being loaded, you need to
+ // disable IE8's prebinding until CF properly handles that situation.
+ //
+ // HKCU\Software\Microsoft\Internet Explorer\Main
+ // Value name: EnablePreBinding (REG_DWORD)
+ // Value: 0
+}
+
+void CFUrlRequestUnittestRunner::ShutDownHostBrowser() {
+ if (ShouldLaunchBrowser()) {
+ win_util::ScopedCOMInitializer com;
+ chrome_frame_test::CloseAllIEWindows();
+ }
+}
+
+// Override virtual void Initialize to not call icu initialize
+void CFUrlRequestUnittestRunner::Initialize() {
+ DCHECK(::GetCurrentThreadId() == test_thread_id_);
+
+ // Start by replicating some of the steps that would otherwise be
+ // done by TestSuite::Initialize. We can't call the base class
+ // directly because it will attempt to initialize some things such as
+ // ICU that have already been initialized for this process.
+ InitializeLogging();
+ base::Time::EnableHiResClockForTests();
+
+#if !defined(PURIFY) && defined(OS_WIN)
+ logging::SetLogAssertHandler(UnitTestAssertHandler);
+#endif // !defined(PURIFY)
+
+ // Next, do some initialization for NetTestSuite.
+ NetTestSuite::InitializeTestThread();
+}
+
+void CFUrlRequestUnittestRunner::Shutdown() {
+ DCHECK(::GetCurrentThreadId() == test_thread_id_);
+ NetTestSuite::Shutdown();
+}
+
+void CFUrlRequestUnittestRunner::OnConnectAutomationProviderToChannel(
+ const std::string& channel_id) {
+ Profile* profile = g_browser_process->profile_manager()->
+ GetDefaultProfile(fake_chrome_.user_data());
+
+ AutomationProviderList* list =
+ g_browser_process->InitAutomationProviderList();
+ DCHECK(list);
+ list->AddProvider(TestAutomationProvider::NewAutomationProvider(profile,
+ channel_id, this));
+}
+
+void CFUrlRequestUnittestRunner::OnInitialTabLoaded() {
+ test_http_server_.reset();
+ StartTests();
+}
+
+void CFUrlRequestUnittestRunner::RunMainUIThread() {
+ DCHECK(MessageLoop::current());
+ DCHECK(MessageLoop::current()->type() == MessageLoop::TYPE_UI);
+
+ // Register the main thread by instantiating it, but don't call any methods.
+ ChromeThread main_thread;
+ DCHECK(ChromeThread::CurrentlyOn(ChromeThread::UI));
+
+ MessageLoop::current()->Run();
+}
+
+void CFUrlRequestUnittestRunner::StartTests() {
+ if (PromptAfterSetup())
+ MessageBoxA(NULL, "click ok to run", "", MB_OK);
+
+ DCHECK_EQ(test_thread_.IsValid(), false);
+ test_thread_.Set(::CreateThread(NULL, 0, RunAllUnittests, this, 0,
+ &test_thread_id_));
+ DCHECK(test_thread_.IsValid());
+}
+
+// static
+DWORD CFUrlRequestUnittestRunner::RunAllUnittests(void* param) {
+ PlatformThread::SetName("CFUrlRequestUnittestRunner");
+ CFUrlRequestUnittestRunner* me =
+ reinterpret_cast<CFUrlRequestUnittestRunner*>(param);
+ me->Run();
+ me->fake_chrome_.ui_loop()->PostTask(FROM_HERE,
+ NewRunnableFunction(TakeDownBrowser, me));
+ return 0;
+}
+
+// static
+void CFUrlRequestUnittestRunner::TakeDownBrowser(
+ CFUrlRequestUnittestRunner* me) {
+ if (PromptAfterSetup())
+ MessageBoxA(NULL, "click ok to exit", "", MB_OK);
+
+ me->ShutDownHostBrowser();
+}
+
+void CFUrlRequestUnittestRunner::InitializeLogging() {
+ FilePath exe;
+ PathService::Get(base::FILE_EXE, &exe);
+ FilePath log_filename = exe.ReplaceExtension(FILE_PATH_LITERAL("log"));
+ logging::InitLogging(log_filename.value().c_str(),
+ logging::LOG_TO_BOTH_FILE_AND_SYSTEM_DEBUG_LOG,
+ logging::LOCK_LOG_FILE,
+ logging::DELETE_OLD_LOG_FILE);
+ // We want process and thread IDs because we may have multiple processes.
+ // Note: temporarily enabled timestamps in an effort to catch bug 6361.
+ logging::SetLogItems(true, true, true, true);
+}
+
+void FilterDisabledTests() {
+ if (::testing::FLAGS_gtest_filter.GetLength() &&
+ ::testing::FLAGS_gtest_filter.Compare("*") != 0) {
+ // Don't override user specified filters.
+ return;
+ }
+
+ const char* disabled_tests[] = {
+ // Tests disabled since they're testing the same functionality used
+ // by the TestAutomationProvider.
+ "URLRequestTest.InterceptNetworkError",
+ "URLRequestTest.InterceptRestartRequired",
+ "URLRequestTest.InterceptRespectsCancelMain",
+ "URLRequestTest.InterceptRespectsCancelRedirect",
+ "URLRequestTest.InterceptRespectsCancelFinal",
+ "URLRequestTest.InterceptRespectsCancelInRestart",
+ "URLRequestTest.InterceptRedirect",
+ "URLRequestTest.InterceptServerError",
+ "URLRequestTestFTP.*",
+
+ // Tests that are currently not working:
+
+ // Temporarily disabled because they needs user input (login dialog).
+ "URLRequestTestHTTP.BasicAuth",
+ "URLRequestTestHTTP.BasicAuthWithCookies",
+
+ // HTTPS tests temporarily disabled due to the certificate error dialog.
+ // TODO(tommi): The tests currently fail though, so need to fix.
+ "HTTPSRequestTest.HTTPSMismatchedTest",
+ "HTTPSRequestTest.HTTPSExpiredTest",
+
+ // Tests chrome's network stack's cache (might not apply to CF).
+ "URLRequestTestHTTP.VaryHeader",
+
+ // I suspect we can only get this one to work (if at all) on IE8 and
+ // later by using the new INTERNET_OPTION_SUPPRESS_BEHAVIOR flags
+ // See http://msdn.microsoft.com/en-us/library/aa385328(VS.85).aspx
+ "URLRequestTest.DoNotSaveCookies",
+ };
+
+ std::string filter("-"); // All following filters will be negative.
+ for (int i = 0; i < arraysize(disabled_tests); ++i) {
+ if (i > 0)
+ filter += ":";
+ filter += disabled_tests[i];
+ }
+
+ ::testing::FLAGS_gtest_filter = filter;
+}
+
+int main(int argc, char** argv) {
+ DialogWatchdog watchdog;
+ // See url_request_unittest.cc for these credentials.
+ SupplyProxyCredentials credentials("user", "secret");
+ watchdog.AddObserver(&credentials);
+ testing::InitGoogleTest(&argc, argv);
+ FilterDisabledTests();
+ CFUrlRequestUnittestRunner test_suite(argc, argv);
+ test_suite.RunMainUIThread();
+ return 0;
+}
diff --git a/chrome_frame/test/net/fake_external_tab.h b/chrome_frame/test/net/fake_external_tab.h
new file mode 100644
index 0000000..6ce4f93
--- /dev/null
+++ b/chrome_frame/test/net/fake_external_tab.h
@@ -0,0 +1,106 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_TEST_NET_FAKE_EXTERNAL_TAB_H_
+#define CHROME_FRAME_TEST_NET_FAKE_EXTERNAL_TAB_H_
+
+#include <string>
+
+#include "base/file_path.h"
+#include "base/message_loop.h"
+
+#include "chrome/app/scoped_ole_initializer.h"
+#include "chrome/browser/browser_process_impl.h"
+
+#include "chrome_frame/test/test_server.h"
+#include "chrome_frame/test/net/test_automation_provider.h"
+#include "chrome_frame/test/net/process_singleton_subclass.h"
+
+#include "net/base/net_test_suite.h"
+
+class ProcessSingleton;
+
+class FakeExternalTab {
+ public:
+ FakeExternalTab();
+ ~FakeExternalTab();
+
+ virtual std::wstring GetProfileName();
+
+ virtual std::wstring GetProfilePath();
+ virtual void Initialize();
+ virtual void Shutdown();
+
+ const FilePath& user_data() const {
+ return user_data_dir_;
+ }
+
+ MessageLoopForUI* ui_loop() {
+ return &loop_;
+ }
+
+ protected:
+ MessageLoopForUI loop_;
+ scoped_ptr<BrowserProcess> browser_process_;
+ FilePath overridden_user_dir_;
+ FilePath user_data_dir_;
+ ScopedOleInitializer ole_initializer_; // For RegisterDropTarget etc to work.
+ scoped_ptr<ProcessSingleton> process_singleton_;
+};
+
+// The "master class" that spins the UI and test threads.
+class CFUrlRequestUnittestRunner
+ : public NetTestSuite,
+ public ProcessSingletonSubclassDelegate,
+ public TestAutomationProviderDelegate {
+ public:
+ CFUrlRequestUnittestRunner(int argc, char** argv);
+ ~CFUrlRequestUnittestRunner();
+
+ virtual void StartChromeFrameInHostBrowser();
+
+ virtual void ShutDownHostBrowser();
+
+ // Overrides to not call icu initialize
+ virtual void Initialize();
+ virtual void Shutdown();
+
+ // ProcessSingletonSubclassDelegate.
+ virtual void OnConnectAutomationProviderToChannel(
+ const std::string& channel_id);
+
+ // TestAutomationProviderDelegate.
+ virtual void OnInitialTabLoaded();
+
+ void RunMainUIThread();
+
+ void StartTests();
+
+ protected:
+ // This is the thread that runs all the UrlRequest tests.
+ // Within its context, the Initialize() and Shutdown() routines above
+ // will be called.
+ static DWORD WINAPI RunAllUnittests(void* param);
+
+ static void TakeDownBrowser(CFUrlRequestUnittestRunner* me);
+
+ protected:
+ // Borrowed from TestSuite::Initialize().
+ void InitializeLogging();
+
+ protected:
+ ScopedHandle test_thread_;
+ DWORD test_thread_id_;
+ scoped_ptr<MessageLoop> test_thread_message_loop_;
+
+ scoped_ptr<test_server::SimpleWebServer> test_http_server_;
+ test_server::SimpleResponse chrome_frame_html_;
+
+ // The fake chrome instance. This instance owns the UI message loop
+ // on the main thread.
+ FakeExternalTab fake_chrome_;
+ scoped_ptr<ProcessSingletonSubclass> pss_subclass_;
+};
+
+#endif // CHROME_FRAME_TEST_NET_FAKE_EXTERNAL_TAB_H_
diff --git a/chrome_frame/test/net/process_singleton_subclass.cc b/chrome_frame/test/net/process_singleton_subclass.cc
new file mode 100644
index 0000000..2206a74
--- /dev/null
+++ b/chrome_frame/test/net/process_singleton_subclass.cc
@@ -0,0 +1,111 @@
+// Copyright (c) 2009 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 "chrome_frame/test/net/process_singleton_subclass.h"
+
+#include "base/command_line.h"
+#include "base/path_service.h"
+#include "base/string_util.h"
+#include "chrome/browser/browser_process_impl.h"
+#include "chrome/browser/profile_manager.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/common/chrome_paths.h"
+#include "chrome/common/chrome_switches.h"
+#include "chrome_frame/test/net/test_automation_provider.h"
+#include "chrome_frame/function_stub.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+ProcessSingletonSubclass::ProcessSingletonSubclass(
+ ProcessSingletonSubclassDelegate* delegate)
+ : stub_(NULL), delegate_(delegate), original_wndproc_(NULL) {
+}
+
+ProcessSingletonSubclass::~ProcessSingletonSubclass() {
+ if (stub_) {
+ stub_->BypassStub(reinterpret_cast<void*>(original_wndproc_));
+ }
+}
+
+bool ProcessSingletonSubclass::Subclass(const FilePath& user_data_dir) {
+ DCHECK(stub_ == NULL);
+ DCHECK(original_wndproc_ == NULL);
+ HWND hwnd = FindWindowEx(HWND_MESSAGE, NULL, chrome::kMessageWindowClass,
+ user_data_dir.ToWStringHack().c_str());
+ if (!::IsWindow(hwnd))
+ return false;
+
+ // The window must be in this process for us to be able to subclass it.
+ DWORD pid = 0;
+ ::GetWindowThreadProcessId(hwnd, &pid);
+ EXPECT_EQ(pid, ::GetCurrentProcessId());
+
+ original_wndproc_ = reinterpret_cast<WNDPROC>(::GetWindowLongPtr(hwnd,
+ GWLP_WNDPROC));
+ stub_ = FunctionStub::Create(reinterpret_cast<uintptr_t>(this),
+ &SubclassWndProc);
+ DCHECK(stub_);
+ ::SetWindowLongPtr(hwnd, GWLP_WNDPROC,
+ reinterpret_cast<LONG_PTR>(stub_->code()));
+ return true;
+}
+
+// static
+LRESULT ProcessSingletonSubclass::SubclassWndProc(ProcessSingletonSubclass* me,
+ HWND hwnd, UINT msg,
+ WPARAM wp, LPARAM lp) {
+ switch (msg) {
+ case WM_COPYDATA:
+ return me->OnCopyData(hwnd, reinterpret_cast<HWND>(wp),
+ reinterpret_cast<COPYDATASTRUCT*>(lp));
+ default:
+ break;
+ }
+
+ return me->original_wndproc_(hwnd, msg, wp, lp);
+}
+
+// static
+LRESULT ProcessSingletonSubclass::OnCopyData(HWND hwnd, HWND from_hwnd,
+ const COPYDATASTRUCT* cds) {
+ // We should have enough room for the shortest command (min_message_size)
+ // and also be a multiple of wchar_t bytes. The shortest command
+ // possible is L"START\0\0" (empty current directory and command line).
+ static const int kMinMessageSize = sizeof(L"START\0");
+ EXPECT_TRUE(kMinMessageSize <= cds->cbData);
+
+ if (kMinMessageSize > cds->cbData)
+ return TRUE;
+
+ // We split the string into 4 parts on NULLs.
+ const wchar_t* begin = reinterpret_cast<const wchar_t*>(cds->lpData);
+ const wchar_t* end = begin + (cds->cbData / sizeof(wchar_t));
+ const wchar_t kNull = L'\0';
+ const wchar_t* eos = wmemchr(begin, kNull, end - begin);
+ EXPECT_NE(eos, end);
+ if (lstrcmpW(begin, L"START") == 0) {
+ begin = eos + 1;
+ EXPECT_TRUE(begin <= end);
+ eos = wmemchr(begin, kNull, end - begin);
+ EXPECT_NE(eos, end);
+
+ // Get current directory.
+ const wchar_t* cur_dir = begin;
+ begin = eos + 1;
+ EXPECT_TRUE(begin <= end);
+ eos = wmemchr(begin, kNull, end - begin);
+ // eos might be equal to end at this point.
+
+ // Get command line.
+ std::wstring cmd_line(begin, static_cast<size_t>(end - begin));
+
+ CommandLine parsed_command_line(L"");
+ parsed_command_line.ParseFromString(cmd_line);
+ std::string channel_id(WideToASCII(parsed_command_line.GetSwitchValue(
+ switches::kAutomationClientChannelID)));
+ EXPECT_FALSE(channel_id.empty());
+
+ delegate_->OnConnectAutomationProviderToChannel(channel_id);
+ }
+ return TRUE;
+}
diff --git a/chrome_frame/test/net/process_singleton_subclass.h b/chrome_frame/test/net/process_singleton_subclass.h
new file mode 100644
index 0000000..dd9ce36
--- /dev/null
+++ b/chrome_frame/test/net/process_singleton_subclass.h
@@ -0,0 +1,36 @@
+// Copyright (c) 2009 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.
+#ifndef CHROME_FRAME_TEST_NET_PROCESS_SINGLETON_SUBCLASS_H_
+#define CHROME_FRAME_TEST_NET_PROCESS_SINGLETON_SUBCLASS_H_
+
+#include <windows.h>
+#include "base/file_path.h"
+
+struct FunctionStub;
+
+class ProcessSingletonSubclassDelegate {
+ public:
+ virtual void OnConnectAutomationProviderToChannel(
+ const std::string& channel_id) = 0;
+};
+
+class ProcessSingletonSubclass {
+ public:
+ ProcessSingletonSubclass(ProcessSingletonSubclassDelegate* delegate);
+ ~ProcessSingletonSubclass();
+
+ bool Subclass(const FilePath& user_data_dir);
+
+ protected:
+ static LRESULT CALLBACK SubclassWndProc(ProcessSingletonSubclass* me,
+ HWND hwnd, UINT msg, WPARAM wp, LPARAM lp);
+ LRESULT OnCopyData(HWND hwnd, HWND from_hwnd, const COPYDATASTRUCT* cds);
+ protected:
+ FunctionStub* stub_;
+ ProcessSingletonSubclassDelegate* delegate_;
+ WNDPROC original_wndproc_;
+};
+
+#endif // CHROME_FRAME_TEST_NET_PROCESS_SINGLETON_SUBCLASS_H_
+
diff --git a/chrome_frame/test/net/test_automation_provider.cc b/chrome_frame/test/net/test_automation_provider.cc
new file mode 100644
index 0000000..3a56aa4
--- /dev/null
+++ b/chrome_frame/test/net/test_automation_provider.cc
@@ -0,0 +1,89 @@
+// Copyright (c) 2009 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 "chrome_frame/test/net/test_automation_provider.h"
+
+#include "base/command_line.h"
+#include "chrome/test/automation/automation_messages.h"
+
+#include "chrome_frame/test/net/test_automation_resource_message_filter.h"
+
+namespace {
+
+// A special command line switch to just run the unit tests without CF in
+// the picture. Can be useful when the harness itself needs to be debugged.
+const wchar_t kNoCfTestRun[] = L"no-cf-test-run";
+
+bool CFTestsDisabled() {
+ static bool switch_present = CommandLine::ForCurrentProcess()->
+ HasSwitch(kNoCfTestRun);
+ return switch_present;
+}
+
+} // end namespace
+
+TestAutomationProvider::TestAutomationProvider(
+ Profile* profile,
+ TestAutomationProviderDelegate* delegate)
+ : AutomationProvider(profile), tab_handle_(-1), delegate_(delegate) {
+ filter_ = new TestAutomationResourceMessageFilter(this);
+ URLRequest::RegisterRequestInterceptor(this);
+}
+
+TestAutomationProvider::~TestAutomationProvider() {
+ URLRequest::UnregisterRequestInterceptor(this);
+}
+
+void TestAutomationProvider::OnMessageReceived(const IPC::Message& msg) {
+ if (filter_->OnMessageReceived(msg))
+ return; // Message handled by the filter.
+
+ __super::OnMessageReceived(msg);
+}
+
+// IPC override to grab the tab handle.
+bool TestAutomationProvider::Send(IPC::Message* msg) {
+ if (msg->type() == AutomationMsg_TabLoaded::ID) {
+ DCHECK(tab_handle_ == -1) << "Currently only support one tab";
+ void* iter = NULL;
+ CHECK(msg->ReadInt(&iter, &tab_handle_));
+ DLOG(INFO) << "Got tab handle: " << tab_handle_;
+ DCHECK(tab_handle_ != -1 && tab_handle_ != 0);
+ delegate_->OnInitialTabLoaded();
+ }
+
+ return AutomationProvider::Send(msg);
+}
+
+URLRequestJob* TestAutomationProvider::MaybeIntercept(URLRequest* request) {
+ if (CFTestsDisabled())
+ return NULL;
+
+ if (request->url().SchemeIs("http") || request->url().SchemeIs("https")) {
+ // Only look at requests that don't have any user data.
+ // ResourceDispatcherHost uses the user data for requests that it manages.
+ // We don't want to mess with those.
+
+ // We could also check if the current thread is our TestUrlRequest thread
+ // and only intercept requests that belong to that thread.
+ if (request->GetUserData(NULL) == NULL) {
+ DCHECK(tab_handle_ != -1);
+ URLRequestAutomationJob* job = new URLRequestAutomationJob(request,
+ tab_handle_, filter_);
+ return job;
+ }
+ }
+
+ return NULL;
+}
+
+// static
+TestAutomationProvider* TestAutomationProvider::NewAutomationProvider(
+ Profile* p, const std::string& channel,
+ TestAutomationProviderDelegate* delegate) {
+ TestAutomationProvider* automation = new TestAutomationProvider(p, delegate);
+ automation->ConnectToChannel(channel);
+ automation->SetExpectedTabCount(1);
+ return automation;
+}
diff --git a/chrome_frame/test/net/test_automation_provider.h b/chrome_frame/test/net/test_automation_provider.h
new file mode 100644
index 0000000..75830ba
--- /dev/null
+++ b/chrome_frame/test/net/test_automation_provider.h
@@ -0,0 +1,52 @@
+// Copyright (c) 2009 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.
+#ifndef CHROME_FRAME_TEST_NET_TEST_AUTOMATION_PROVIDER_H_
+#define CHROME_FRAME_TEST_NET_TEST_AUTOMATION_PROVIDER_H_
+
+#include "chrome/browser/automation/automation_provider.h"
+
+class TestAutomationResourceMessageFilter;
+
+// Callback interface for TestAutomationProvider.
+class TestAutomationProviderDelegate {
+ public:
+ virtual void OnInitialTabLoaded() = 0;
+};
+
+// A slightly customized version of AutomationProvider.
+// We override AutomationProvider to be able to filter received messages
+// (see TestAutomationResourceMessageFilter) and know when the initial
+// ExternalTab has been loaded.
+// In order to intercept UrlRequests and make the URLRequestAutomationJob class
+// handle requests from unit tests, we also implement URLRequest::Interceptor.
+class TestAutomationProvider
+ : public AutomationProvider,
+ public URLRequest::Interceptor {
+ public:
+ explicit TestAutomationProvider(Profile* profile,
+ TestAutomationProviderDelegate* delegate);
+
+ virtual ~TestAutomationProvider();
+
+ // AutomationProvider overrides.
+ virtual void OnMessageReceived(const IPC::Message& msg);
+ virtual bool Send(IPC::Message* msg);
+
+ // URLRequest::Interceptor.
+ virtual URLRequestJob* MaybeIntercept(URLRequest* request);
+
+ // Call to instantiate and initialize a new instance of
+ // TestAutomationProvider.
+ static TestAutomationProvider* NewAutomationProvider(
+ Profile* p,
+ const std::string& channel,
+ TestAutomationProviderDelegate* delegate);
+
+ protected:
+ scoped_refptr<TestAutomationResourceMessageFilter> filter_;
+ int tab_handle_;
+ TestAutomationProviderDelegate* delegate_;
+};
+
+#endif CHROME_FRAME_TEST_NET_TEST_AUTOMATION_PROVIDER_H_
diff --git a/chrome_frame/test/net/test_automation_resource_message_filter.cc b/chrome_frame/test/net/test_automation_resource_message_filter.cc
new file mode 100644
index 0000000..32ef532
--- /dev/null
+++ b/chrome_frame/test/net/test_automation_resource_message_filter.cc
@@ -0,0 +1,60 @@
+// Copyright (c) 2009 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 "chrome_frame/test/net/test_automation_resource_message_filter.h"
+
+TestAutomationResourceMessageFilter::TestAutomationResourceMessageFilter(
+ AutomationProvider* automation) : automation_(automation) {
+}
+
+bool TestAutomationResourceMessageFilter::Send(IPC::Message* message) {
+ return automation_->Send(message);
+}
+
+// static
+void TestAutomationResourceMessageFilter::OnRequestMessage(
+ URLRequestAutomationJob* job, IPC::Message* msg) {
+ job->OnMessage(*msg);
+ delete msg;
+}
+
+bool TestAutomationResourceMessageFilter::OnMessageReceived(
+ const IPC::Message& message) {
+ // See if we want to process this message... call the base class
+ // for filter messages, send the message to the correct thread
+ // for URL requests.
+ bool handled = false;
+ int request_id = URLRequestAutomationJob::MayFilterMessage(message);
+ if (request_id) {
+ RequestMap::iterator it = requests_.find(request_id);
+ if (it != requests_.end()) {
+ handled = true;
+ IPC::Message* msg = new IPC::Message(message);
+ RequestJob& job = it->second;
+ job.loop_->PostTask(FROM_HERE, NewRunnableFunction(OnRequestMessage,
+ job.job_, msg));
+ }
+ } else {
+ handled = AutomationResourceMessageFilter::OnMessageReceived(message);
+ }
+ return handled;
+}
+
+// Add request to the list of outstanding requests.
+bool TestAutomationResourceMessageFilter::RegisterRequest(
+ URLRequestAutomationJob* job) {
+ // Store the request in an internal map like the parent class
+ // does, but also store the current loop pointer so we can send
+ // request messages to that loop.
+ DCHECK(requests_.end() == requests_.find(job->id()));
+ RequestJob request_job = { MessageLoop::current(), job };
+ requests_[job->id()] = request_job;
+ return true;
+}
+
+// Remove request from the list of outstanding requests.
+void TestAutomationResourceMessageFilter::UnRegisterRequest(
+ URLRequestAutomationJob* job) {
+ requests_.erase(job->id());
+}
diff --git a/chrome_frame/test/net/test_automation_resource_message_filter.h b/chrome_frame/test/net/test_automation_resource_message_filter.h
new file mode 100644
index 0000000..310307a
--- /dev/null
+++ b/chrome_frame/test/net/test_automation_resource_message_filter.h
@@ -0,0 +1,50 @@
+// Copyright (c) 2009 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.
+#ifndef CHROME_FRAME_TEST_NET_TEST_AUTOMATION_RESOURCE_MESSAGE_FILTER_H_
+#define CHROME_FRAME_TEST_NET_TEST_AUTOMATION_RESOURCE_MESSAGE_FILTER_H_
+
+#include "chrome/browser/automation/automation_provider.h"
+#include "chrome/browser/automation/url_request_automation_job.h"
+
+// Performs the same duties as AutomationResourceMessageFilter but with one
+// difference. Instead of being tied to an IPC channel running on Chrome's
+// IO thread, this instance runs on the unit test's IO thread (all URL request
+// tests have their own IO loop) and is tied to an instance of
+// AutomationProvider (TestAutomationProvider to be exact).
+//
+// Messages from the AutomationProvider that are destined to request objects
+// owned by this class are marshaled over to the request's IO thread instead
+// of using the thread the messages are received on. This way we allow the
+// URL request tests to run sequentially as they were written while still
+// allowing the automation layer to work as it normally does (i.e. process
+// its messages on the receiving thread).
+class TestAutomationResourceMessageFilter
+ : public AutomationResourceMessageFilter {
+ public:
+ TestAutomationResourceMessageFilter(AutomationProvider* automation);
+
+ virtual bool Send(IPC::Message* message);
+
+ static void OnRequestMessage(URLRequestAutomationJob* job,
+ IPC::Message* msg);
+
+ virtual bool OnMessageReceived(const IPC::Message& message);
+
+ // Add request to the list of outstanding requests.
+ virtual bool RegisterRequest(URLRequestAutomationJob* job);
+ // Remove request from the list of outstanding requests.
+ virtual void UnRegisterRequest(URLRequestAutomationJob* job);
+
+ protected:
+ AutomationProvider* automation_;
+ // declare the request map.
+ struct RequestJob {
+ MessageLoop* loop_;
+ scoped_refptr<URLRequestAutomationJob> job_;
+ };
+ typedef std::map<int, RequestJob> RequestMap;
+ RequestMap requests_;
+};
+
+#endif // CHROME_FRAME_TEST_NET_TEST_AUTOMATION_RESOURCE_MESSAGE_FILTER_H_
diff --git a/chrome_frame/test/perf/chrome_frame_perftest.cc b/chrome_frame/test/perf/chrome_frame_perftest.cc
new file mode 100644
index 0000000..79bbd7b
--- /dev/null
+++ b/chrome_frame/test/perf/chrome_frame_perftest.cc
@@ -0,0 +1,1137 @@
+// Copyright (c) 2006-2009 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 "chrome_frame/test/perf/chrome_frame_perftest.h"
+
+#include <atlwin.h>
+#include <atlhost.h>
+#include <map>
+#include <vector>
+#include <string>
+
+#include "chrome_tab.h" // Generated from chrome_tab.idl.
+
+#include "base/file_util.h"
+#include "base/registry.h"
+#include "base/scoped_ptr.h"
+#include "base/scoped_bstr_win.h"
+#include "base/scoped_comptr_win.h"
+#include "base/string_util.h"
+#include "base/time.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/common/chrome_paths.h"
+#include "chrome/test/chrome_process_util.h"
+#include "chrome/test/perf/mem_usage.h"
+#include "chrome/test/ui/ui_test.h"
+
+#include "chrome_frame/test_utils.h"
+#include "chrome_frame/utils.h"
+
+const wchar_t kSilverlightControlKey[] =
+ L"CLSID\\{DFEAF541-F3E1-4c24-ACAC-99C30715084A}\\InprocServer32";
+
+const wchar_t kFlashControlKey[] =
+ L"CLSID\\{D27CDB6E-AE6D-11cf-96B8-444553540000}\\InprocServer32";
+
+using base::TimeDelta;
+using base::TimeTicks;
+
+// Callback description for onload, onloaderror, onmessage
+static _ATL_FUNC_INFO g_single_param = {CC_STDCALL, VT_EMPTY, 1, {VT_VARIANT}};
+// Simple class that forwards the callbacks.
+template <typename T>
+class DispCallback
+ : public IDispEventSimpleImpl<1, DispCallback<T>, &IID_IDispatch> {
+ public:
+ typedef HRESULT (T::*Method)(VARIANT* param);
+
+ DispCallback(T* owner, Method method) : owner_(owner), method_(method) {
+ }
+
+ BEGIN_SINK_MAP(DispCallback)
+ SINK_ENTRY_INFO(1, IID_IDispatch, DISPID_VALUE, OnCallback, &g_single_param)
+ END_SINK_MAP()
+
+ virtual ULONG STDMETHODCALLTYPE AddRef() {
+ return owner_->AddRef();
+ }
+ virtual ULONG STDMETHODCALLTYPE Release() {
+ return owner_->Release();
+ }
+
+ STDMETHOD(OnCallback)(VARIANT param) {
+ return (owner_->*method_)(&param);
+ }
+
+ IDispatch* ToDispatch() {
+ return reinterpret_cast<IDispatch*>(this);
+ }
+
+ T* owner_;
+ Method method_;
+};
+
+// This class implements an ActiveX container which hosts the ChromeFrame
+// ActiveX control. It provides hooks which can be implemented by derived
+// classes for implementing performance measurement, etc.
+class ChromeFrameActiveXContainer
+ : public CWindowImpl<ChromeFrameActiveXContainer, CWindow, CWinTraits <
+ WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
+ WS_EX_APPWINDOW | WS_EX_WINDOWEDGE> >,
+ public CComObjectRootEx<CComSingleThreadModel>,
+ public IPropertyNotifySink {
+ public:
+ ~ChromeFrameActiveXContainer() {
+ if (m_hWnd)
+ DestroyWindow();
+ }
+
+ DECLARE_WND_CLASS_EX(L"ChromeFrameActiveX_container", 0, 0)
+
+ BEGIN_COM_MAP(ChromeFrameActiveXContainer)
+ COM_INTERFACE_ENTRY(IPropertyNotifySink)
+ END_COM_MAP()
+
+ BEGIN_MSG_MAP(ChromeFrameActiveXContainer)
+ MESSAGE_HANDLER(WM_CREATE, OnCreate)
+ MESSAGE_HANDLER(WM_DESTROY, OnDestroy)
+ END_MSG_MAP()
+
+ HRESULT OnMessageCallback(VARIANT* param) {
+ DLOG(INFO) << __FUNCTION__;
+ OnMessageCallbackImpl(param);
+ return S_OK;
+ }
+
+ HRESULT OnLoadErrorCallback(VARIANT* param) {
+ DLOG(INFO) << __FUNCTION__ << " " << param->bstrVal;
+ OnLoadErrorCallbackImpl(param);
+ return S_OK;
+ }
+
+ HRESULT OnLoadCallback(VARIANT* param) {
+ DLOG(INFO) << __FUNCTION__ << " " << param->bstrVal;
+ OnLoadCallbackImpl(param);
+ return S_OK;
+ }
+
+ ChromeFrameActiveXContainer() :
+ prop_notify_cookie_(0),
+ onmsg_(this, &ChromeFrameActiveXContainer::OnMessageCallback),
+ onloaderror_(this, &ChromeFrameActiveXContainer::OnLoadErrorCallback),
+ onload_(this, &ChromeFrameActiveXContainer::OnLoadCallback) {
+ }
+
+ LRESULT OnCreate(UINT , WPARAM , LPARAM , BOOL& ) {
+ chromeview_.Attach(m_hWnd);
+ return 0;
+ }
+
+ // This will be called twice.
+ // Once from CAxHostWindow::OnDestroy (through DefWindowProc)
+ // and once more from the ATL since CAxHostWindow::OnDestroy claims the
+ // message is not handled.
+ LRESULT OnDestroy(UINT, WPARAM, LPARAM, BOOL& handled) { // NOLINT
+ if (prop_notify_cookie_) {
+ AtlUnadvise(tab_, IID_IPropertyNotifySink, prop_notify_cookie_);
+ prop_notify_cookie_ = 0;
+ }
+
+ tab_.Release();
+ return 0;
+ }
+
+ virtual void OnFinalMessage(HWND /*hWnd*/) {
+ ::PostQuitMessage(6);
+ }
+
+ static const wchar_t* GetWndCaption() {
+ return L"ChromeFrame Container";
+ }
+
+ // IPropertyNotifySink
+ STDMETHOD(OnRequestEdit)(DISPID disp_id) {
+ OnRequestEditImpl(disp_id);
+ return S_OK;
+ }
+
+ STDMETHOD(OnChanged)(DISPID disp_id) {
+ if (disp_id != DISPID_READYSTATE)
+ return S_OK;
+
+ long ready_state;
+ HRESULT hr = tab_->get_readyState(&ready_state);
+ DCHECK(hr == S_OK);
+
+ OnReadyStateChanged(ready_state);
+
+ if (ready_state == READYSTATE_COMPLETE) {
+ if(!starting_url_.empty()) {
+ Navigate(starting_url_.c_str());
+ } else {
+ PostMessage(WM_CLOSE);
+ }
+ } else if (ready_state == READYSTATE_UNINITIALIZED) {
+ DLOG(ERROR) << __FUNCTION__ << " Chrome launch failed.";
+ }
+
+ return S_OK;
+ }
+
+ void CreateChromeFrameWindow(const std::string& starting_url) {
+ starting_url_ = starting_url;
+ RECT rc = { 0, 0, 800, 600 };
+ Create(NULL, rc);
+ DCHECK(m_hWnd);
+ ShowWindow(SW_SHOWDEFAULT);
+ }
+
+ void CreateControl(bool setup_event_sinks) {
+ HRESULT hr = chromeview_.CreateControl(L"ChromeTab.ChromeFrame");
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ hr = chromeview_.QueryControl(tab_.Receive());
+ EXPECT_HRESULT_SUCCEEDED(hr);
+
+ if (setup_event_sinks)
+ SetupEventSinks();
+ }
+
+ void Navigate(const char* url) {
+ BeforeNavigateImpl(url);
+
+ HRESULT hr = tab_->put_src(ScopedBstr(UTF8ToWide(url).c_str()));
+ DCHECK(hr == S_OK) << "Chrome frame NavigateToURL(" << url
+ << StringPrintf(L") failed 0x%08X", hr);
+ }
+
+ void SetupEventSinks() {
+ HRESULT hr = AtlAdvise(tab_, this, IID_IPropertyNotifySink,
+ &prop_notify_cookie_);
+ DCHECK(hr == S_OK) << "AtlAdvice for IPropertyNotifySink failed " << hr;
+
+ CComVariant onmessage(onmsg_.ToDispatch());
+ CComVariant onloaderror(onloaderror_.ToDispatch());
+ CComVariant onload(onload_.ToDispatch());
+ EXPECT_HRESULT_SUCCEEDED(tab_->put_onmessage(onmessage));
+ EXPECT_HRESULT_SUCCEEDED(tab_->put_onloaderror(onloaderror));
+ EXPECT_HRESULT_SUCCEEDED(tab_->put_onload(onload));
+ }
+
+ protected:
+ // These functions are implemented by derived classes for special behavior
+ // like performance measurement, etc.
+ virtual void OnReadyStateChanged(long ready_state) {}
+ virtual void OnRequestEditImpl(DISPID disp_id) {}
+
+ virtual void OnMessageCallbackImpl(VARIANT* param) {}
+
+ virtual void OnLoadCallbackImpl(VARIANT* param) {
+ PostMessage(WM_CLOSE);
+ }
+
+ virtual void OnLoadErrorCallbackImpl(VARIANT* param) {
+ PostMessage(WM_CLOSE);
+ }
+ virtual void BeforeNavigateImpl(const char* url) {}
+
+ CAxWindow chromeview_;
+ ScopedComPtr<IChromeFrame> tab_;
+ DWORD prop_notify_cookie_;
+ DispCallback<ChromeFrameActiveXContainer> onmsg_;
+ DispCallback<ChromeFrameActiveXContainer> onloaderror_;
+ DispCallback<ChromeFrameActiveXContainer> onload_;
+ std::string starting_url_;
+};
+
+// This class overrides the hooks provided by the ChromeFrameActiveXContainer
+// class and measures performance at various stages, like initialzation of
+// the Chrome frame widget, navigation, etc.
+class ChromeFrameActiveXContainerPerf : public ChromeFrameActiveXContainer {
+ public:
+ ChromeFrameActiveXContainerPerf() {}
+
+ void CreateControl(bool setup_event_sinks) {
+ perf_initialize_.reset(new PerfTimeLogger("Fully initialized"));
+ PerfTimeLogger perf_create("Create Control");
+
+ HRESULT hr = chromeview_.CreateControl(L"ChromeTab.ChromeFrame");
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ hr = chromeview_.QueryControl(tab_.Receive());
+ EXPECT_HRESULT_SUCCEEDED(hr);
+
+ perf_create.Done();
+ if (setup_event_sinks)
+ SetupEventSinks();
+ }
+
+ protected:
+ virtual void OnReadyStateChanged(long ready_state) {
+ // READYSTATE_COMPLETE is fired when the automation server is ready.
+ if (ready_state == READYSTATE_COMPLETE) {
+ perf_initialize_->Done();
+ } else if (ready_state == READYSTATE_INTERACTIVE) {
+ // Window ready. Currently we never receive this notification because it
+ // is fired before we finish setting up our hosting environment.
+ // This is because of how ATL is written. Moving forward we might
+ // have our own hosting classes and then have more control over when we
+ // set up the prop notify sink.
+ } else {
+ DCHECK(ready_state != READYSTATE_UNINITIALIZED) << "failed to initialize";
+ }
+ }
+
+ virtual void OnLoadCallbackImpl(VARIANT* param) {
+ PostMessage(WM_CLOSE);
+ perf_navigate_->Done();
+ }
+
+ virtual void OnLoadErrorCallbackImpl(VARIANT* param) {
+ PostMessage(WM_CLOSE);
+ perf_navigate_->Done();
+ }
+
+ virtual void BeforeNavigateImpl(const char* url ) {
+ std::string test_name = "Navigate ";
+ test_name += url;
+ perf_navigate_.reset(new PerfTimeLogger(test_name.c_str()));
+ }
+
+ scoped_ptr<PerfTimeLogger> perf_initialize_;
+ scoped_ptr<PerfTimeLogger> perf_navigate_;
+};
+
+// This class provides common functionality which can be used for most of the
+// ChromeFrame/Tab performance tests.
+class ChromeFramePerfTestBase : public UITest {
+ public:
+ ChromeFramePerfTestBase() {}
+ protected:
+ scoped_ptr<ScopedChromeFrameRegistrar> chrome_frame_registrar_;
+};
+
+class ChromeFrameStartupTest : public ChromeFramePerfTestBase {
+ public:
+ ChromeFrameStartupTest() {}
+
+ virtual void SetUp() {
+ ASSERT_TRUE(PathService::Get(chrome::DIR_APP, &dir_app_));
+
+ chrome_dll_ = dir_app_.Append(FILE_PATH_LITERAL("chrome.dll"));
+ chrome_exe_ = dir_app_.Append(
+ FilePath::FromWStringHack(chrome::kBrowserProcessExecutableName));
+ chrome_frame_dll_ = dir_app_.Append(
+ FILE_PATH_LITERAL("servers\\npchrome_tab.dll"));
+ }
+ virtual void TearDown() {}
+
+ // TODO(iyengar)
+ // This function is similar to the RunStartupTest function used in chrome
+ // startup tests. Refactor into a common implementation.
+ void RunStartupTest(const char* graph, const char* trace,
+ const char* startup_url, bool test_cold,
+ int total_binaries, const FilePath binaries_to_evict[],
+ bool important, bool ignore_cache_error) {
+ const int kNumCycles = 20;
+
+ startup_url_ = startup_url;
+
+ TimeDelta timings[kNumCycles];
+
+ for (int i = 0; i < kNumCycles; ++i) {
+ if (test_cold) {
+ for (int binary_index = 0; binary_index < total_binaries;
+ binary_index++) {
+ bool result = EvictFileFromSystemCacheWrapper(
+ binaries_to_evict[binary_index]);
+ if (!ignore_cache_error) {
+ ASSERT_TRUE(result);
+ } else if (!result) {
+ printf("\nFailed to evict file %ls from cache. Not running test\n",
+ binaries_to_evict[binary_index].value().c_str());
+ return;
+ }
+ }
+ }
+
+ TimeTicks start_time, end_time;
+
+ RunStartupTestImpl(&start_time, &end_time);
+
+ timings[i] = end_time - start_time;
+
+ CoFreeUnusedLibraries();
+ ASSERT_TRUE(GetModuleHandle(L"npchrome_tab.dll") == NULL);
+
+ // TODO(beng): Can't shut down so quickly. Figure out why, and fix. If we
+ // do, we crash.
+ PlatformThread::Sleep(50);
+ }
+
+ std::string times;
+ for (int i = 0; i < kNumCycles; ++i)
+ StringAppendF(&times, "%.2f,", timings[i].InMillisecondsF());
+
+ PrintResultList(graph, "", trace, times, "ms", important);
+ }
+
+ FilePath dir_app_;
+ FilePath chrome_dll_;
+ FilePath chrome_exe_;
+ FilePath chrome_frame_dll_;
+
+ protected:
+ // Individual startup tests should implement this function.
+ virtual void RunStartupTestImpl(TimeTicks* start_time,
+ TimeTicks* end_time) {}
+
+ // The host is torn down by this function. It should not be used after
+ // this function returns.
+ static void ReleaseHostComReferences(CAxWindow& host) {
+ CComPtr<IAxWinHostWindow> spWinHost;
+ host.QueryHost(&spWinHost);
+ ASSERT_TRUE(spWinHost != NULL);
+
+ // Hack to get the host to release all interfaces and thus ensure that
+ // the COM server can be unloaded.
+ CAxHostWindow* host_window = static_cast<CAxHostWindow*>(spWinHost.p);
+ host_window->ReleaseAll();
+ host.DestroyWindow();
+ }
+
+ std::string startup_url_;
+};
+
+class ChromeFrameStartupTestActiveX : public ChromeFrameStartupTest {
+ public:
+ virtual void SetUp() {
+ // Register the Chrome Frame DLL in the build directory.
+ chrome_frame_registrar_.reset(new ScopedChromeFrameRegistrar);
+
+ ChromeFrameStartupTest::SetUp();
+ }
+
+ protected:
+ virtual void RunStartupTestImpl(TimeTicks* start_time,
+ TimeTicks* end_time) {
+ *start_time = TimeTicks::Now();
+ SimpleModule module;
+ AtlAxWinInit();
+ CComObjectStackEx<ChromeFrameActiveXContainer> wnd;
+ wnd.CreateChromeFrameWindow(startup_url_);
+ wnd.CreateControl(true);
+ module.RunMessageLoop();
+ *end_time = TimeTicks::Now();
+ }
+};
+
+// This class measures the load time of chrome and chrome frame binaries
+class ChromeFrameBinariesLoadTest : public ChromeFrameStartupTestActiveX {
+ protected:
+ virtual void RunStartupTestImpl(TimeTicks* start_time,
+ TimeTicks* end_time) {
+ *start_time = TimeTicks::Now();
+
+ HMODULE chrome_exe = LoadLibrary(chrome_exe_.ToWStringHack().c_str());
+ ASSERT_TRUE(chrome_exe != NULL);
+
+ HMODULE chrome_dll = LoadLibrary(chrome_dll_.ToWStringHack().c_str());
+ ASSERT_TRUE(chrome_dll != NULL);
+
+ HMODULE chrome_tab_dll =
+ LoadLibrary(chrome_frame_dll_.ToWStringHack().c_str());
+ ASSERT_TRUE(chrome_tab_dll != NULL);
+
+ *end_time = TimeTicks::Now();
+
+ FreeLibrary(chrome_exe);
+ FreeLibrary(chrome_dll);
+ FreeLibrary(chrome_tab_dll);
+ }
+};
+
+// This class provides functionality to run the startup performance test for
+// the ChromeFrame ActiveX against a reference build. At this point we only run
+// this test in warm mode.
+class ChromeFrameStartupTestActiveXReference
+ : public ChromeFrameStartupTestActiveX {
+ public:
+ // override the browser directory to use the reference build instead.
+ virtual void SetUp() {
+ // Register the reference build Chrome Frame DLL.
+ chrome_frame_registrar_.reset(new ScopedChromeFrameRegistrar);
+ chrome_frame_registrar_->RegisterReferenceChromeFrameBuild();
+
+ ChromeFrameStartupTest::SetUp();
+ chrome_frame_dll_ = FilePath::FromWStringHack(
+ chrome_frame_registrar_->GetChromeFrameDllPath());
+ }
+
+ virtual void TearDown() {
+ // Reregister the Chrome Frame DLL in the build directory.
+ chrome_frame_registrar_.reset(NULL);
+ }
+};
+
+// This class provides base functionality to measure ChromeFrame memory
+// usage.
+// TODO(iyengar)
+// Some of the functionality in this class like printing the results, etc
+// is based on the chrome\test\memory_test.cc. We need to factor out
+// the common code.
+class ChromeFrameMemoryTest : public ChromeFramePerfTestBase {
+
+ // Contains information about the memory consumption of a process.
+ class ProcessMemoryInfo {
+ public:
+ // Default constructor
+ // Added to enable us to add ProcessMemoryInfo instances to a map.
+ ProcessMemoryInfo()
+ : process_id_(0),
+ peak_virtual_size_(0),
+ virtual_size_(0),
+ peak_working_set_size_(0),
+ working_set_size_(0),
+ chrome_browser_process_(false),
+ chrome_frame_memory_test_instance_(NULL) {}
+
+ ProcessMemoryInfo(base::ProcessId process_id, bool chrome_browser_process,
+ ChromeFrameMemoryTest* memory_test_instance)
+ : process_id_(process_id),
+ peak_virtual_size_(0),
+ virtual_size_(0),
+ peak_working_set_size_(0),
+ working_set_size_(0),
+ chrome_browser_process_(chrome_browser_process),
+ chrome_frame_memory_test_instance_(memory_test_instance) {}
+
+ bool GetMemoryConsumptionDetails() {
+ return GetMemoryInfo(process_id_,
+ &peak_virtual_size_,
+ &virtual_size_,
+ &peak_working_set_size_,
+ &working_set_size_);
+ }
+
+ void Print(const char* test_name) {
+ std::string trace_name(test_name);
+
+ ASSERT_TRUE(chrome_frame_memory_test_instance_ != NULL);
+
+ if (chrome_browser_process_) {
+ chrome_frame_memory_test_instance_->PrintResult(
+ "vm_final_browser", "", trace_name + "_vm_b",
+ virtual_size_ / 1024, "KB", false /* not important */);
+ chrome_frame_memory_test_instance_->PrintResult(
+ "ws_final_browser", "", trace_name + "_ws_b",
+ working_set_size_ / 1024, "KB", false /* not important */);
+ } else if (process_id_ == GetCurrentProcessId()) {
+ chrome_frame_memory_test_instance_->PrintResult(
+ "vm_current_process", "", trace_name + "_vm_c",
+ virtual_size_ / 1024, "KB", false /* not important */);
+ chrome_frame_memory_test_instance_->PrintResult(
+ "ws_current_process", "", trace_name + "_ws_c",
+ working_set_size_ / 1024, "KB", false /* not important */);
+ }
+
+ printf("\n");
+ }
+
+ int process_id_;
+ size_t peak_virtual_size_;
+ size_t virtual_size_;
+ size_t peak_working_set_size_;
+ size_t working_set_size_;
+ // Set to true if this is the chrome browser process.
+ bool chrome_browser_process_;
+
+ // A reference to the ChromeFrameMemoryTest instance. Used to print memory
+ // consumption information.
+ ChromeFrameMemoryTest* chrome_frame_memory_test_instance_;
+ };
+
+ // This map tracks memory usage for a process. It is keyed on the process
+ // id.
+ typedef std::map<DWORD, ProcessMemoryInfo> ProcessMemoryConsumptionMap;
+
+ public:
+ ChromeFrameMemoryTest()
+ : current_url_index_(0),
+ browser_pid_(0) {}
+
+ virtual void SetUp() {
+ // Register the Chrome Frame DLL in the build directory.
+ chrome_frame_registrar_.reset(new ScopedChromeFrameRegistrar);
+ }
+
+ void RunTest(const char* test_name, char* urls[], int total_urls) {
+ ASSERT_TRUE(urls != NULL);
+ ASSERT_GT(total_urls, 0);
+
+ // Record the initial CommitCharge. This is a system-wide measurement,
+ // so if other applications are running, they can create variance in this
+ // test.
+ start_commit_charge_ = GetSystemCommitCharge();
+
+ for (int i = 0; i < total_urls; i++)
+ urls_.push_back(urls[i]);
+
+ std::string url;
+ GetNextUrl(&url);
+ ASSERT_TRUE(!url.empty());
+
+ StartTest(url, test_name);
+ }
+
+ void OnNavigationSuccess(VARIANT* param) {
+ ASSERT_TRUE(param != NULL);
+ ASSERT_EQ(VT_BSTR, param->vt);
+
+ DLOG(INFO) << __FUNCTION__ << " " << param->bstrVal;
+ InitiateNextNavigation();
+ }
+
+ void OnNavigationFailure(VARIANT* param) {
+ ASSERT_TRUE(param != NULL);
+ ASSERT_EQ(VT_BSTR, param->vt);
+
+ DLOG(INFO) << __FUNCTION__ << " " << param->bstrVal;
+ InitiateNextNavigation();
+ }
+
+ protected:
+ bool GetNextUrl(std::string* url) {
+ if (current_url_index_ >= urls_.size())
+ return false;
+
+ *url = urls_[current_url_index_++];
+ return true;
+ }
+
+ // Returns the path of the current chrome.exe being used by this test.
+ // This could return the regular chrome path or that of the reference
+ // build.
+ std::wstring GetChromeExePath() {
+ std::wstring chrome_exe_path =
+ chrome_frame_registrar_->GetChromeFrameDllPath();
+ EXPECT_FALSE(chrome_exe_path.empty());
+
+ file_util::UpOneDirectory(&chrome_exe_path);
+
+ std::wstring chrome_exe_test_path = chrome_exe_path;
+ file_util::AppendToPath(&chrome_exe_test_path,
+ chrome::kBrowserProcessExecutableName);
+
+ if (!file_util::PathExists(chrome_exe_test_path)) {
+ file_util::UpOneDirectory(&chrome_exe_path);
+
+ chrome_exe_test_path = chrome_exe_path;
+ file_util::AppendToPath(&chrome_exe_test_path,
+ chrome::kBrowserProcessExecutableName);
+ }
+
+ EXPECT_TRUE(file_util::PathExists(chrome_exe_test_path));
+
+ return chrome_exe_path;
+ }
+
+ void InitiateNextNavigation() {
+ if (browser_pid_ == 0) {
+ std::wstring profile_directory;
+ if (GetUserProfileBaseDirectory(&profile_directory)) {
+ file_util::AppendToPath(&profile_directory,
+ GetHostProcessName(false));
+ }
+
+ user_data_dir_ = FilePath::FromWStringHack(profile_directory);
+ browser_pid_ = ChromeBrowserProcessId(user_data_dir_);
+ }
+
+ EXPECT_TRUE(static_cast<int>(browser_pid_) > 0);
+
+ // Get the memory consumption information for the child processes
+ // of the chrome browser.
+ ChromeProcessList child_processes = GetBrowserChildren();
+ ChromeProcessList::iterator index;
+ for (index = child_processes.begin(); index != child_processes.end();
+ ++index) {
+ AccountProcessMemoryUsage(*index);
+ }
+
+ // TODO(iyengar): Bug 2953
+ // Need to verify if this is still true.
+ // The automation crashes periodically if we cycle too quickly.
+ // To make these tests more reliable, slowing them down a bit.
+ Sleep(200);
+
+ std::string url;
+ bool next_url = GetNextUrl(&url);
+ if (!url.empty()) {
+ NavigateImpl(url);
+ } else {
+ TestCompleted();
+ }
+ }
+
+ void PrintResults(const char* test_name) {
+ PrintMemoryUsageInfo(test_name);
+ memory_consumption_map_.clear();
+
+ // Added to give the OS some time to flush the used pages for the
+ // chrome processes which would have exited by now.
+ Sleep(200);
+
+ size_t end_commit_charge = GetSystemCommitCharge();
+ size_t commit_size = end_commit_charge - start_commit_charge_;
+
+ std::string trace_name(test_name);
+ trace_name.append("_cc");
+
+ PrintResult("commit_charge", "", trace_name,
+ commit_size / 1024, "KB", true /* important */);
+ printf("\n");
+ }
+
+ ChromeProcessList GetBrowserChildren() {
+ ChromeProcessList list = GetRunningChromeProcesses(user_data_dir_);
+ ChromeProcessList::iterator browser =
+ std::find(list.begin(), list.end(), browser_pid_);
+ if (browser != list.end()) {
+ list.erase(browser);
+ }
+ return list;
+ }
+
+ void AccountProcessMemoryUsage(DWORD process_id) {
+ ProcessMemoryInfo process_memory_info(process_id,
+ process_id == browser_pid_, this);
+
+ ASSERT_TRUE(process_memory_info.GetMemoryConsumptionDetails());
+
+ memory_consumption_map_[process_id] = process_memory_info;
+ }
+
+ void PrintMemoryUsageInfo(const char* test_name) {
+ printf("\n");
+
+ std::string trace_name(test_name);
+
+ ProcessMemoryConsumptionMap::iterator index;
+ size_t total_virtual_size = 0;
+ size_t total_working_set_size = 0;
+
+ for (index = memory_consumption_map_.begin();
+ index != memory_consumption_map_.end();
+ ++index) {
+ ProcessMemoryInfo& memory_info = (*index).second;
+ memory_info.Print(test_name);
+
+ total_virtual_size += memory_info.virtual_size_;
+ total_working_set_size += memory_info.working_set_size_;
+ }
+
+ printf("\n");
+
+ PrintResult("vm_final_total", "", trace_name + "_vm",
+ total_virtual_size / 1024, "KB",
+ false /* not important */);
+ PrintResult("ws_final_total", "", trace_name + "_ws",
+ total_working_set_size / 1024, "KB",
+ true /* important */);
+ }
+
+ // Should never get called.
+ virtual void StartTest(const std::string& url,
+ const std::string& test_name) = 0 {
+ ASSERT_FALSE(false);
+ }
+
+ // Should never get called.
+ virtual void NavigateImpl(const std::string& url) = 0 {
+ ASSERT_FALSE(false);
+ }
+
+ virtual void TestCompleted() = 0 {
+ ASSERT_FALSE(false);
+ }
+
+ // Holds the commit charge at the start of the memory test run.
+ size_t start_commit_charge_;
+
+ // The index of the URL being tested.
+ size_t current_url_index_;
+
+ // The chrome browser pid.
+ base::ProcessId browser_pid_;
+
+ // Contains the list of urls against which the tests are run.
+ std::vector<std::string> urls_;
+
+ ProcessMemoryConsumptionMap memory_consumption_map_;
+};
+
+// This class provides functionality to run the memory test against a reference
+// chrome frame build.
+class ChromeFrameMemoryTestReference : public ChromeFrameMemoryTest {
+ public:
+ virtual void SetUp() {
+ chrome_frame_registrar_.reset(new ScopedChromeFrameRegistrar);
+ chrome_frame_registrar_->RegisterReferenceChromeFrameBuild();
+ }
+
+ virtual void TearDown() {
+ // Reregisters the chrome frame DLL in the build directory.
+ chrome_frame_registrar_.reset(NULL);
+ }
+};
+
+// This class overrides the hooks provided by the ChromeFrameActiveXContainer
+// class and calls back into the ChromeFrameMemoryTest object instance,
+// which measures ChromeFrame memory usage.
+class ChromeFrameActiveXContainerMemory : public ChromeFrameActiveXContainer {
+ public:
+ ChromeFrameActiveXContainerMemory()
+ : delegate_(NULL) {}
+
+ ~ChromeFrameActiveXContainerMemory() {}
+
+ void Initialize(ChromeFrameMemoryTest* delegate) {
+ ASSERT_TRUE(delegate != NULL);
+ delegate_ = delegate;
+ }
+
+ protected:
+ virtual void OnLoadCallbackImpl(VARIANT* param) {
+ delegate_->OnNavigationSuccess(param);
+ }
+
+ virtual void OnLoadErrorCallbackImpl(VARIANT* param) {
+ delegate_->OnNavigationFailure(param);
+ }
+
+ ChromeFrameMemoryTest* delegate_;
+};
+
+// This class runs memory tests against the ChromeFrame ActiveX.
+template<class MemoryTestBase>
+class ChromeFrameActiveXMemoryTest : public MemoryTestBase {
+ public:
+ ChromeFrameActiveXMemoryTest()
+ : chrome_frame_container_(NULL),
+ test_completed_(false) {}
+
+ ~ChromeFrameActiveXMemoryTest() {
+ }
+
+ void StartTest(const std::string& url, const std::string& test_name) {
+ ASSERT_TRUE(chrome_frame_container_ == NULL);
+
+ test_name_ = test_name;
+
+ SimpleModule module;
+ AtlAxWinInit();
+
+ CComObject<ChromeFrameActiveXContainerMemory>::CreateInstance(
+ &chrome_frame_container_);
+ chrome_frame_container_->AddRef();
+
+ chrome_frame_container_->Initialize(this);
+
+ chrome_frame_container_->CreateChromeFrameWindow(url.c_str());
+ chrome_frame_container_->CreateControl(true);
+
+ module.RunMessageLoop();
+
+ chrome_frame_container_->Release();
+
+ PrintResults(test_name_.c_str());
+
+ CoFreeUnusedLibraries();
+ //ASSERT_TRUE(GetModuleHandle(L"npchrome_tab.dll") == NULL);
+ }
+
+ void NavigateImpl(const std::string& url) {
+ ASSERT_TRUE(chrome_frame_container_ != NULL);
+ ASSERT_TRUE(!url.empty());
+ chrome_frame_container_->Navigate(url.c_str());
+ }
+
+ void TestCompleted() {
+ // This can get called multiple times if the last url results in a
+ // redirect.
+ if (!test_completed_) {
+ ASSERT_NE(browser_pid_, 0);
+
+ // Measure memory usage for the browser process.
+ AccountProcessMemoryUsage(browser_pid_);
+ // Measure memory usage for the current process.
+ AccountProcessMemoryUsage(GetCurrentProcessId());
+
+ test_completed_ = true;
+ EXPECT_TRUE(PostMessage(static_cast<HWND>(*chrome_frame_container_),
+ WM_CLOSE, 0, 0));
+ }
+ }
+
+ protected:
+ CComObject<ChromeFrameActiveXContainerMemory>* chrome_frame_container_;
+ std::string test_name_;
+ bool test_completed_;
+};
+
+// This class runs tests to measure chrome frame creation only. This will help
+// track overall page load performance with chrome frame instances.
+class ChromeFrameCreationTest : public ChromeFrameStartupTest {
+ protected:
+ virtual void RunStartupTestImpl(TimeTicks* start_time,
+ TimeTicks* end_time) {
+ SimpleModule module;
+ AtlAxWinInit();
+ CComObjectStackEx<ChromeFrameActiveXContainer> wnd;
+ wnd.CreateChromeFrameWindow(startup_url_);
+ *start_time = TimeTicks::Now();
+ wnd.CreateControl(false);
+ *end_time = TimeTicks::Now();
+ }
+};
+
+// This class provides functionality to run the chrome frame
+// performance test against a reference build.
+class ChromeFrameCreationTestReference : public ChromeFrameCreationTest {
+ public:
+ // override the browser directory to use the reference build instead.
+ virtual void SetUp() {
+ chrome_frame_registrar_.reset(new ScopedChromeFrameRegistrar);
+ chrome_frame_registrar_->RegisterReferenceChromeFrameBuild();
+ ChromeFrameStartupTest::SetUp();
+ }
+
+ virtual void TearDown() {
+ chrome_frame_registrar_.reset(NULL);
+ }
+};
+
+// This class measures the creation time for Flash, which would be used
+// as a baseline to measure chrome frame creation performance.
+class FlashCreationTest : public ChromeFrameStartupTest {
+ protected:
+ virtual void RunStartupTestImpl(TimeTicks* start_time,
+ TimeTicks* end_time) {
+ SimpleModule module;
+ AtlAxWinInit();
+ CAxWindow host;
+ RECT rc = {0, 0, 800, 600};
+ host.Create(NULL, rc, NULL,
+ WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
+ WS_EX_APPWINDOW | WS_EX_WINDOWEDGE);
+ EXPECT_TRUE(host.m_hWnd != NULL);
+
+ *start_time = TimeTicks::Now();
+ HRESULT hr = host.CreateControl(L"ShockwaveFlash.ShockwaveFlash");
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ *end_time = TimeTicks::Now();
+
+ ReleaseHostComReferences(host);
+ }
+};
+
+// This class measures the creation time for Silverlight, which would be used
+// as a baseline to measure chrome frame creation performance.
+class SilverlightCreationTest : public ChromeFrameStartupTest {
+ protected:
+ virtual void RunStartupTestImpl(TimeTicks* start_time,
+ TimeTicks* end_time) {
+ SimpleModule module;
+ AtlAxWinInit();
+ CAxWindow host;
+ RECT rc = {0, 0, 800, 600};
+ host.Create(NULL, rc, NULL,
+ WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
+ WS_EX_APPWINDOW | WS_EX_WINDOWEDGE);
+ EXPECT_TRUE(host.m_hWnd != NULL);
+
+ *start_time = TimeTicks::Now();
+ HRESULT hr = host.CreateControl(L"AgControl.AgControl");
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ *end_time = TimeTicks::Now();
+
+ ReleaseHostComReferences(host);
+ }
+};
+
+TEST(ChromeFramePerf, DISABLED_HostActiveX) {
+ // TODO(stoyan): Create a low integrity level thread && perform the test there
+ SimpleModule module;
+ AtlAxWinInit();
+ CComObjectStackEx<ChromeFrameActiveXContainerPerf> wnd;
+ wnd.CreateChromeFrameWindow("http://www.google.com");
+ wnd.CreateControl(true);
+ module.RunMessageLoop();
+}
+
+TEST(ChromeFramePerf, DISABLED_HostActiveXInvalidURL) {
+ // TODO(stoyan): Create a low integrity level thread && perform the test there
+ SimpleModule module;
+ AtlAxWinInit();
+ CComObjectStackEx<ChromeFrameActiveXContainerPerf> wnd;
+ wnd.CreateChromeFrameWindow("http://non-existent-domain.org/");
+ wnd.CreateControl(true);
+ module.RunMessageLoop();
+}
+
+TEST_F(ChromeFrameStartupTestActiveX, PerfWarm) {
+ RunStartupTest("warm", "t", "about:blank", false /* cold */, 0, NULL,
+ true /* important */, false);
+}
+
+TEST_F(ChromeFrameBinariesLoadTest, PerfWarm) {
+ RunStartupTest("binary_load_warm", "t", "", false /* cold */, 0, NULL,
+ true /* important */, false);
+}
+
+TEST_F(ChromeFrameStartupTestActiveX, PerfCold) {
+ FilePath binaries_to_evict[] = {chrome_exe_, chrome_dll_, chrome_frame_dll_};
+ RunStartupTest("cold", "t", "about:blank", true /* cold */,
+ arraysize(binaries_to_evict), binaries_to_evict,
+ false /* not important */, false);
+}
+
+TEST_F(ChromeFrameBinariesLoadTest, PerfCold) {
+ FilePath binaries_to_evict[] = {chrome_exe_, chrome_dll_, chrome_frame_dll_};
+ RunStartupTest("binary_load_cold", "t", "", true /* cold */,
+ arraysize(binaries_to_evict), binaries_to_evict,
+ false /* not important */, false);
+}
+
+TEST_F(ChromeFrameStartupTestActiveXReference, PerfWarm) {
+ RunStartupTest("warm", "t_ref", "about:blank", false /* cold */, 0, NULL,
+ true /* important */, false);
+}
+
+TEST_F(ChromeFrameStartupTestActiveX, PerfChromeFrameInitializationWarm) {
+ RunStartupTest("ChromeFrame_init_warm", "t", "", false /* cold */, 0,
+ NULL, true /* important */, false);
+}
+
+TEST_F(ChromeFrameStartupTestActiveX, PerfChromeFrameInitializationCold) {
+ FilePath binaries_to_evict[] = {chrome_frame_dll_};
+ RunStartupTest("ChromeFrame_init_cold", "t", "", true /* cold */,
+ arraysize(binaries_to_evict), binaries_to_evict,
+ false /* not important */, false);
+}
+
+TEST_F(ChromeFrameStartupTestActiveXReference,
+ PerfChromeFrameInitializationWarm) {
+ RunStartupTest("ChromeFrame_init_warm", "t_ref", "", false /* cold */, 0,
+ NULL, true /* important */, false);
+}
+
+typedef ChromeFrameActiveXMemoryTest<ChromeFrameMemoryTest>
+ RegularChromeFrameActiveXMemoryTest;
+
+TEST_F(RegularChromeFrameActiveXMemoryTest, MemoryTestAboutBlank) {
+ char *urls[] = {"about:blank"};
+ RunTest("memory_about_blank", urls, arraysize(urls));
+}
+
+// TODO(iyengar)
+// Revisit why the chrome frame dll does not unload correctly when this test is
+// run.
+TEST_F(RegularChromeFrameActiveXMemoryTest, DISABLED_MemoryTestUrls) {
+ // TODO(iyengar)
+ // We should use static pages to measure memory usage.
+ char *urls[] = {
+ "http://www.youtube.com/watch?v=PN2HAroA12w",
+ "http://www.youtube.com/watch?v=KmLJDrsaJmk&feature=channel"
+ };
+
+ RunTest("memory", urls, arraysize(urls));
+}
+
+typedef ChromeFrameActiveXMemoryTest<ChromeFrameMemoryTestReference>
+ ReferenceBuildChromeFrameActiveXMemoryTest;
+
+TEST_F(ReferenceBuildChromeFrameActiveXMemoryTest, MemoryTestAboutBlank) {
+ char *urls[] = {"about:blank"};
+ RunTest("memory_about_blank_reference", urls, arraysize(urls));
+}
+
+// TODO(iyengar)
+// Revisit why the chrome frame dll does not unload correctly when this test is
+// run.
+TEST_F(ReferenceBuildChromeFrameActiveXMemoryTest, DISABLED_MemoryTestUrls) {
+ // TODO(iyengar)
+ // We should use static pages to measure memory usage.
+ char *urls[] = {
+ "http://www.youtube.com/watch?v=PN2HAroA12w",
+ "http://www.youtube.com/watch?v=KmLJDrsaJmk&feature=channel"
+ };
+
+ RunTest("memory_reference", urls, arraysize(urls));
+}
+
+TEST_F(ChromeFrameCreationTest, PerfWarm) {
+ RunStartupTest("creation_warm", "t", "", false /* cold */, 0,
+ NULL, true /* important */, false);
+}
+
+TEST_F(ChromeFrameCreationTestReference, PerfWarm) {
+ RunStartupTest("creation_warm", "t_ref", "about:blank", false /* cold */, 0,
+ NULL, true /* not important */, false);
+}
+
+TEST_F(FlashCreationTest, PerfWarm) {
+ RunStartupTest("creation_warm", "t_flash", "", false /* cold */, 0, NULL,
+ true /* not important */, false);
+}
+
+TEST_F(SilverlightCreationTest, DISABLED_PerfWarm) {
+ RunStartupTest("creation_warm", "t_silverlight", "", false /* cold */, 0,
+ NULL, false /* not important */, false);
+}
+
+TEST_F(ChromeFrameCreationTest, PerfCold) {
+ FilePath binaries_to_evict[] = {chrome_frame_dll_};
+
+ RunStartupTest("creation_cold", "t", "", true /* cold */,
+ arraysize(binaries_to_evict), binaries_to_evict,
+ true /* important */, false);
+}
+
+// Attempt to evict the Flash control can fail on the buildbot as the dll
+// is marked read only. The test run is aborted if we fail to evict the file
+// from the cache. This could also fail if the Flash control is in use.
+// On Vista this could fail because of UAC
+TEST_F(FlashCreationTest, PerfCold) {
+ RegKey flash_key(HKEY_CLASSES_ROOT, kFlashControlKey);
+
+ std::wstring plugin_path;
+ ASSERT_TRUE(flash_key.ReadValue(L"", &plugin_path));
+ ASSERT_FALSE(plugin_path.empty());
+
+ FilePath flash_path = FilePath::FromWStringHack(plugin_path);
+ FilePath binaries_to_evict[] = {flash_path};
+
+ RunStartupTest("creation_cold", "t_flash", "", true /* cold */,
+ arraysize(binaries_to_evict), binaries_to_evict,
+ false/* important */, true);
+}
+
+// This test would fail on Vista due to UAC or if the Silverlight control is
+// in use. The test run is aborted if we fail to evict the file from the cache.
+// Disabling this test as the Silverlight dll does not seem to get unloaded
+// correctly causing the attempt to evict the dll from the system cache to
+// fail.
+TEST_F(SilverlightCreationTest, DISABLED_PerfCold) {
+ RegKey silverlight_key(HKEY_CLASSES_ROOT, kSilverlightControlKey);
+
+ std::wstring plugin_path;
+ ASSERT_TRUE(silverlight_key.ReadValue(L"", &plugin_path));
+ ASSERT_FALSE(plugin_path.empty());
+
+ FilePath silverlight_path = FilePath::FromWStringHack(plugin_path);
+ FilePath binaries_to_evict[] = {silverlight_path};
+
+ RunStartupTest("creation_cold", "t_silverlight", "", true /* cold */,
+ arraysize(binaries_to_evict), binaries_to_evict,
+ false /* important */, true);
+}
diff --git a/chrome_frame/test/perf/chrome_frame_perftest.h b/chrome_frame/test/perf/chrome_frame_perftest.h
new file mode 100644
index 0000000..5b895f3
--- /dev/null
+++ b/chrome_frame/test/perf/chrome_frame_perftest.h
@@ -0,0 +1,21 @@
+// Copyright (c) 2006-2009 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.
+#ifndef CHROME_FRAME_TEST_PERF_CHROME_FRAME_PERFTEST_H_
+#define CHROME_FRAME_TEST_PERF_CHROME_FRAME_PERFTEST_H_
+#include <atlbase.h>
+#include "base/logging.h"
+#include "base/perftimer.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+class SimpleModule : public CAtlExeModuleT<SimpleModule> {
+ public:
+ // The ATL code does not set _pAtlModule to NULL on destruction, and therefore
+ // creating new module (for another test) will ASSERT in constructor.
+ ~SimpleModule() {
+ Term();
+ _pAtlModule = NULL;
+ }
+};
+#endif // CHROME_FRAME_TEST_PERF_CHROME_FRAME_PERFTEST_H_
+
diff --git a/chrome_frame/test/perf/chrometab_perftests.vcproj b/chrome_frame/test/perf/chrometab_perftests.vcproj
new file mode 100644
index 0000000..b894e6a
--- /dev/null
+++ b/chrome_frame/test/perf/chrometab_perftests.vcproj
@@ -0,0 +1,247 @@
+<?xml version="1.0" encoding="Windows-1252"?>
+<VisualStudioProject
+ ProjectType="Visual C++"
+ Version="8.00"
+ Name="chrometab_perftests"
+ ProjectGUID="{760ABE9D-2B3E-48C5-A571-6CD221A04BD6}"
+ RootNamespace="chrometab_perftests"
+ Keyword="Win32Proj"
+ >
+ <Platforms>
+ <Platform
+ Name="Win32"
+ />
+ </Platforms>
+ <ToolFiles>
+ </ToolFiles>
+ <Configurations>
+ <Configuration
+ Name="Debug|Win32"
+ OutputDirectory="..\..\chrome\$(ConfigurationName)"
+ IntermediateDirectory="$(ConfigurationName)"
+ ConfigurationType="1"
+ InheritedPropertySheets=".\chrometab_perftests.vsprops;$(SolutionDir)..\build\debug.vsprops"
+ >
+ <Tool
+ Name="VCPreBuildEventTool"
+ />
+ <Tool
+ Name="VCCustomBuildTool"
+ />
+ <Tool
+ Name="VCXMLDataGeneratorTool"
+ />
+ <Tool
+ Name="VCWebServiceProxyGeneratorTool"
+ />
+ <Tool
+ Name="VCMIDLTool"
+ />
+ <Tool
+ Name="VCCLCompilerTool"
+ Optimization="0"
+ AdditionalIncludeDirectories=".."
+ PreprocessorDefinitions="WIN32;_DEBUG;_CONSOLE"
+ MinimalRebuild="true"
+ BasicRuntimeChecks="3"
+ RuntimeLibrary="1"
+ UsePrecompiledHeader="0"
+ WarningLevel="3"
+ Detect64BitPortabilityProblems="false"
+ DebugInformationFormat="4"
+ />
+ <Tool
+ Name="VCManagedResourceCompilerTool"
+ />
+ <Tool
+ Name="VCResourceCompilerTool"
+ />
+ <Tool
+ Name="VCPreLinkEventTool"
+ />
+ <Tool
+ Name="VCLinkerTool"
+ LinkIncremental="2"
+ GenerateDebugInformation="true"
+ SubSystem="1"
+ TargetMachine="1"
+ />
+ <Tool
+ Name="VCALinkTool"
+ />
+ <Tool
+ Name="VCManifestTool"
+ />
+ <Tool
+ Name="VCXDCMakeTool"
+ />
+ <Tool
+ Name="VCBscMakeTool"
+ />
+ <Tool
+ Name="VCFxCopTool"
+ />
+ <Tool
+ Name="VCAppVerifierTool"
+ />
+ <Tool
+ Name="VCWebDeploymentTool"
+ />
+ <Tool
+ Name="VCPostBuildEventTool"
+ />
+ </Configuration>
+ <Configuration
+ Name="Release|Win32"
+ OutputDirectory="..\..\chrome\$(ConfigurationName)"
+ IntermediateDirectory="$(ConfigurationName)"
+ ConfigurationType="1"
+ InheritedPropertySheets=".\chrometab_perftests.vsprops;$(SolutionDir)..\build\release.vsprops"
+ WholeProgramOptimization="1"
+ >
+ <Tool
+ Name="VCPreBuildEventTool"
+ />
+ <Tool
+ Name="VCCustomBuildTool"
+ />
+ <Tool
+ Name="VCXMLDataGeneratorTool"
+ />
+ <Tool
+ Name="VCWebServiceProxyGeneratorTool"
+ />
+ <Tool
+ Name="VCMIDLTool"
+ />
+ <Tool
+ Name="VCCLCompilerTool"
+ AdditionalIncludeDirectories=".."
+ PreprocessorDefinitions="WIN32;NDEBUG;_CONSOLE"
+ RuntimeLibrary="0"
+ UsePrecompiledHeader="0"
+ WarningLevel="3"
+ Detect64BitPortabilityProblems="false"
+ DebugInformationFormat="3"
+ />
+ <Tool
+ Name="VCManagedResourceCompilerTool"
+ />
+ <Tool
+ Name="VCResourceCompilerTool"
+ />
+ <Tool
+ Name="VCPreLinkEventTool"
+ />
+ <Tool
+ Name="VCLinkerTool"
+ LinkIncremental="1"
+ GenerateDebugInformation="true"
+ SubSystem="1"
+ OptimizeReferences="2"
+ EnableCOMDATFolding="2"
+ TargetMachine="1"
+ />
+ <Tool
+ Name="VCALinkTool"
+ />
+ <Tool
+ Name="VCManifestTool"
+ />
+ <Tool
+ Name="VCXDCMakeTool"
+ />
+ <Tool
+ Name="VCBscMakeTool"
+ />
+ <Tool
+ Name="VCFxCopTool"
+ />
+ <Tool
+ Name="VCAppVerifierTool"
+ />
+ <Tool
+ Name="VCWebDeploymentTool"
+ />
+ <Tool
+ Name="VCPostBuildEventTool"
+ />
+ </Configuration>
+ </Configurations>
+ <References>
+ </References>
+ <Files>
+ <Filter
+ Name="Source Files"
+ Filter="cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx"
+ UniqueIdentifier="{4FC737F1-C7A5-4376-A066-2A32D752A2FF}"
+ >
+ <File
+ RelativePath=".\chrometab_perftest.cc"
+ >
+ </File>
+ <File
+ RelativePath=".\chrometab_perftest.h"
+ >
+ </File>
+ <File
+ RelativePath=".\silverlight.cc"
+ >
+ </File>
+ <Filter
+ Name="Common"
+ >
+ <File
+ RelativePath="..\..\chrome\test\chrome_process_util.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\..\chrome\test\chrome_process_util.h"
+ >
+ </File>
+ <File
+ RelativePath="..\..\chrome\test\chrome_process_util_win.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\html_utils.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\..\chrome\test\perf\mem_usage.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\..\base\perf_test_suite.h"
+ >
+ </File>
+ <File
+ RelativePath="..\..\base\perftimer.cc"
+ >
+ </File>
+ <File
+ RelativePath="run_all.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\test_utils.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\test_utils.h"
+ >
+ </File>
+ <File
+ RelativePath="..\..\chrome\test\ui\ui_test.cc"
+ >
+ </File>
+ <File
+ RelativePath="..\utils.cc"
+ >
+ </File>
+ </Filter>
+ </Filter>
+ </Files>
+ <Globals>
+ </Globals>
+</VisualStudioProject>
diff --git a/chrome_frame/test/perf/chrometab_perftests.vsprops b/chrome_frame/test/perf/chrometab_perftests.vsprops
new file mode 100644
index 0000000..3891381
--- /dev/null
+++ b/chrome_frame/test/perf/chrometab_perftests.vsprops
@@ -0,0 +1,14 @@
+<?xml version="1.0" encoding="Windows-1252"?>
+<VisualStudioPropertySheet
+ ProjectType="Visual C++"
+ Version="8.00"
+ Name="chrometeb_perftests"
+ InheritedPropertySheets="$(SolutionDir)..\build\common.vsprops;$(SolutionDir)..\skia\using_skia.vsprops;$(SolutionDir)..\third_party\libxml\build\using_libxml.vsprops;$(SolutionDir)..\third_party\libxslt\build\using_libxslt.vsprops;..\..\testing\using_gtest.vsprops;$(SolutionDir)..\chrome\third_party\wtl\using_wtl.vsprops;$(SolutionDir)..\tools\grit\build\using_generated_resources.vsprops"
+ >
+ <Tool
+ Name="VCCLCompilerTool"
+ PreprocessorDefinitions="_ATL_APARTMENT_THREADED;_ATL_CSTRING_EXPLICIT_CONSTRUCTORS"
+ AdditionalIncludeDirectories=""
+ />
+
+</VisualStudioPropertySheet>
diff --git a/chrome_frame/test/perf/run_all.cc b/chrome_frame/test/perf/run_all.cc
new file mode 100644
index 0000000..93a5a67
--- /dev/null
+++ b/chrome_frame/test/perf/run_all.cc
@@ -0,0 +1,17 @@
+// Copyright (c) 2006-2009 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 "base/platform_thread.h"
+#include "base/perf_test_suite.h"
+#include "base/scoped_ptr.h"
+#include "chrome/common/chrome_paths.h"
+#include "chrome_frame/test_utils.h"
+
+int main(int argc, char **argv) {
+ PerfTestSuite perf_suite(argc, argv);
+ chrome::RegisterPathProvider();
+ PlatformThread::SetName("ChromeFrame perf tests");
+ // Use ctor/raii to register the local Chrome Frame dll.
+ scoped_ptr<ScopedChromeFrameRegistrar> registrar(new ScopedChromeFrameRegistrar);
+ return perf_suite.Run();
+}
diff --git a/chrome_frame/test/perf/silverlight.cc b/chrome_frame/test/perf/silverlight.cc
new file mode 100644
index 0000000..3016c2b
--- /dev/null
+++ b/chrome_frame/test/perf/silverlight.cc
@@ -0,0 +1,165 @@
+// Copyright (c) 2009 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 <atlbase.h>
+#include <atlwin.h>
+#include <atlhost.h>
+#include "base/scoped_comptr_win.h"
+#include "chrome_frame/test/perf/chrome_frame_perftest.h"
+
+interface IXcpControlDownloadCallback;
+interface __declspec(uuid("1B36028E-B491-4bb2-8584-8A9E0A677D6E"))
+IXcpControlHost : public IUnknown {
+ typedef enum {
+ XcpHostOption_FreezeOnInitialFrame = 0x001,
+ XcpHostOption_DisableFullScreen = 0x002,
+ XcpHostOption_DisableManagedExecution = 0x008,
+ XcpHostOption_EnableCrossDomainDownloads = 0x010,
+ XcpHostOption_UseCustomAppDomain = 0x020,
+ XcpHostOption_DisableNetworking = 0x040,
+ XcpHostOption_DisableScriptCallouts = 0x080,
+ XcpHostOption_EnableHtmlDomAccess = 0x100,
+ XcpHostOption_EnableScriptableObjectAccess = 0x200,
+ } XcpHostOptions;
+
+ STDMETHOD(GetHostOptions)(DWORD* pdwOptions) PURE;
+ STDMETHOD(NotifyLoaded()) PURE;
+ STDMETHOD(NotifyError)(BSTR bstrError, BSTR bstrSource,
+ long nLine, long nColumn) PURE;
+ STDMETHOD(InvokeHandler)(BSTR bstrName, VARIANT varArg1, VARIANT varArg2,
+ VARIANT* pvarResult) PURE;
+ STDMETHOD(GetBaseUrl)(BSTR* pbstrUrl) PURE;
+ STDMETHOD(GetNamedSource)(BSTR bstrSourceName, BSTR* pbstrSource) PURE;
+ STDMETHOD(DownloadUrl)(BSTR bstrUrl, IXcpControlDownloadCallback* pCallback,
+ IStream** ppStream) PURE;
+};
+
+// Not templatized, to trade execution speed vs typing
+class IXcpControlHostImpl : public IXcpControlHost {
+ public:
+ STDMETHOD(GetHostOptions)(DWORD* pdwOptions) {
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(NotifyLoaded()) {
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(NotifyError)(BSTR bstrError, BSTR bstrSource,
+ long nLine, long nColumn) {
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(InvokeHandler)(BSTR bstrName, VARIANT varArg1, VARIANT varArg2,
+ VARIANT* pvarResult) {
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(GetBaseUrl)(BSTR* pbstrUrl) {
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(GetNamedSource)(BSTR bstrSourceName, BSTR* pbstrSource) {
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(DownloadUrl)(BSTR bstrUrl, IXcpControlDownloadCallback* pCallback,
+ IStream** ppStream) {
+ return E_NOTIMPL;
+ }
+};
+
+// Silverlight container. Supports do-nothing implementation of IXcpControlHost.
+// Should be extended to do some real movie-or-something download.
+class SilverlightContainer :
+ public IServiceProviderImpl<SilverlightContainer>,
+ public IXcpControlHostImpl,
+ public CWindowImpl<SilverlightContainer, CWindow, CWinTraits<
+ WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
+ WS_EX_APPWINDOW | WS_EX_WINDOWEDGE> >,
+ public CComObjectRootEx<CComSingleThreadModel> {
+ public:
+ DECLARE_WND_CLASS_EX(L"Silverlight_container", 0, 0)
+ BEGIN_COM_MAP(SilverlightContainer)
+ COM_INTERFACE_ENTRY(IServiceProvider)
+ COM_INTERFACE_ENTRY(IXcpControlHost)
+ END_COM_MAP()
+
+ BEGIN_SERVICE_MAP(SilverlightContainer)
+ SERVICE_ENTRY(__uuidof(IXcpControlHost))
+ END_SERVICE_MAP()
+
+ BEGIN_MSG_MAP(ChromeFrameActiveXContainer)
+ MESSAGE_HANDLER(WM_DESTROY, OnDestroy)
+ END_MSG_MAP()
+
+ LRESULT OnDestroy(UINT, WPARAM, LPARAM, BOOL& handled) {
+ host_.Release();
+ return 0;
+ }
+
+ virtual void OnFinalMessage(HWND ) {
+ }
+
+ static const wchar_t* GetWndCaption() {
+ return L"Silverlight Container";
+ }
+
+ HRESULT CreateWndAndHost(RECT* r) {
+ Create(NULL, r);
+ ShowWindow(SW_SHOWDEFAULT);
+
+ CComPtr<IUnknown> spUnkContainer;
+ HRESULT hr = CAxHostWindow::_CreatorClass::CreateInstance(NULL,
+ __uuidof(IAxWinHostWindow), reinterpret_cast<void**>(&host_));
+ if (SUCCEEDED(hr)) {
+ CComPtr<IObjectWithSite> p;
+ hr = host_.QueryInterface(&p);
+ if (SUCCEEDED(hr)) {
+ p->SetSite(GetUnknown());
+ }
+ }
+ return hr;
+ }
+
+ HRESULT CreateControl() {
+ HRESULT hr = host_->CreateControl(L"AgControl.AgControl", m_hWnd, NULL);
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ return hr;
+ }
+
+ ScopedComPtr<IAxWinHostWindow> host_;
+};
+
+// Create and in-place Silverlight control. Should be extended to do something
+// more meaningful.
+TEST(ChromeFramePerf, DISABLED_HostSilverlight2) {
+ SimpleModule module;
+ AtlAxWinInit();
+ CComObjectStackEx<SilverlightContainer> wnd;
+ RECT rc = {0, 0, 800, 600};
+ wnd.CreateWndAndHost(&rc);
+ PerfTimeLogger perf_create("Create Silverlight Control2");
+ wnd.CreateControl();
+ perf_create.Done();
+ wnd.DestroyWindow();
+}
+
+// Simplest test - creates in-place Silverlight control.
+TEST(ChromeFramePerf, DISABLED_HostSilverlight) {
+ SimpleModule module;
+ AtlAxWinInit();
+ CAxWindow host;
+ RECT rc = {0, 0, 800, 600};
+ PerfTimeLogger perf_create("Create Silverlight Control");
+ host.Create(NULL, rc, L"AgControl.AgControl",
+ WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN | WS_CLIPSIBLINGS,
+ WS_EX_APPWINDOW | WS_EX_WINDOWEDGE);
+ EXPECT_TRUE(host.m_hWnd != NULL);
+ ScopedComPtr<IDispatch> disp;
+ HRESULT hr = host.QueryControl(disp.Receive());
+ EXPECT_HRESULT_SUCCEEDED(hr);
+ disp.Release();
+ perf_create.Done();
+}
+
diff --git a/chrome_frame/test/run_all_unittests.cc b/chrome_frame/test/run_all_unittests.cc
new file mode 100644
index 0000000..787d765
--- /dev/null
+++ b/chrome_frame/test/run_all_unittests.cc
@@ -0,0 +1,49 @@
+// Copyright (c) 2006-2008 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 <atlbase.h>
+
+#include "base/at_exit.h"
+#include "base/platform_thread.h"
+#include "base/process_util.h"
+#include "base/test_suite.h"
+#include "base/command_line.h"
+#include "chrome/common/chrome_paths.h"
+
+#include "chrome_frame/test_utils.h"
+
+// To enable ATL-based code to run in this module
+class ChromeFrameUnittestsModule
+ : public CAtlExeModuleT<ChromeFrameUnittestsModule> {
+};
+
+ChromeFrameUnittestsModule _AtlModule;
+
+const wchar_t kNoRegistrationSwitch[] = L"no-registration";
+
+int main(int argc, char **argv) {
+ base::EnableTerminationOnHeapCorruption();
+ PlatformThread::SetName("ChromeFrame tests");
+
+ TestSuite test_suite(argc, argv);
+
+ // If mini_installer is used to register CF, we use the switch
+ // --no-registration to avoid repetitive registration.
+ if (CommandLine::ForCurrentProcess()->HasSwitch(kNoRegistrationSwitch)) {
+ return test_suite.Run();
+ } else {
+ // Register paths needed by the ScopedChromeFrameRegistrar.
+ chrome::RegisterPathProvider();
+
+ // This will register the chrome frame in the build directory. It currently
+ // leaves that chrome frame registered once the tests are done. It must be
+ // constructed AFTER the TestSuite is created since TestSuites create THE
+ // AtExitManager.
+ // TODO(robertshield): Make these tests restore the original registration
+ // once done.
+ ScopedChromeFrameRegistrar registrar;
+
+ return test_suite.Run();
+ }
+}
diff --git a/chrome_frame/test/test_server.cc b/chrome_frame/test/test_server.cc
new file mode 100644
index 0000000..79ea2cf
--- /dev/null
+++ b/chrome_frame/test/test_server.cc
@@ -0,0 +1,211 @@
+// Copyright (c) 2006-2008 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 "base/logging.h"
+#include "base/registry.h"
+#include "base/string_util.h"
+
+#include "chrome_frame/test/test_server.h"
+
+#include "net/base/winsock_init.h"
+#include "net/http/http_util.h"
+
+namespace test_server {
+const char kDefaultHeaderTemplate[] =
+ "HTTP/1.1 %hs\r\n"
+ "Connection: close\r\n"
+ "Content-Type: %hs\r\n"
+ "Content-Length: %i\r\n\r\n";
+const char kStatusOk[] = "200 OK";
+const char kStatusNotFound[] = "404 Not Found";
+const char kDefaultContentType[] = "text/html; charset=UTF-8";
+
+void Request::ParseHeaders(const std::string& headers) {
+ size_t pos = headers.find("\r\n");
+ DCHECK(pos != std::string::npos);
+ if (pos != std::string::npos) {
+ headers_ = headers.substr(pos + 2);
+
+ StringTokenizer tokenizer(headers.begin(), headers.begin() + pos, " ");
+ std::string* parse[] = { &method_, &path_, &version_ };
+ int field = 0;
+ while (tokenizer.GetNext() && field < arraysize(parse)) {
+ parse[field++]->assign(tokenizer.token_begin(),
+ tokenizer.token_end());
+ }
+ }
+
+ // Check for content-length in case we're being sent some data.
+ net::HttpUtil::HeadersIterator it(headers_.begin(), headers_.end(),
+ "\r\n");
+ while (it.GetNext()) {
+ if (LowerCaseEqualsASCII(it.name(), "content-length")) {
+ content_length_ = StringToInt(it.values().c_str());
+ break;
+ }
+ }
+}
+
+bool Connection::CheckRequestReceived() {
+ bool ready = false;
+ if (request_.method().length()) {
+ // Headers have already been parsed. Just check content length.
+ ready = (data_.size() >= request_.content_length());
+ } else {
+ size_t index = data_.find("\r\n\r\n");
+ if (index != std::string::npos) {
+ // Parse the headers before returning and chop them of the
+ // data buffer we've already received.
+ std::string headers(data_.substr(0, index + 2));
+ request_.ParseHeaders(headers);
+ data_.erase(0, index + 4);
+ ready = (data_.size() >= request_.content_length());
+ }
+ }
+
+ return ready;
+}
+
+bool FileResponse::GetContentType(std::string* content_type) const {
+ size_t length = ContentLength();
+ char buffer[4096];
+ void* data = NULL;
+
+ if (length) {
+ // Create a copy of the first few bytes of the file.
+ // If we try and use the mapped file directly, FindMimeFromData will crash
+ // 'cause it cheats and temporarily tries to write to the buffer!
+ length = std::min(arraysize(buffer), length);
+ memcpy(buffer, file_->data(), length);
+ data = buffer;
+ }
+
+ LPOLESTR mime_type = NULL;
+ FindMimeFromData(NULL, file_path_.value().c_str(), data, length, NULL,
+ FMFD_DEFAULT, &mime_type, 0);
+ if (mime_type) {
+ *content_type = WideToASCII(mime_type);
+ ::CoTaskMemFree(mime_type);
+ }
+
+ return content_type->length() > 0;
+}
+
+void FileResponse::WriteContents(ListenSocket* socket) const {
+ DCHECK(file_.get());
+ if (file_.get()) {
+ socket->Send(reinterpret_cast<const char*>(file_->data()),
+ file_->length(), false);
+ }
+}
+
+size_t FileResponse::ContentLength() const {
+ if (file_.get() == NULL) {
+ file_.reset(new file_util::MemoryMappedFile());
+ if (!file_->Initialize(file_path_)) {
+ NOTREACHED();
+ file_.reset();
+ }
+ }
+ return file_.get() ? file_->length() : 0;
+}
+
+bool RedirectResponse::GetCustomHeaders(std::string* headers) const {
+ *headers = StringPrintf("HTTP/1.1 302 Found\r\n"
+ "Connection: close\r\n"
+ "Content-Length: 0\r\n"
+ "Content-Type: text/html\r\n"
+ "Location: %hs\r\n\r\n", redirect_url_.c_str());
+ return true;
+}
+
+SimpleWebServer::SimpleWebServer(int port) {
+ CHECK(MessageLoop::current()) << "SimpleWebServer requires a message loop";
+ net::EnsureWinsockInit();
+ AddResponse(&quit_);
+ server_ = ListenSocket::Listen("127.0.0.1", port, this);
+ DCHECK(server_.get() != NULL);
+}
+
+SimpleWebServer::~SimpleWebServer() {
+ ConnectionList::const_iterator it;
+ for (it = connections_.begin(); it != connections_.end(); it++)
+ delete (*it);
+ connections_.clear();
+}
+
+void SimpleWebServer::AddResponse(Response* response) {
+ responses_.push_back(response);
+}
+
+Response* SimpleWebServer::FindResponse(const Request& request) const {
+ std::list<Response*>::const_iterator it;
+ for (it = responses_.begin(); it != responses_.end(); it++) {
+ Response* response = (*it);
+ if (response->Matches(request)) {
+ return response;
+ }
+ }
+ return NULL;
+}
+
+Connection* SimpleWebServer::FindConnection(const ListenSocket* socket) const {
+ ConnectionList::const_iterator it;
+ for (it = connections_.begin(); it != connections_.end(); it++) {
+ if ((*it)->IsSame(socket)) {
+ return (*it);
+ }
+ }
+ return NULL;
+}
+
+void SimpleWebServer::DidAccept(ListenSocket* server,
+ ListenSocket* connection) {
+ connections_.push_back(new Connection(connection));
+}
+
+void SimpleWebServer::DidRead(ListenSocket* connection,
+ const std::string& data) {
+ Connection* c = FindConnection(connection);
+ DCHECK(c);
+ c->AddData(data);
+ if (c->CheckRequestReceived()) {
+ const Request& request = c->request();
+ Response* response = FindResponse(request);
+ if (response) {
+ std::string headers;
+ if (!response->GetCustomHeaders(&headers)) {
+ std::string content_type;
+ if (!response->GetContentType(&content_type))
+ content_type = kDefaultContentType;
+ headers = StringPrintf(kDefaultHeaderTemplate, kStatusOk,
+ content_type.c_str(), response->ContentLength());
+ }
+
+ connection->Send(headers, false);
+ response->WriteContents(connection);
+ response->IncrementAccessCounter();
+ } else {
+ std::string payload = "sorry, I can't find " + request.path();
+ std::string headers(StringPrintf(kDefaultHeaderTemplate, kStatusNotFound,
+ kDefaultContentType, payload.length()));
+ connection->Send(headers, false);
+ connection->Send(payload, false);
+ }
+ }
+}
+
+void SimpleWebServer::DidClose(ListenSocket* sock) {
+ // To keep the historical list of connections reasonably tidy, we delete
+ // 404's when the connection ends.
+ Connection* c = FindConnection(sock);
+ DCHECK(c);
+ if (!FindResponse(c->request())) {
+ // extremely inefficient, but in one line and not that common... :)
+ connections_.erase(std::find(connections_.begin(), connections_.end(), c));
+ delete c;
+ }
+}
+
+} // namespace test_server
diff --git a/chrome_frame/test/test_server.h b/chrome_frame/test/test_server.h
new file mode 100644
index 0000000..896b2b3
--- /dev/null
+++ b/chrome_frame/test/test_server.h
@@ -0,0 +1,296 @@
+// Copyright (c) 2006-2008 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.
+
+#ifndef CHROME_FRAME_TEST_TEST_SERVER_H_
+#define CHROME_FRAME_TEST_TEST_SERVER_H_
+
+// Implementation of an HTTP server for tests.
+// To instantiate the server, make sure you have a message loop on the
+// current thread and then create an instance of the SimpleWebServer class.
+// The server uses two basic concepts, a request and a response.
+// The Response interface represents an item (e.g. a document) available from
+// the server. A Request object represents a request from a client (e.g. a
+// browser). There are several basic Response classes implemented in this file,
+// all derived from the Response interface.
+//
+// Here's a simple example that starts a web server that can serve up
+// a single document (http://localhost:1337/foo).
+// All other requests will get a 404.
+//
+// MessageLoopForUI loop;
+// test_server::SimpleWebServer server(1337);
+// test_server::SimpleResponse document("/foo", "Hello World!");
+// test_server.AddResponse(&document);
+// loop.MessageLoop::Run();
+//
+// To close the web server, just go to http://localhost:1337/quit.
+//
+// All Response classes count how many times they have been accessed. Just
+// call Response::accessed().
+//
+// To implement a custom response object (e.g. to match against a request
+// based on some data, serve up dynamic content or take some action on the
+// server), just inherit from one of the response classes or directly from the
+// Response interface and add your response object to the server's list of
+// response objects.
+
+#include <list>
+#include <string>
+
+#include "base/basictypes.h"
+#include "base/file_util.h"
+#include "net/base/listen_socket.h"
+
+namespace test_server {
+
+class Request {
+ public:
+ Request() : content_length_(0) {
+ }
+
+ void ParseHeaders(const std::string& headers);
+
+ const std::string& method() const {
+ return method_;
+ }
+
+ const std::string& path() const {
+ return path_;
+ }
+
+ const std::string& headers() const {
+ return headers_;
+ }
+
+ size_t content_length() const {
+ return content_length_;
+ }
+
+ protected:
+ std::string method_;
+ std::string path_;
+ std::string version_;
+ std::string headers_;
+ size_t content_length_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(Request);
+};
+
+// Manages request headers for a single request.
+// For each successful request that's made, the server will keep an instance
+// of this class so that they can be checked even after the server has been
+// shut down.
+class Connection {
+ public:
+ explicit Connection(ListenSocket* sock) : socket_(sock) {
+ }
+
+ ~Connection() {
+ }
+
+ bool IsSame(const ListenSocket* socket) const {
+ return socket_ == socket;
+ }
+
+ void AddData(const std::string& data) {
+ data_ += data;
+ }
+
+ bool CheckRequestReceived();
+
+ const Request& request() const {
+ return request_;
+ }
+
+ protected:
+ scoped_refptr<ListenSocket> socket_;
+ std::string data_;
+ Request request_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(Connection);
+};
+
+// Abstract interface with default implementations for some of the methods and
+// a counter for how many times the response object has served requests.
+class Response {
+ public:
+ Response() : accessed_(0) {
+ }
+
+ virtual ~Response() {
+ }
+
+ // Returns true if this response object should be used for a given request.
+ virtual bool Matches(const Request& r) const = 0;
+
+ // Response objects can optionally supply their own HTTP headers, completely
+ // bypassing the default ones.
+ virtual bool GetCustomHeaders(std::string* headers) const {
+ return false;
+ }
+
+ // Optionally provide a content type. Return false if you don't specify
+ // a content type.
+ virtual bool GetContentType(std::string* content_type) const {
+ return false;
+ }
+
+ virtual size_t ContentLength() const {
+ return 0;
+ }
+
+ virtual void WriteContents(ListenSocket* socket) const {
+ }
+
+ void IncrementAccessCounter() {
+ accessed_++;
+ }
+
+ size_t accessed() const {
+ return accessed_;
+ }
+
+ protected:
+ size_t accessed_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(Response);
+};
+
+// Partial implementation of Response that matches a request's path.
+// This is just a convenience implementation for the boilerplate implementation
+// of Matches(). Don't instantiate directly.
+class ResponseForPath : public Response {
+ public:
+ explicit ResponseForPath(const char* request_path)
+ : request_path_(request_path) {
+ }
+
+ virtual bool Matches(const Request& r) const {
+ return r.path().compare(request_path_) == 0;
+ }
+
+ protected:
+ std::string request_path_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(ResponseForPath);
+};
+
+// A very basic implementation of a response.
+// A simple response matches a single document path on the server
+// (e.g. "/foo") and returns a document in the form of a string.
+class SimpleResponse : public ResponseForPath {
+ public:
+ SimpleResponse(const char* request_path, const std::string& contents)
+ : ResponseForPath(request_path), contents_(contents) {
+ }
+
+ virtual void WriteContents(ListenSocket* socket) const {
+ socket->Send(contents_.c_str(), contents_.length(), false);
+ }
+
+ virtual size_t ContentLength() const {
+ return contents_.length();
+ }
+
+ protected:
+ std::string contents_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(SimpleResponse);
+};
+
+// To serve up files from the web server, create an instance of FileResponse
+// and add it to the server's list of responses. The content type of the
+// file will be determined by calling FindMimeFromData which examines the
+// contents of the file and performs registry lookups.
+class FileResponse : public ResponseForPath {
+ public:
+ FileResponse(const char* request_path, const FilePath& file_path)
+ : ResponseForPath(request_path), file_path_(file_path) {
+ }
+
+ virtual bool GetContentType(std::string* content_type) const;
+ virtual void WriteContents(ListenSocket* socket) const;
+ virtual size_t ContentLength() const;
+
+ protected:
+ FilePath file_path_;
+ mutable scoped_ptr<file_util::MemoryMappedFile> file_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(FileResponse);
+};
+
+// Returns a 302 (temporary redirect) to redirect the client from a path
+// on the test server to a different URL.
+class RedirectResponse : public ResponseForPath {
+ public:
+ RedirectResponse(const char* request_path, const std::string& redirect_url)
+ : ResponseForPath(request_path), redirect_url_(redirect_url) {
+ }
+
+ virtual bool GetCustomHeaders(std::string* headers) const;
+
+ protected:
+ std::string redirect_url_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(RedirectResponse);
+};
+
+// typedef for a list of connections. Used by SimpleWebServer.
+typedef std::list<Connection*> ConnectionList;
+
+// Implementation of a simple http server.
+// Before creating an instance of the server, make sure the current thread
+// has a message loop.
+class SimpleWebServer : public ListenSocket::ListenSocketDelegate {
+ public:
+ explicit SimpleWebServer(int port);
+ virtual ~SimpleWebServer();
+
+ void AddResponse(Response* response);
+
+ // ListenSocketDelegate overrides.
+ virtual void DidAccept(ListenSocket* server, ListenSocket* connection);
+ virtual void DidRead(ListenSocket* connection, const std::string& data);
+ virtual void DidClose(ListenSocket* sock);
+
+ const ConnectionList& connections() {
+ return connections_;
+ }
+
+ protected:
+ class QuitResponse : public SimpleResponse {
+ public:
+ QuitResponse()
+ : SimpleResponse("/quit", "So long and thanks for all the fish.") {
+ }
+
+ virtual void QuitResponse::WriteContents(ListenSocket* socket) const {
+ SimpleResponse::WriteContents(socket);
+ MessageLoop::current()->Quit();
+ }
+ };
+
+ Response* FindResponse(const Request& request) const;
+ Connection* FindConnection(const ListenSocket* socket) const;
+
+ protected:
+ scoped_refptr<ListenSocket> server_;
+ ConnectionList connections_;
+ std::list<Response*> responses_;
+ QuitResponse quit_;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(SimpleWebServer);
+};
+
+} // namespace test_server
+
+#endif // CHROME_FRAME_TEST_TEST_SERVER_H_
diff --git a/chrome_frame/test/test_server_test.cc b/chrome_frame/test/test_server_test.cc
new file mode 100644
index 0000000..4bd139e
--- /dev/null
+++ b/chrome_frame/test/test_server_test.cc
@@ -0,0 +1,194 @@
+// Copyright (c) 2006-2008 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 <windows.h>
+#include <wininet.h>
+
+#include "base/basictypes.h"
+#include "base/path_service.h"
+#include "base/scoped_handle_win.h"
+#include "chrome_frame/test/test_server.h"
+#include "net/base/cookie_monster.h"
+#include "net/base/host_resolver_proc.h"
+#include "net/disk_cache/disk_cache.h"
+#include "net/http/http_cache.h"
+#include "net/proxy/proxy_service.h"
+#include "net/url_request/url_request.h"
+#include "net/url_request/url_request_unittest.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+class TestServerTest: public testing::Test {
+ protected:
+ virtual void SetUp() {
+ PathService::Get(base::DIR_SOURCE_ROOT, &source_path_);
+ source_path_ = source_path_.Append(FILE_PATH_LITERAL("chrome_frame"));
+ }
+ virtual void TearDown() {
+ }
+
+ public:
+ const FilePath& source_path() const {
+ return source_path_;
+ }
+
+ protected:
+ FilePath source_path_;
+};
+
+namespace {
+
+class ScopedInternet {
+ public:
+ explicit ScopedInternet(HINTERNET handle)
+ : h_(handle) {
+ }
+ ~ScopedInternet() {
+ if (h_) {
+ InternetCloseHandle(h_);
+ }
+ }
+
+ operator HINTERNET() {
+ return h_;
+ }
+
+ protected:
+ HINTERNET h_;
+};
+
+class URLRequestTestContext : public URLRequestContext {
+ public:
+ URLRequestTestContext() {
+ host_resolver_ = net::CreateSystemHostResolver();
+ proxy_service_ = net::ProxyService::CreateNull();
+ ssl_config_service_ = new net::SSLConfigServiceDefaults;
+ http_transaction_factory_ =
+ new net::HttpCache(
+ net::HttpNetworkLayer::CreateFactory(host_resolver_, proxy_service_,
+ ssl_config_service_),
+ disk_cache::CreateInMemoryCacheBackend(0));
+ // In-memory cookie store.
+ cookie_store_ = new net::CookieMonster();
+ }
+
+ virtual ~URLRequestTestContext() {
+ delete http_transaction_factory_;
+ }
+};
+
+class TestURLRequest : public URLRequest {
+ public:
+ TestURLRequest(const GURL& url, Delegate* delegate)
+ : URLRequest(url, delegate) {
+ set_context(new URLRequestTestContext());
+ }
+};
+
+class UrlTaskChain {
+ public:
+ UrlTaskChain(const char* url, UrlTaskChain* next)
+ : url_(url), next_(next) {
+ }
+
+ void Run() {
+ EXPECT_EQ(0, delegate_.response_started_count());
+
+ MessageLoopForIO loop;
+
+ TestURLRequest r(GURL(url_), &delegate_);
+ r.Start();
+ EXPECT_TRUE(r.is_pending());
+
+ MessageLoop::current()->Run();
+
+ EXPECT_EQ(1, delegate_.response_started_count());
+ EXPECT_FALSE(delegate_.received_data_before_response());
+ EXPECT_NE(0, delegate_.bytes_received());
+ }
+
+ UrlTaskChain* next() const {
+ return next_;
+ }
+
+ const std::string& response() const {
+ return delegate_.data_received();
+ }
+
+ protected:
+ std::string url_;
+ TestDelegate delegate_;
+ UrlTaskChain* next_;
+};
+
+DWORD WINAPI FetchUrl(void* param) {
+ UrlTaskChain* task = reinterpret_cast<UrlTaskChain*>(param);
+ while (task != NULL) {
+ task->Run();
+ task = task->next();
+ }
+
+ return 0;
+}
+
+struct QuitMessageHit {
+ explicit QuitMessageHit(MessageLoopForUI* loop) : loop_(loop), hit_(false) {
+ }
+
+ MessageLoopForUI* loop_;
+ bool hit_;
+};
+
+void QuitMessageLoop(QuitMessageHit* msg) {
+ msg->hit_ = true;
+ msg->loop_->PostTask(FROM_HERE, new MessageLoop::QuitTask);
+}
+
+} // end namespace
+
+TEST_F(TestServerTest, DISABLED_TestServer) {
+ // The web server needs a loop to exist on this thread during construction
+ // the loop must be created before we construct the server.
+ MessageLoopForUI loop;
+
+ test_server::SimpleWebServer server(1337);
+ test_server::SimpleResponse person("/person", "Guthrie Govan!");
+ server.AddResponse(&person);
+ test_server::FileResponse file("/file", source_path().Append(
+ FILE_PATH_LITERAL("CFInstance.js")));
+ server.AddResponse(&file);
+ test_server::RedirectResponse redir("/goog", "http://www.google.com/");
+ server.AddResponse(&redir);
+
+ // We should never hit this, but it's our way to break out of the test if
+ // things start hanging.
+ QuitMessageHit quit_msg(&loop);
+ loop.PostDelayedTask(FROM_HERE,
+ NewRunnableFunction(QuitMessageLoop, &quit_msg),
+ 10 * 1000);
+
+ UrlTaskChain quit_task("http://localhost:1337/quit", NULL);
+ UrlTaskChain fnf_task("http://localhost:1337/404", &quit_task);
+ UrlTaskChain person_task("http://localhost:1337/person", &fnf_task);
+ UrlTaskChain file_task("http://localhost:1337/file", &person_task);
+ UrlTaskChain goog_task("http://localhost:1337/goog", &file_task);
+
+ DWORD tid = 0;
+ ScopedHandle worker(::CreateThread(NULL, 0, FetchUrl, &goog_task, 0, &tid));
+ loop.MessageLoop::Run();
+
+ EXPECT_FALSE(quit_msg.hit_);
+ if (!quit_msg.hit_) {
+ EXPECT_EQ(::WaitForSingleObject(worker, 10 * 1000), WAIT_OBJECT_0);
+
+ EXPECT_EQ(person.accessed(), 1);
+ EXPECT_EQ(file.accessed(), 1);
+ EXPECT_EQ(redir.accessed(), 1);
+
+ EXPECT_TRUE(person_task.response().find("Guthrie") != std::string::npos);
+ EXPECT_TRUE(file_task.response().find("function") != std::string::npos);
+ EXPECT_TRUE(goog_task.response().find("<title>") != std::string::npos);
+ } else {
+ ::TerminateThread(worker, ~0);
+ }
+}
diff --git a/chrome_frame/test/util_unittests.cc b/chrome_frame/test/util_unittests.cc
new file mode 100644
index 0000000..dc3d72d
--- /dev/null
+++ b/chrome_frame/test/util_unittests.cc
@@ -0,0 +1,120 @@
+// Copyright (c) 2006-2009 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 "base/file_version_info.h"
+#include "chrome_frame/test/chrome_frame_unittests.h"
+#include "chrome_frame/utils.h"
+
+const wchar_t kChannelName[] = L"-dev";
+const wchar_t kSuffix[] = L"-fix";
+
+TEST(UtilTests, AppendSuffixToChannelNameTest) {
+ std::wstring str_base;
+ std::wstring channel_name(kChannelName);
+ std::wstring suffix(kSuffix);
+
+ str_base = L"2.0-dev-bar";
+ EXPECT_TRUE(AppendSuffixToChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev-fix-bar", str_base.c_str());
+
+ str_base = L"2.0-dev-fix-bar";
+ EXPECT_FALSE(AppendSuffixToChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev-fix-bar", str_base.c_str());
+
+ str_base = L"2.0-dev-bar-dev-bar";
+ EXPECT_TRUE(AppendSuffixToChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev-fix-bar-dev-bar", str_base.c_str());
+
+ str_base = L"2.0";
+ EXPECT_FALSE(AppendSuffixToChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0", str_base.c_str());
+
+ str_base = L"2.0-devvvv";
+ EXPECT_TRUE(AppendSuffixToChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev-fixvvv", str_base.c_str());
+}
+
+TEST(UtilTests, RemoveSuffixFromStringTest) {
+ std::wstring str_base;
+ std::wstring channel_name(kChannelName);
+ std::wstring suffix(kSuffix);
+
+ str_base = L"2.0-dev-fix";
+ EXPECT_TRUE(RemoveSuffixFromChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev", str_base.c_str());
+
+ str_base = L"2.0-dev-fix-full";
+ EXPECT_TRUE(RemoveSuffixFromChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev-full", str_base.c_str());
+
+ str_base = L"2.0";
+ EXPECT_FALSE(RemoveSuffixFromChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0", str_base.c_str());
+
+ str_base = L"2.0-dev";
+ EXPECT_FALSE(RemoveSuffixFromChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev", str_base.c_str());
+
+ str_base = L"2.0-fix";
+ EXPECT_FALSE(RemoveSuffixFromChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-fix", str_base.c_str());
+
+ str_base = L"2.0-full-fix";
+ EXPECT_FALSE(RemoveSuffixFromChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-full-fix", str_base.c_str());
+
+ str_base = L"2.0-dev-dev-fix";
+ EXPECT_TRUE(RemoveSuffixFromChannelName(&str_base, channel_name, suffix));
+ EXPECT_STREQ(L"2.0-dev-dev", str_base.c_str());
+}
+
+TEST(UtilTests, GetModuleVersionTest) {
+ HMODULE mod = GetModuleHandle(L"kernel32.dll");
+ EXPECT_NE(mod, static_cast<HMODULE>(NULL));
+ wchar_t path[MAX_PATH] = {0};
+ GetModuleFileName(mod, path, arraysize(path));
+
+ // Use the method that goes to disk
+ scoped_ptr<FileVersionInfo> base_info(
+ FileVersionInfo::CreateFileVersionInfo(path));
+ EXPECT_TRUE(base_info.get() != NULL);
+
+ // Use the method that doesn't go to disk
+ uint32 low = 0, high = 0;
+ EXPECT_TRUE(GetModuleVersion(mod, &high, &low));
+ EXPECT_NE(high, 0);
+ EXPECT_NE(low, 0);
+
+ // Make sure they give the same results.
+ VS_FIXEDFILEINFO* fixed_info = base_info->fixed_file_info();
+ EXPECT_TRUE(fixed_info != NULL);
+
+ EXPECT_EQ(fixed_info->dwFileVersionMS, static_cast<DWORD>(high));
+ EXPECT_EQ(fixed_info->dwFileVersionLS, static_cast<DWORD>(low));
+}
+
+TEST(UtilTests, HaveSameOrigin) {
+ struct OriginCompare {
+ const char* a;
+ const char* b;
+ bool same_origin;
+ } test_cases[] = {
+ { "", "", true },
+ { "*", "*", true },
+ { "*", "+", false },
+ { "http://www.google.com/", "http://www.google.com/", true },
+ { "http://www.google.com", "http://www.google.com/", true },
+ { "http://www.google.com:80/", "http://www.google.com/", true },
+ { "http://www.google.com:8080/", "http://www.google.com/", false },
+ { "https://www.google.com/", "http://www.google.com/", false },
+ { "http://docs.google.com/", "http://www.google.com/", false },
+ { "https://www.google.com/", "https://www.google.com:443/", true },
+ { "https://www.google.com/", "https://www.google.com:443", true },
+ };
+
+ for (int i = 0; i < arraysize(test_cases); ++i) {
+ const OriginCompare& test = test_cases[i];
+ EXPECT_EQ(test.same_origin, HaveSameOrigin(test.a, test.b));
+ }
+}
diff --git a/chrome_frame/test_utils.cc b/chrome_frame/test_utils.cc
new file mode 100644
index 0000000..2d47c71
--- /dev/null
+++ b/chrome_frame/test_utils.cc
@@ -0,0 +1,95 @@
+// Copyright (c) 2009 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 "chrome_frame/test_utils.h"
+
+#include <atlbase.h>
+#include <atlwin.h>
+
+#include "base/file_path.h"
+#include "base/file_util.h"
+#include "base/path_service.h"
+#include "chrome/common/chrome_paths.h"
+#include "testing/gtest/include/gtest/gtest.h"
+
+// Statics
+
+std::wstring ScopedChromeFrameRegistrar::GetChromeFrameBuildPath() {
+ std::wstring build_path;
+ PathService::Get(chrome::DIR_APP, &build_path);
+ file_util::AppendToPath(&build_path, L"servers\\npchrome_tab.dll");
+ file_util::PathExists(build_path);
+ return build_path;
+}
+
+void ScopedChromeFrameRegistrar::RegisterDefaults() {
+ std::wstring dll_path_ = GetChromeFrameBuildPath();
+ RegisterAtPath(dll_path_);
+}
+
+void ScopedChromeFrameRegistrar::RegisterAtPath(
+ const std::wstring& path) {
+
+ ASSERT_FALSE(path.empty());
+ HMODULE chrome_frame_dll_handle = LoadLibrary(path.c_str());
+ ASSERT_TRUE(chrome_frame_dll_handle != NULL);
+
+ typedef HRESULT (STDAPICALLTYPE* DllRegisterServerFn)();
+ DllRegisterServerFn register_server =
+ reinterpret_cast<DllRegisterServerFn>(GetProcAddress(
+ chrome_frame_dll_handle, "DllRegisterServer"));
+
+ ASSERT_TRUE(register_server != NULL);
+ EXPECT_HRESULT_SUCCEEDED((*register_server)());
+
+ DllRegisterServerFn register_npapi_server =
+ reinterpret_cast<DllRegisterServerFn>(GetProcAddress(
+ chrome_frame_dll_handle, "RegisterNPAPIPlugin"));
+
+ if (register_npapi_server != NULL)
+ EXPECT_HRESULT_SUCCEEDED((*register_npapi_server)());
+
+ ASSERT_TRUE(FreeLibrary(chrome_frame_dll_handle));
+}
+
+// Non-statics
+
+ScopedChromeFrameRegistrar::ScopedChromeFrameRegistrar() {
+ original_dll_path_ = GetChromeFrameBuildPath();
+ RegisterChromeFrameAtPath(original_dll_path_);
+}
+
+ScopedChromeFrameRegistrar::~ScopedChromeFrameRegistrar() {
+ if (FilePath(original_dll_path_) != FilePath(new_chrome_frame_dll_path_)) {
+ RegisterChromeFrameAtPath(original_dll_path_);
+ }
+}
+
+void ScopedChromeFrameRegistrar::RegisterChromeFrameAtPath(
+ const std::wstring& path) {
+ RegisterAtPath(path);
+ new_chrome_frame_dll_path_ = path;
+}
+
+void ScopedChromeFrameRegistrar::RegisterReferenceChromeFrameBuild() {
+ std::wstring reference_build_dir;
+ ASSERT_TRUE(PathService::Get(chrome::DIR_APP, &reference_build_dir));
+
+ file_util::UpOneDirectory(&reference_build_dir);
+ file_util::UpOneDirectory(&reference_build_dir);
+
+ file_util::AppendToPath(&reference_build_dir, L"chrome_frame");
+ file_util::AppendToPath(&reference_build_dir, L"tools");
+ file_util::AppendToPath(&reference_build_dir, L"test");
+ file_util::AppendToPath(&reference_build_dir, L"reference_build");
+ file_util::AppendToPath(&reference_build_dir, L"chrome");
+ file_util::AppendToPath(&reference_build_dir, L"servers");
+ file_util::AppendToPath(&reference_build_dir, L"npchrome_tab.dll");
+
+ RegisterChromeFrameAtPath(reference_build_dir);
+}
+
+std::wstring ScopedChromeFrameRegistrar::GetChromeFrameDllPath() const {
+ return new_chrome_frame_dll_path_;
+}
diff --git a/chrome_frame/test_utils.h b/chrome_frame/test_utils.h
new file mode 100644
index 0000000..fb470a0
--- /dev/null
+++ b/chrome_frame/test_utils.h
@@ -0,0 +1,39 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_TEST_UTILS_H_
+#define CHROME_FRAME_TEST_UTILS_H_
+
+#include <string>
+
+// Helper class used to register different chrome frame DLLs while running
+// tests. At construction, this registers the DLL found in the build path.
+// At destruction, again registers the DLL found in the build path if another
+// DLL has since been registered. Triggers GTEST asserts on failure.
+//
+// TODO(robertshield): Ideally, make this class restore the originally
+// registered chrome frame DLL (e.g. by looking in HKCR) on destruction.
+class ScopedChromeFrameRegistrar {
+ public:
+ ScopedChromeFrameRegistrar();
+ virtual ~ScopedChromeFrameRegistrar();
+
+ void RegisterChromeFrameAtPath(const std::wstring& path);
+ void RegisterReferenceChromeFrameBuild();
+
+ std::wstring GetChromeFrameDllPath() const;
+
+ static std::wstring GetChromeFrameBuildPath();
+ static void RegisterAtPath(const std::wstring& path);
+ static void RegisterDefaults();
+
+ private:
+ // Contains the path of the most recently registered npchrome_tab.dll.
+ std::wstring new_chrome_frame_dll_path_;
+
+ // Contains the path of the npchrome_tab.dll to be registered at destruction.
+ std::wstring original_dll_path_;
+};
+
+#endif // CHROME_FRAME_TEST_UTILS_H_
diff --git a/chrome_frame/tools/test/page_cycler/cf_cycler.py b/chrome_frame/tools/test/page_cycler/cf_cycler.py
new file mode 100644
index 0000000..fd49ae8
--- /dev/null
+++ b/chrome_frame/tools/test/page_cycler/cf_cycler.py
@@ -0,0 +1,99 @@
+# Copyright (c) 2009 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.
+
+"""Automates IE to visit a list of web sites while running CF in full tab mode.
+
+The page cycler automates IE and navigates it to a series of URLs. It is
+designed to be run with Chrome Frame configured to load every URL inside
+CF full tab mode.
+
+TODO(robertshield): Make use of the python unittest module as per
+review comments.
+"""
+
+import optparse
+import sys
+import time
+import win32com.client
+import win32gui
+
+def LoadSiteList(path):
+ """Loads a list of URLs from |path|.
+
+ Expects the URLs to be separated by newlines, with no leading or trailing
+ whitespace.
+
+ Args:
+ path: The path to a file containing a list of new-line separated URLs.
+
+ Returns:
+ A list of strings, each one a URL.
+ """
+ f = open(path)
+ urls = f.readlines()
+ f.close()
+ return urls
+
+def LaunchIE():
+ """Starts up IE, makes it visible and returns the automation object.
+
+ Returns:
+ The IE automation object.
+ """
+ ie = win32com.client.Dispatch("InternetExplorer.Application")
+ ie.visible = 1
+ win32gui.SetForegroundWindow(ie.HWND)
+ return ie
+
+def RunTest(url, ie):
+ """Loads |url| into the InternetExplorer.Application instance in |ie|.
+
+ Waits for the Document object to be created and then waits for
+ the document ready state to reach READYSTATE_COMPLETE.
+ Args:
+ url: A string containing the url to navigate to.
+ ie: The IE automation object to navigate.
+ """
+
+ print "Navigating to " + url
+ ie.Navigate(url)
+ timer = 0
+
+ READYSTATE_COMPLETE = 4
+
+ last_ready_state = -1
+ for retry in xrange(60):
+ try:
+ # TODO(robertshield): Become an event sink instead of polling for
+ # changes to the ready state.
+ last_ready_state = ie.Document.ReadyState
+ if last_ready_state == READYSTATE_COMPLETE:
+ break
+ except:
+ # TODO(robertshield): Find the precise exception related to ie.Document
+ # being not accessible and handle it here.
+ print "Unexpected error:", sys.exc_info()[0]
+ raise
+ time.sleep(1)
+
+ if last_ready_state != READYSTATE_COMPLETE:
+ print "Timeout waiting for " + url
+
+def main():
+ parser = optparse.OptionParser()
+ parser.add_option('-u', '--url_list', default='urllist',
+ help='The path to the list of URLs')
+ (opts, args) = parser.parse_args()
+
+ urls = LoadSiteList(opts.url_list)
+ ie = LaunchIE()
+ for url in urls:
+ RunTest(url, ie)
+ time.sleep(1)
+ ie.visible = 0
+ ie.Quit()
+
+
+if __name__ == '__main__':
+ main()
diff --git a/chrome_frame/tools/test/page_cycler/urllist b/chrome_frame/tools/test/page_cycler/urllist
new file mode 100644
index 0000000..3242774
--- /dev/null
+++ b/chrome_frame/tools/test/page_cycler/urllist
@@ -0,0 +1,500 @@
+http://www.yahoo.com
+http://www.msn.com
+http://www.google.com
+http://www.youtube.com
+http://www.live.com
+http://www.myspace.com
+http://www.baidu.com
+http://www.orkut.com
+http://www.wikipedia.org
+http://www.qq.com
+http://www.yahoo.co.jp
+http://www.microsoft.com
+http://www.megaupload.com
+http://www.sina.com.cn
+http://www.hi5.com
+http://www.blogger.com
+http://www.facebook.com
+http://www.rapidshare.com
+http://www.ebay.com
+http://www.fotolog.net
+http://www.friendster.com
+http://www.sohu.com
+http://www.163.com
+http://www.google.co.uk
+http://www.mail.ru
+http://www.google.fr
+http://www.google.com.br
+http://www.google.de
+http://www.passport.net
+http://www.taobao.com
+http://www.amazon.com
+http://www.skyblog.com
+http://www.yahoo.com.cn
+http://www.google.cl
+http://www.bbc.co.uk
+http://www.google.es
+http://www.google.cn
+http://www.google.pl
+http://www.imdb.com
+http://www.tom.com
+http://www.yandex.ru
+http://www.wretch.cc
+http://www.google.co.jp
+http://www.uol.com.br
+http://www.flickr.com
+http://www.photobucket.com
+http://www.craigslist.org
+http://www.go.com
+http://www.onet.pl
+http://www.imageshack.us
+http://www.google.com.sa
+http://www.google.com.mx
+http://www.aol.com
+http://www.google.com.tr
+http://www.dailymotion.com
+http://www.allegro.pl
+http://www.rambler.ru
+http://www.fc2.com
+http://www.xunlei.com
+http://www.ebay.co.uk
+http://www.google.com.ar
+http://www.google.ca
+http://www.seznam.cz
+http://www.mixi.jp
+http://www.free.fr
+http://www.imagevenue.com
+http://www.google.com.pe
+http://www.google.it
+http://www.google.co.in
+http://www.terra.com.br
+http://www.cnn.com
+http://www.livejournal.com
+http://www.skyrock.com
+http://www.ebay.de
+http://www.geocities.com
+http://www.adultfriendfinder.com
+http://www.wp.pl
+http://www.googlesyndication.com
+http://www.adobe.com
+http://www.google.co.ve
+http://www.bebo.com
+http://www.xanga.com
+http://www.globo.com
+http://www.google.com.co
+http://www.google.com.eg
+http://www.one.lt
+http://www.wordpress.com
+http://www.rakuten.co.jp
+http://www.uwants.com
+http://www.sendspace.com
+http://www.apple.com
+http://www.google.com.vn
+http://www.discuss.com.hk
+http://www.deviantart.com
+http://www.digg.com
+http://www.tagged.com
+http://www.livedoor.com
+http://www.vnexpress.net
+http://www.eastmoney.com
+http://www.soso.com
+http://www.starware.com
+http://www.badongo.com
+http://www.google.co.th
+http://www.naver.com
+http://www.rediff.com
+http://www.iwiw.hu
+http://www.icq.com
+http://www.vnet.cn
+http://www.ig.com.br
+http://www.badoo.com
+http://www.alibaba.com
+http://www.google.com.au
+http://www.maktoob.com
+http://www.newsgroup.la
+http://www.about.com
+http://www.download.com
+http://www.kooora.com
+http://www.google.co.il
+http://www.comcast.net
+http://www.google.nl
+http://www.google.be
+http://www.megarotic.com
+http://www.sexyono.com
+http://www.sourceforge.net
+http://www.walla.co.il
+http://www.fotka.pl
+http://www.pchome.com.tw
+http://www.theplanet.com
+http://www.google.ae
+http://www.orange.fr
+http://www.interia.pl
+http://www.invisionfree.com
+http://www.rapidshare.de
+http://www.tudou.com
+http://www.statcounter.com
+http://www.56.com
+http://www.netlog.com
+http://www.4shared.com
+http://www.mywebsearch.com
+http://www.digitalpoint.com
+http://www.mininova.org
+http://www.dantri.com.vn
+http://www.126.com
+http://www.goo.ne.jp
+http://www.dell.com
+http://www.google.co.hu
+http://www.google.pt
+http://www.information.com
+http://www.youporn.com
+http://www.metacafe.com
+http://www.google.lt
+http://www.google.ro
+http://www.nba.com
+http://www.metroflog.com
+http://www.miniclip.com
+http://www.cnet.com
+http://www.sogou.com
+http://www.ebay.fr
+http://www.weather.com
+http://www.depositfiles.com
+http://www.hp.com
+http://www.google.ru
+http://www.51.com
+http://www.infoseek.co.jp
+http://www.linkedin.com
+http://www.gmx.net
+http://www.xinhuanet.com
+http://www.narod.ru
+http://www.multiply.com
+http://www.pomoho.com
+http://www.nytimes.com
+http://www.yourfilehost.com
+http://www.gamespot.com
+http://www.typepad.com
+http://www.ask.com
+http://www.89.com
+http://www.geocities.jp
+http://www.yam.com
+http://www.neopets.com
+http://www.mercadolivre.com.br
+http://www.centrum.cz
+http://www.mapquest.com
+http://www.univision.com
+http://www.zol.com.cn
+http://www.veoh.com
+http://www.amazon.co.jp
+http://www.6park.com
+http://www.24h.com.vn
+http://www.china.com
+http://www.minijuegos.com
+http://www.sh3bwah.com
+http://www.mop.com
+http://www.miarroba.com
+http://www.tianya.cn
+http://www.ign.com
+http://www.technorati.com
+http://www.yesky.com
+http://www.pornotube.com
+http://www.google.hr
+http://www.cmfu.com
+http://www.aebn.net
+http://www.mynet.com
+http://www.nifty.com
+http://www.livejasmin.com
+http://www.wwe.com
+http://www.daum.net
+http://www.nastydollars.com
+http://www.filefront.com
+http://www.bangbros1.com
+http://www.doubleclick.com
+http://www.torrentspy.com
+http://www.zaycev.net
+http://www.chinaren.com
+http://www.youku.com
+http://www.altavista.com
+http://www.spiegel.de
+http://www.icio.us
+http://www.hinet.net
+http://www.anonym.to
+http://www.usercash.com
+http://www.gamer.com.tw
+http://www.sapo.pt
+http://www.2ch.net
+http://www.tripod.com
+http://www.reference.com
+http://www.blogchina.com
+http://www.piczo.com
+http://www.startimes2.com
+http://www.zshare.net
+http://www.uusee.com
+http://www.dada.net
+http://www.amazon.co.uk
+http://www.panet.co.il
+http://www.6rb.com
+http://www.smileycentral.com
+http://www.juggcrew.com
+http://www.liveinternet.ru
+http://www.google.com.my
+http://www.mercadolibre.com.ar
+http://www.sakura.ne.jp
+http://www.gamefaqs.com
+http://www.jrj.com.cn
+http://www.fanfiction.net
+http://www.libero.it
+http://www.google.se
+http://www.milliyet.com.tr
+http://www.imeem.com
+http://www.archive.org
+http://www.mediafire.com
+http://www.google.com.sg
+http://www.blogfa.com
+http://www.filefactory.com
+http://www.softonic.com
+http://www.homeway.com.cn
+http://www.google.com.tw
+http://www.pconline.com.cn
+http://www.blog.cz
+http://www.symantec.com
+http://www.biglobe.ne.jp
+http://www.mlb.com
+http://www.webshots.com
+http://www.hatena.ne.jp
+http://www.phoenixtv.com
+http://www.google.co.cr
+http://www.nate.com
+http://www.google.gr
+http://www.mobile.de
+http://www.pornaccess.com
+http://www.ynet.co.il
+http://www.amazon.de
+http://www.imagefap.com
+http://www.whenu.com
+http://www.vietnamnet.vn
+http://www.ebay.it
+http://www.hurriyet.com.tr
+http://www.freewebs.com
+http://www.tuoitre.com.vn
+http://www.php.net
+http://www.broadcaster.com
+http://www.google.at
+http://www.xnxx.com
+http://www.studiverzeichnis.com
+http://www.elmundo.es
+http://www.it168.com
+http://www.lide.cz
+http://www.wikimedia.org
+http://www.wefong.com
+http://www.mercadolibre.com.mx
+http://www.thepiratebay.org
+http://www.istockphoto.com
+http://www.sanook.com
+http://www.petardas.com
+http://www.answers.com
+http://www.no-ip.com
+http://www.match.com
+http://www.ebay.com.au
+http://www.payserve.com
+http://www.soufun.com
+http://www.flurl.com
+http://www.alice.it
+http://www.globalsearch.cz
+http://www.888.com
+http://www.isohunt.com
+http://www.poco.cn
+http://www.ameblo.jp
+http://www.heise.de
+http://www.6rbtop.com
+http://www.google.co.id
+http://www.godaddy.com
+http://www.google.fi
+http://www.bala.com.cn
+http://www.break.com
+http://www.stumbleupon.com
+http://www.feedburner.com
+http://www.ngoisao.net
+http://www.atnext.com
+http://www.screensavers.com
+http://www.17173.com
+http://www.domaintools.com
+http://www.nih.gov
+http://www.novinky.cz
+http://www.freeones.com
+http://www.last.fm
+http://www.skype.com
+http://www.google.bg
+http://www.21cn.com
+http://www.o2.pl
+http://www.musica.com
+http://www.pplive.com
+http://www.hkjc.com
+http://www.grono.net
+http://www.bramjnet.com
+http://www.adbrite.com
+http://www.divx.com
+http://www.pcpop.com
+http://www.verycd.com
+http://www.cnfol.com
+http://www.google.ch
+http://www.myway.com
+http://www.wangyou.com
+http://www.gazeta.pl
+http://www.milta1980.co.uk
+http://www.prizee.com
+http://www.livescore.com
+http://www.facebox.com
+http://www.overture.com
+http://www.google.com.ua
+http://www.cartoonnetwork.com
+http://www.nicovideo.jp
+http://www.ku6.com
+http://www.it.com.cn
+http://www.one.lv
+http://www.t-online.de
+http://www.kooxoo.com
+http://www.super.cz
+http://www.google.co.ma
+http://www.xuite.net
+http://www.qihoo.com
+http://www.freelotto.com
+http://www.sedoparking.com
+http://www.monografias.com
+http://www.dmoz.org
+http://www.foxsports.com
+http://www.mocxi.com
+http://www.google.sk
+http://www.as7apcool.com
+http://www.igw.net.sa
+http://www.slide.com
+http://www.zhanzuo.com
+http://www.idnes.cz
+http://www.mobile9.com
+http://www.hawaaworld.com
+http://www.aljazeera.net
+http://www.nana.co.il
+http://www.seesaa.net
+http://www.51job.com
+http://www.abv.bg
+http://www.paipai.com
+http://www.real.com
+http://www.cctv.com
+http://www.tv.com
+http://www.terra.cl
+http://www.google.com.ph
+http://www.wannawatch.com
+http://www.thottbot.com
+http://www.4399.com
+http://www.5show.com
+http://www.vkontakte.ru
+http://www.cricinfo.com
+http://www.hyves.nl
+http://www.google.ie
+http://www.cnnic.cn
+http://www.tinypic.com
+http://www.forumer.com
+http://www.demonoid.com
+http://www.ouou.com
+http://www.mozilla.com
+http://www.naukri.com
+http://www.volam.com.vn
+http://www.torrentz.com
+http://www.google.co.za
+http://www.google.jo
+http://www.dvd4arab.com
+http://www.msn.ca
+http://www.clarin.com
+http://www.rincondelvago.com
+http://www.dmm.co.jp
+http://www.1ting.com
+http://www.everythinggirl.com
+http://www.freemail.hu
+http://www.xtube.com
+http://www.4399.net
+http://www.ocn.ne.jp
+http://www.aweber.com
+http://www.monster.com
+http://www.chinaz.com
+http://www.careerbuilder.com
+http://www.megaclick.com
+http://www.jeeran.com
+http://www.w3.org
+http://www.uume.com
+http://www.sitepoint.com
+http://www.people.com.cn
+http://www.google.com.uy
+http://www.mybloglog.com
+http://www.9you.com
+http://www.ebay.com.cn
+http://www.zedo.com
+http://www.juegosjuegos.com
+http://www.sweetim.com
+http://www.ikea.com
+http://www.walmart.com
+http://www.earthlink.net
+http://www.turboupload.com
+http://www.crunchyroll.com
+http://www.google.com.bh
+http://www.flogao.com.br
+http://www.usps.com
+http://www.6arab.com
+http://www.target.com
+http://www.wordreference.com
+http://www.perfspot.com
+http://www.gigasize.com
+http://www.mercadolibre.com.ve
+http://www.stockstar.com
+http://www.3721.com
+http://www.upspiral.com
+http://www.clubbox.co.kr
+http://www.trademe.co.nz
+http://www.nokia.com
+http://www.met-art.com
+http://www.ppstream.com
+http://www.ibm.com
+http://www.qianlong.com
+http://www.softpedia.com
+http://www.adultadworld.com
+http://www.webmasterworld.com
+http://www.ifolder.ru
+http://www.mofile.com
+http://www.delfi.lt
+http://www.sina.com.hk
+http://www.google.com.bo
+http://www.excite.co.jp
+http://www.myfreepaysite.com
+http://www.ev1servers.net
+http://www.dyndns.org
+http://www.wordpress.org
+http://www.drivecleaner.com
+http://www.google.co.nz
+http://www.people.com
+http://www.expedia.com
+http://www.sonyericsson.com
+http://www.msn.com.cn
+http://www.hubotv.com
+http://www.bangbros.com
+http://www.mediaplex.com
+http://www.google.com.qa
+http://www.videosz.com
+http://www.leo.org
+http://www.atlas.cz
+http://www.kinghost.com
+http://www.aljayyash.net
+http://www.irc-galleria.net
+http://www.xiaonei.com
+http://www.nnm.ru
+http://www.tv-links.co.uk
+http://www.indiatimes.com
+http://www.icbc.com.cn
+http://www.draugiem.lv
+http://www.6rooms.com
+http://www.hc360.com
+http://www.focus.cn
+http://www.atdmt.com
+http://www.chilewarez.org
+http://www.tripadvisor.com
+http://www.mcanime.net
+http://www.esnips.com
+http://www.chinamobile.com
+http://www.ebay.ca
+http://www.im.tv
diff --git a/chrome_frame/tools/test/reference_build/chrome/First Run b/chrome_frame/tools/test/reference_build/chrome/First Run
new file mode 100644
index 0000000..852ad16
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/First Run
@@ -0,0 +1 @@
+krome
diff --git a/chrome_frame/tools/test/reference_build/chrome/avcodec-52.dll b/chrome_frame/tools/test/reference_build/chrome/avcodec-52.dll
new file mode 100644
index 0000000..9b55506
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/avcodec-52.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/avformat-52.dll b/chrome_frame/tools/test/reference_build/chrome/avformat-52.dll
new file mode 100644
index 0000000..33d3dcf
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/avformat-52.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/avutil-50.dll b/chrome_frame/tools/test/reference_build/chrome/avutil-50.dll
new file mode 100644
index 0000000..63f4d2d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/avutil-50.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/chrome.dll b/chrome_frame/tools/test/reference_build/chrome/chrome.dll
new file mode 100644
index 0000000..bc17fe4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/chrome.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/chrome.exe b/chrome_frame/tools/test/reference_build/chrome/chrome.exe
new file mode 100644
index 0000000..040fbaa
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/chrome.exe
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/chrome_dll.pdb b/chrome_frame/tools/test/reference_build/chrome/chrome_dll.pdb
new file mode 100644
index 0000000..203f157
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/chrome_dll.pdb
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/chrome_exe.pdb b/chrome_frame/tools/test/reference_build/chrome/chrome_exe.pdb
new file mode 100644
index 0000000..5d79f8b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/chrome_exe.pdb
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/crash_service.exe b/chrome_frame/tools/test/reference_build/chrome/crash_service.exe
new file mode 100644
index 0000000..1eac8ee
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/crash_service.exe
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/gears.dll b/chrome_frame/tools/test/reference_build/chrome/gears.dll
new file mode 100644
index 0000000..db6d864
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/gears.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/icudt42.dll b/chrome_frame/tools/test/reference_build/chrome/icudt42.dll
new file mode 100644
index 0000000..40a49c1
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/icudt42.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ar.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ar.dll
new file mode 100644
index 0000000..d925ac3
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ar.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/bg.dll b/chrome_frame/tools/test/reference_build/chrome/locales/bg.dll
new file mode 100644
index 0000000..90d4933
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/bg.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/bn.dll b/chrome_frame/tools/test/reference_build/chrome/locales/bn.dll
new file mode 100644
index 0000000..3364aaa
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/bn.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ca.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ca.dll
new file mode 100644
index 0000000..9d8a07a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ca.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/cs.dll b/chrome_frame/tools/test/reference_build/chrome/locales/cs.dll
new file mode 100644
index 0000000..c4b5d1c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/cs.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/da.dll b/chrome_frame/tools/test/reference_build/chrome/locales/da.dll
new file mode 100644
index 0000000..239cf00
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/da.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/de.dll b/chrome_frame/tools/test/reference_build/chrome/locales/de.dll
new file mode 100644
index 0000000..44da5d3
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/de.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/el.dll b/chrome_frame/tools/test/reference_build/chrome/locales/el.dll
new file mode 100644
index 0000000..4afbade
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/el.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/en-GB.dll b/chrome_frame/tools/test/reference_build/chrome/locales/en-GB.dll
new file mode 100644
index 0000000..a3cc174
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/en-GB.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/en-US.dll b/chrome_frame/tools/test/reference_build/chrome/locales/en-US.dll
new file mode 100644
index 0000000..6b4db3c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/en-US.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/es-419.dll b/chrome_frame/tools/test/reference_build/chrome/locales/es-419.dll
new file mode 100644
index 0000000..58ee8bc
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/es-419.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/es.dll b/chrome_frame/tools/test/reference_build/chrome/locales/es.dll
new file mode 100644
index 0000000..96168ad
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/es.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/et.dll b/chrome_frame/tools/test/reference_build/chrome/locales/et.dll
new file mode 100644
index 0000000..237750df
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/et.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/fi.dll b/chrome_frame/tools/test/reference_build/chrome/locales/fi.dll
new file mode 100644
index 0000000..579a2a2
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/fi.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/fil.dll b/chrome_frame/tools/test/reference_build/chrome/locales/fil.dll
new file mode 100644
index 0000000..12cc22e
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/fil.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/fr.dll b/chrome_frame/tools/test/reference_build/chrome/locales/fr.dll
new file mode 100644
index 0000000..87003ac
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/fr.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/gu.dll b/chrome_frame/tools/test/reference_build/chrome/locales/gu.dll
new file mode 100644
index 0000000..0fdc291
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/gu.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/he.dll b/chrome_frame/tools/test/reference_build/chrome/locales/he.dll
new file mode 100644
index 0000000..332fa93
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/he.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/hi.dll b/chrome_frame/tools/test/reference_build/chrome/locales/hi.dll
new file mode 100644
index 0000000..1eee966
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/hi.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/hr.dll b/chrome_frame/tools/test/reference_build/chrome/locales/hr.dll
new file mode 100644
index 0000000..debe9d0
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/hr.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/hu.dll b/chrome_frame/tools/test/reference_build/chrome/locales/hu.dll
new file mode 100644
index 0000000..8f6a8c2
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/hu.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/id.dll b/chrome_frame/tools/test/reference_build/chrome/locales/id.dll
new file mode 100644
index 0000000..3c78bcc
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/id.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/it.dll b/chrome_frame/tools/test/reference_build/chrome/locales/it.dll
new file mode 100644
index 0000000..25c39f7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/it.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ja.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ja.dll
new file mode 100644
index 0000000..1378e47
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ja.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/kn.dll b/chrome_frame/tools/test/reference_build/chrome/locales/kn.dll
new file mode 100644
index 0000000..6154988
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/kn.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ko.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ko.dll
new file mode 100644
index 0000000..bbe62ed
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ko.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/lt.dll b/chrome_frame/tools/test/reference_build/chrome/locales/lt.dll
new file mode 100644
index 0000000..fd9069e
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/lt.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/lv.dll b/chrome_frame/tools/test/reference_build/chrome/locales/lv.dll
new file mode 100644
index 0000000..a402aec
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/lv.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ml.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ml.dll
new file mode 100644
index 0000000..6902ead
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ml.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/mr.dll b/chrome_frame/tools/test/reference_build/chrome/locales/mr.dll
new file mode 100644
index 0000000..d63aafe
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/mr.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/nb.dll b/chrome_frame/tools/test/reference_build/chrome/locales/nb.dll
new file mode 100644
index 0000000..7043aa6
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/nb.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/nl.dll b/chrome_frame/tools/test/reference_build/chrome/locales/nl.dll
new file mode 100644
index 0000000..52fbf1d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/nl.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/or.dll b/chrome_frame/tools/test/reference_build/chrome/locales/or.dll
new file mode 100644
index 0000000..34428ad
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/or.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/pl.dll b/chrome_frame/tools/test/reference_build/chrome/locales/pl.dll
new file mode 100644
index 0000000..2836731
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/pl.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/pt-BR.dll b/chrome_frame/tools/test/reference_build/chrome/locales/pt-BR.dll
new file mode 100644
index 0000000..5aa52e8
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/pt-BR.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/pt-PT.dll b/chrome_frame/tools/test/reference_build/chrome/locales/pt-PT.dll
new file mode 100644
index 0000000..cc523c9
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/pt-PT.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ro.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ro.dll
new file mode 100644
index 0000000..0ee4d70
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ro.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ru.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ru.dll
new file mode 100644
index 0000000..3fbfcfe
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ru.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/sk.dll b/chrome_frame/tools/test/reference_build/chrome/locales/sk.dll
new file mode 100644
index 0000000..da49806
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/sk.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/sl.dll b/chrome_frame/tools/test/reference_build/chrome/locales/sl.dll
new file mode 100644
index 0000000..186d2f0
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/sl.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/sr.dll b/chrome_frame/tools/test/reference_build/chrome/locales/sr.dll
new file mode 100644
index 0000000..9278069
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/sr.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/sv.dll b/chrome_frame/tools/test/reference_build/chrome/locales/sv.dll
new file mode 100644
index 0000000..29ff6bf
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/sv.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/ta.dll b/chrome_frame/tools/test/reference_build/chrome/locales/ta.dll
new file mode 100644
index 0000000..711739a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/ta.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/te.dll b/chrome_frame/tools/test/reference_build/chrome/locales/te.dll
new file mode 100644
index 0000000..5468bed
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/te.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/th.dll b/chrome_frame/tools/test/reference_build/chrome/locales/th.dll
new file mode 100644
index 0000000..bb41ca2
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/th.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/tr.dll b/chrome_frame/tools/test/reference_build/chrome/locales/tr.dll
new file mode 100644
index 0000000..5a9333c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/tr.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/uk.dll b/chrome_frame/tools/test/reference_build/chrome/locales/uk.dll
new file mode 100644
index 0000000..830904a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/uk.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/vi.dll b/chrome_frame/tools/test/reference_build/chrome/locales/vi.dll
new file mode 100644
index 0000000..3c3c425
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/vi.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/zh-CN.dll b/chrome_frame/tools/test/reference_build/chrome/locales/zh-CN.dll
new file mode 100644
index 0000000..0a09475
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/zh-CN.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/locales/zh-TW.dll b/chrome_frame/tools/test/reference_build/chrome/locales/zh-TW.dll
new file mode 100644
index 0000000..237c46c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/locales/zh-TW.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/mini_installer.pdb b/chrome_frame/tools/test/reference_build/chrome/mini_installer.pdb
new file mode 100644
index 0000000..111f61d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/mini_installer.pdb
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/BottomUpProfileDataGridTree.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/BottomUpProfileDataGridTree.js
new file mode 100644
index 0000000..89b4ddc
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/BottomUpProfileDataGridTree.js
@@ -0,0 +1,252 @@
+/*
+ * Copyright (C) 2009 280 North Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+// Bottom Up Profiling shows the entire callstack backwards:
+// The root node is a representation of each individual function called, and each child of that node represents
+// a reverse-callstack showing how many of those calls came from it. So, unlike top-down, the statistics in
+// each child still represent the root node. We have to be particularly careful of recursion with this mode
+// because a root node can represent itself AND an ancestor.
+
+WebInspector.BottomUpProfileDataGridTree = function(/*ProfileView*/ aProfileView, /*ProfileNode*/ aProfileNode)
+{
+ WebInspector.ProfileDataGridTree.call(this, aProfileView, aProfileNode);
+
+ // Iterate each node in pre-order.
+ var profileNodeUIDs = 0;
+ var profileNodeGroups = [[], [aProfileNode]];
+ var visitedProfileNodesForCallUID = {};
+
+ this._remainingNodeInfos = [];
+
+ for (var profileNodeGroupIndex = 0; profileNodeGroupIndex < profileNodeGroups.length; ++profileNodeGroupIndex) {
+ var parentProfileNodes = profileNodeGroups[profileNodeGroupIndex];
+ var profileNodes = profileNodeGroups[++profileNodeGroupIndex];
+ var count = profileNodes.length;
+
+ for (var index = 0; index < count; ++index) {
+ var profileNode = profileNodes[index];
+
+ if (!profileNode.UID)
+ profileNode.UID = ++profileNodeUIDs;
+
+ if (profileNode.head && profileNode !== profileNode.head) {
+ // The total time of this ancestor is accounted for if we're in any form of recursive cycle.
+ var visitedNodes = visitedProfileNodesForCallUID[profileNode.callUID];
+ var totalTimeAccountedFor = false;
+
+ if (!visitedNodes) {
+ visitedNodes = {}
+ visitedProfileNodesForCallUID[profileNode.callUID] = visitedNodes;
+ } else {
+ // The total time for this node has already been accounted for iff one of it's parents has already been visited.
+ // We can do this check in this style because we are traversing the tree in pre-order.
+ var parentCount = parentProfileNodes.length;
+ for (var parentIndex = 0; parentIndex < parentCount; ++parentIndex) {
+ if (visitedNodes[parentProfileNodes[parentIndex].UID]) {
+ totalTimeAccountedFor = true;
+ break;
+ }
+ }
+ }
+
+ visitedNodes[profileNode.UID] = true;
+
+ this._remainingNodeInfos.push({ ancestor:profileNode, focusNode:profileNode, totalTimeAccountedFor:totalTimeAccountedFor });
+ }
+
+ var children = profileNode.children;
+ if (children.length) {
+ profileNodeGroups.push(parentProfileNodes.concat([profileNode]))
+ profileNodeGroups.push(children);
+ }
+ }
+ }
+
+ // Populate the top level nodes.
+ WebInspector.BottomUpProfileDataGridNode.prototype._populate.call(this);
+
+ return this;
+}
+
+WebInspector.BottomUpProfileDataGridTree.prototype = {
+ // When focusing, we keep the entire callstack up to this ancestor.
+ focus: function(/*ProfileDataGridNode*/ profileDataGridNode)
+ {
+ if (!profileDataGridNode)
+ return;
+
+ this._save();
+
+ var currentNode = profileDataGridNode;
+ var focusNode = profileDataGridNode;
+
+ while (currentNode.parent && (currentNode instanceof WebInspector.ProfileDataGridNode)) {
+ currentNode._takePropertiesFromProfileDataGridNode(profileDataGridNode);
+
+ focusNode = currentNode;
+ currentNode = currentNode.parent;
+
+ if (currentNode instanceof WebInspector.ProfileDataGridNode)
+ currentNode._keepOnlyChild(focusNode);
+ }
+
+ this.children = [focusNode];
+ this.totalTime = profileDataGridNode.totalTime;
+ },
+
+ exclude: function(/*ProfileDataGridNode*/ profileDataGridNode)
+ {
+ if (!profileDataGridNode)
+ return;
+
+ this._save();
+
+ var excludedCallUID = profileDataGridNode.callUID;
+ var excludedTopLevelChild = this.childrenByCallUID[excludedCallUID];
+
+ // If we have a top level node that is excluded, get rid of it completely (not keeping children),
+ // since bottom up data relies entirely on the root node.
+ if (excludedTopLevelChild)
+ this.children.remove(excludedTopLevelChild);
+
+ var children = this.children;
+ var count = children.length;
+
+ for (var index = 0; index < count; ++index)
+ children[index]._exclude(excludedCallUID);
+
+ if (this.lastComparator)
+ this.sort(this.lastComparator, true);
+ }
+}
+
+WebInspector.BottomUpProfileDataGridTree.prototype.__proto__ = WebInspector.ProfileDataGridTree.prototype;
+
+WebInspector.BottomUpProfileDataGridNode = function(/*ProfileView*/ profileView, /*ProfileNode*/ profileNode, /*BottomUpProfileDataGridTree*/ owningTree)
+{
+ // In bottom up mode, our parents are our children since we display an inverted tree.
+ // However, we don't want to show the very top parent since it is redundant.
+ var hasChildren = !!(profileNode.parent && profileNode.parent.parent);
+
+ WebInspector.ProfileDataGridNode.call(this, profileView, profileNode, owningTree, hasChildren);
+
+ this._remainingNodeInfos = [];
+}
+
+WebInspector.BottomUpProfileDataGridNode.prototype = {
+ _takePropertiesFromProfileDataGridNode: function(/*ProfileDataGridNode*/ profileDataGridNode)
+ {
+ this._save();
+
+ this.selfTime = profileDataGridNode.selfTime;
+ this.totalTime = profileDataGridNode.totalTime;
+ this.numberOfCalls = profileDataGridNode.numberOfCalls;
+ },
+
+ // When focusing, we keep just the members of the callstack.
+ _keepOnlyChild: function(/*ProfileDataGridNode*/ child)
+ {
+ this._save();
+
+ this.removeChildren();
+ this.appendChild(child);
+ },
+
+ _exclude: function(aCallUID)
+ {
+ if (this._remainingNodeInfos)
+ this._populate();
+
+ this._save();
+
+ var children = this.children;
+ var index = this.children.length;
+
+ while (index--)
+ children[index]._exclude(aCallUID);
+
+ var child = this.childrenByCallUID[aCallUID];
+
+ if (child)
+ this._merge(child, true);
+ },
+
+ _merge: function(/*ProfileDataGridNode*/ child, /*Boolean*/ shouldAbsorb)
+ {
+ this.selfTime -= child.selfTime;
+
+ WebInspector.ProfileDataGridNode.prototype._merge.call(this, child, shouldAbsorb);
+ },
+
+ _populate: function(event)
+ {
+ var remainingNodeInfos = this._remainingNodeInfos;
+ var count = remainingNodeInfos.length;
+
+ for (var index = 0; index < count; ++index) {
+ var nodeInfo = remainingNodeInfos[index];
+ var ancestor = nodeInfo.ancestor;
+ var focusNode = nodeInfo.focusNode;
+ var child = this.findChild(ancestor);
+
+ // If we already have this child, then merge the data together.
+ if (child) {
+ var totalTimeAccountedFor = nodeInfo.totalTimeAccountedFor;
+
+ child.selfTime += focusNode.selfTime;
+ child.numberOfCalls += focusNode.numberOfCalls;
+
+ if (!totalTimeAccountedFor)
+ child.totalTime += focusNode.totalTime;
+ } else {
+ // If not, add it as a true ancestor.
+ // In heavy mode, we take our visual identity from ancestor node...
+ var child = new WebInspector.BottomUpProfileDataGridNode(this.profileView, ancestor, this.tree);
+
+ if (ancestor !== focusNode) {
+ // but the actual statistics from the "root" node (bottom of the callstack).
+ child.selfTime = focusNode.selfTime;
+ child.totalTime = focusNode.totalTime;
+ child.numberOfCalls = focusNode.numberOfCalls;
+ }
+
+ this.appendChild(child);
+ }
+
+ var parent = ancestor.parent;
+ if (parent && parent.parent) {
+ nodeInfo.ancestor = parent;
+ child._remainingNodeInfos.push(nodeInfo);
+ }
+ }
+
+ delete this._remainingNodeInfos;
+
+ if (this.removeEventListener)
+ this.removeEventListener("populate", this._populate, this);
+ }
+}
+
+WebInspector.BottomUpProfileDataGridNode.prototype.__proto__ = WebInspector.ProfileDataGridNode.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Breakpoint.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Breakpoint.js
new file mode 100644
index 0000000..292975a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Breakpoint.js
@@ -0,0 +1,96 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Breakpoint = function(url, line, sourceID, condition)
+{
+ this.url = url;
+ this.line = line;
+ this.sourceID = sourceID;
+ this._enabled = true;
+ this._sourceText = "";
+ this._condition = condition || "";
+}
+
+WebInspector.Breakpoint.prototype = {
+ get enabled()
+ {
+ return this._enabled;
+ },
+
+ set enabled(x)
+ {
+ if (this._enabled === x)
+ return;
+
+ this._enabled = x;
+
+ if (this._enabled)
+ this.dispatchEventToListeners("enabled");
+ else
+ this.dispatchEventToListeners("disabled");
+ },
+
+ get sourceText()
+ {
+ return this._sourceText;
+ },
+
+ set sourceText(text)
+ {
+ this._sourceText = text;
+ this.dispatchEventToListeners("text-changed");
+ },
+
+ get label()
+ {
+ var displayName = (this.url ? WebInspector.displayNameForURL(this.url) : WebInspector.UIString("(program)"));
+ return displayName + ":" + this.line;
+ },
+
+ get id()
+ {
+ return this.sourceID + ":" + this.line;
+ },
+
+ get condition()
+ {
+ return this._condition;
+ },
+
+ set condition(c)
+ {
+ c = c || "";
+ if (this._condition === c)
+ return;
+
+ this._condition = c;
+ this.dispatchEventToListeners("condition-changed");
+
+ if (this.enabled)
+ InspectorController.updateBreakpoint(this.sourceID, this.line, c);
+ }
+}
+
+WebInspector.Breakpoint.prototype.__proto__ = WebInspector.Object.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/BreakpointsSidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/BreakpointsSidebarPane.js
new file mode 100644
index 0000000..e6edece
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/BreakpointsSidebarPane.js
@@ -0,0 +1,167 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.BreakpointsSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Breakpoints"));
+
+ this.breakpoints = {};
+
+ this.listElement = document.createElement("ol");
+ this.listElement.className = "breakpoint-list";
+
+ this.emptyElement = document.createElement("div");
+ this.emptyElement.className = "info";
+ this.emptyElement.textContent = WebInspector.UIString("No Breakpoints");
+
+ this.bodyElement.appendChild(this.emptyElement);
+}
+
+WebInspector.BreakpointsSidebarPane.prototype = {
+ addBreakpoint: function(breakpoint)
+ {
+ if (this.breakpoints[breakpoint.id])
+ return;
+
+ this.breakpoints[breakpoint.id] = breakpoint;
+
+ breakpoint.addEventListener("enabled", this._breakpointEnableChanged, this);
+ breakpoint.addEventListener("disabled", this._breakpointEnableChanged, this);
+ breakpoint.addEventListener("text-changed", this._breakpointTextChanged, this);
+
+ this._appendBreakpointElement(breakpoint);
+
+ if (this.emptyElement.parentElement) {
+ this.bodyElement.removeChild(this.emptyElement);
+ this.bodyElement.appendChild(this.listElement);
+ }
+
+ if (!InspectorController.debuggerEnabled() || !breakpoint.sourceID)
+ return;
+
+ if (breakpoint.enabled)
+ InspectorController.addBreakpoint(breakpoint.sourceID, breakpoint.line, breakpoint.condition);
+ },
+
+ _appendBreakpointElement: function(breakpoint)
+ {
+ function checkboxClicked()
+ {
+ breakpoint.enabled = !breakpoint.enabled;
+ }
+
+ function labelClicked()
+ {
+ var script = WebInspector.panels.scripts.scriptOrResourceForID(breakpoint.sourceID);
+ if (script)
+ WebInspector.panels.scripts.showScript(script, breakpoint.line);
+ }
+
+ var breakpointElement = document.createElement("li");
+ breakpoint._breakpointListElement = breakpointElement;
+ breakpointElement._breakpointObject = breakpoint;
+
+ var checkboxElement = document.createElement("input");
+ checkboxElement.className = "checkbox-elem";
+ checkboxElement.type = "checkbox";
+ checkboxElement.checked = breakpoint.enabled;
+ checkboxElement.addEventListener("click", checkboxClicked, false);
+ breakpointElement.appendChild(checkboxElement);
+
+ var labelElement = document.createElement("a");
+ labelElement.textContent = breakpoint.label;
+ labelElement.addEventListener("click", labelClicked, false);
+ breakpointElement.appendChild(labelElement);
+
+ var sourceTextElement = document.createElement("div");
+ sourceTextElement.textContent = breakpoint.sourceText;
+ sourceTextElement.className = "source-text";
+ breakpointElement.appendChild(sourceTextElement);
+
+ var currentElement = this.listElement.firstChild;
+ while (currentElement) {
+ var currentBreak = currentElement._breakpointObject;
+ if (currentBreak.url > breakpoint.url) {
+ this.listElement.insertBefore(breakpointElement, currentElement);
+ return;
+ } else if (currentBreak.url == breakpoint.url && currentBreak.line > breakpoint.line) {
+ this.listElement.insertBefore(breakpointElement, currentElement);
+ return;
+ }
+ currentElement = currentElement.nextSibling;
+ }
+ this.listElement.appendChild(breakpointElement);
+ },
+
+ removeBreakpoint: function(breakpoint)
+ {
+ if (!this.breakpoints[breakpoint.id])
+ return;
+ delete this.breakpoints[breakpoint.id];
+
+ breakpoint.removeEventListener("enabled", null, this);
+ breakpoint.removeEventListener("disabled", null, this);
+ breakpoint.removeEventListener("text-changed", null, this);
+
+ var element = breakpoint._breakpointListElement;
+ element.parentElement.removeChild(element);
+
+ if (!this.listElement.firstChild) {
+ this.bodyElement.removeChild(this.listElement);
+ this.bodyElement.appendChild(this.emptyElement);
+ }
+
+ if (!InspectorController.debuggerEnabled() || !breakpoint.sourceID)
+ return;
+
+ InspectorController.removeBreakpoint(breakpoint.sourceID, breakpoint.line);
+ },
+
+ _breakpointEnableChanged: function(event)
+ {
+ var breakpoint = event.target;
+
+ var checkbox = breakpoint._breakpointListElement.firstChild;
+ checkbox.checked = breakpoint.enabled;
+
+ if (!InspectorController.debuggerEnabled() || !breakpoint.sourceID)
+ return;
+
+ if (breakpoint.enabled)
+ InspectorController.addBreakpoint(breakpoint.sourceID, breakpoint.line, breakpoint.condition);
+ else
+ InspectorController.removeBreakpoint(breakpoint.sourceID, breakpoint.line);
+ },
+
+ _breakpointTextChanged: function(event)
+ {
+ var breakpoint = event.target;
+
+ var sourceTextElement = breakpoint._breakpointListElement.firstChild.nextSibling.nextSibling;
+ sourceTextElement.textContent = breakpoint.sourceText;
+ }
+}
+
+WebInspector.BreakpointsSidebarPane.prototype.__proto__ = WebInspector.SidebarPane.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/CallStackSidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/CallStackSidebarPane.js
new file mode 100644
index 0000000..2fe4315
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/CallStackSidebarPane.js
@@ -0,0 +1,166 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.CallStackSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Call Stack"));
+
+ this._shortcuts = {};
+
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.Period,
+ WebInspector.KeyboardShortcut.Modifiers.Ctrl);
+ this._shortcuts[shortcut] = this._selectNextCallFrameOnStack.bind(this);
+
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.Comma,
+ WebInspector.KeyboardShortcut.Modifiers.Ctrl);
+ this._shortcuts[shortcut] = this._selectPreviousCallFrameOnStack.bind(this);
+}
+
+WebInspector.CallStackSidebarPane.prototype = {
+ update: function(callFrames, sourceIDMap)
+ {
+ this.bodyElement.removeChildren();
+
+ this.placards = [];
+ delete this._selectedCallFrame;
+
+ if (!callFrames) {
+ var infoElement = document.createElement("div");
+ infoElement.className = "info";
+ infoElement.textContent = WebInspector.UIString("Not Paused");
+ this.bodyElement.appendChild(infoElement);
+ return;
+ }
+
+ var title;
+ var subtitle;
+ var scriptOrResource;
+
+ for (var i = 0; i < callFrames.length; ++i) {
+ var callFrame = callFrames[i];
+ switch (callFrame.type) {
+ case "function":
+ title = callFrame.functionName || WebInspector.UIString("(anonymous function)");
+ break;
+ case "program":
+ title = WebInspector.UIString("(program)");
+ break;
+ }
+
+ scriptOrResource = sourceIDMap[callFrame.sourceID];
+ subtitle = WebInspector.displayNameForURL(scriptOrResource.sourceURL || scriptOrResource.url);
+
+ if (callFrame.line > 0) {
+ if (subtitle)
+ subtitle += ":" + callFrame.line;
+ else
+ subtitle = WebInspector.UIString("line %d", callFrame.line);
+ }
+
+ var placard = new WebInspector.Placard(title, subtitle);
+ placard.callFrame = callFrame;
+
+ placard.element.addEventListener("click", this._placardSelected.bind(this), false);
+
+ this.placards.push(placard);
+ this.bodyElement.appendChild(placard.element);
+ }
+ },
+
+ get selectedCallFrame()
+ {
+ return this._selectedCallFrame;
+ },
+
+ set selectedCallFrame(x)
+ {
+ if (this._selectedCallFrame === x)
+ return;
+
+ this._selectedCallFrame = x;
+
+ for (var i = 0; i < this.placards.length; ++i) {
+ var placard = this.placards[i];
+ placard.selected = (placard.callFrame === this._selectedCallFrame);
+ }
+
+ this.dispatchEventToListeners("call frame selected");
+ },
+
+ handleKeyEvent: function(event)
+ {
+ var shortcut = WebInspector.KeyboardShortcut.makeKeyFromEvent(event);
+ var handler = this._shortcuts[shortcut];
+ if (handler) {
+ handler(event);
+ event.preventDefault();
+ event.handled = true;
+ }
+ },
+
+ _selectNextCallFrameOnStack: function()
+ {
+ var index = this._selectedCallFrameIndex();
+ if (index == -1)
+ return;
+ this._selectedPlacardByIndex(index + 1);
+ },
+
+ _selectPreviousCallFrameOnStack: function()
+ {
+ var index = this._selectedCallFrameIndex();
+ if (index == -1)
+ return;
+ this._selectedPlacardByIndex(index - 1);
+ },
+
+ _selectedPlacardByIndex: function(index)
+ {
+ if (index < 0 || index >= this.placards.length)
+ return;
+ var placard = this.placards[index];
+ this.selectedCallFrame = placard.callFrame
+ },
+
+ _selectedCallFrameIndex: function()
+ {
+ if (!this._selectedCallFrame)
+ return -1;
+ for (var i = 0; i < this.placards.length; ++i) {
+ var placard = this.placards[i];
+ if (placard.callFrame === this._selectedCallFrame)
+ return i;
+ }
+ return -1;
+ },
+
+ _placardSelected: function(event)
+ {
+ var placardElement = event.target.enclosingNodeOrSelfWithClass("placard");
+ this.selectedCallFrame = placardElement.placard.callFrame;
+ }
+}
+
+WebInspector.CallStackSidebarPane.prototype.__proto__ = WebInspector.SidebarPane.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Callback.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Callback.js
new file mode 100644
index 0000000..8ae7f95
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Callback.js
@@ -0,0 +1,56 @@
+/*
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Callback = function()
+{
+ this._lastCallbackId = 1;
+ this._callbacks = {};
+}
+
+WebInspector.Callback.prototype = {
+ wrap: function(callback)
+ {
+ var callbackId = this._lastCallbackId++;
+ this._callbacks[callbackId] = callback || function() {};
+ return callbackId;
+ },
+
+ processCallback: function(callbackId, opt_vararg)
+ {
+ var args = Array.prototype.slice.call(arguments, 1);
+ var callback = this._callbacks[callbackId];
+ callback.apply(null, args);
+ delete this._callbacks[callbackId];
+ }
+}
+
+WebInspector.Callback._INSTANCE = new WebInspector.Callback();
+WebInspector.Callback.wrap = WebInspector.Callback._INSTANCE.wrap.bind(WebInspector.Callback._INSTANCE);
+WebInspector.Callback.processCallback = WebInspector.Callback._INSTANCE.processCallback.bind(WebInspector.Callback._INSTANCE);
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ChangesView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ChangesView.js
new file mode 100644
index 0000000..802fdba
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ChangesView.js
@@ -0,0 +1,80 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ChangesView = function(drawer)
+{
+ WebInspector.View.call(this);
+ this.element.innerHTML = "<div style=\"bottom:25%;color:rgb(192,192,192);font-size:12px;height:65px;left:0px;margin:auto;position:absolute;right:0px;text-align:center;top:0px;\"><h1>Not Implemented Yet</h1></div>";
+
+ this.drawer = drawer;
+
+ this.clearButton = document.createElement("button");
+ this.clearButton.id = "clear-changes-status-bar-item";
+ this.clearButton.title = WebInspector.UIString("Clear changes log.");
+ this.clearButton.className = "status-bar-item";
+ this.clearButton.addEventListener("click", this._clearButtonClicked.bind(this), false);
+
+ this.toggleChangesButton = document.getElementById("changes-status-bar-item");
+ this.toggleChangesButton.title = WebInspector.UIString("Show changes view.");
+ this.toggleChangesButton.addEventListener("click", this._toggleChangesButtonClicked.bind(this), false);
+ var anchoredStatusBar = document.getElementById("anchored-status-bar-items");
+ anchoredStatusBar.appendChild(this.toggleChangesButton);
+}
+
+WebInspector.ChangesView.prototype = {
+ _clearButtonClicked: function()
+ {
+ // Not Implemented Yet
+ },
+
+ _toggleChangesButtonClicked: function()
+ {
+ this.drawer.visibleView = this;
+ },
+
+ attach: function(mainElement, statusBarElement)
+ {
+ mainElement.appendChild(this.element);
+ statusBarElement.appendChild(this.clearButton);
+ },
+
+ show: function()
+ {
+ this.toggleChangesButton.addStyleClass("toggled-on");
+ this.toggleChangesButton.title = WebInspector.UIString("Hide changes view.");
+ },
+
+ hide: function()
+ {
+ this.toggleChangesButton.removeStyleClass("toggled-on");
+ this.toggleChangesButton.title = WebInspector.UIString("Show changes view.");
+ }
+}
+
+WebInspector.ChangesView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Color.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Color.js
new file mode 100644
index 0000000..9d9cd76
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Color.js
@@ -0,0 +1,661 @@
+/*
+ * Copyright (C) 2009 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Color = function(str)
+{
+ this.value = str;
+ this._parse();
+}
+
+WebInspector.Color.prototype = {
+ get shorthex()
+ {
+ if ("_short" in this)
+ return this._short;
+
+ if (!this.simple)
+ return null;
+
+ var hex = this.hex;
+ if (hex.charAt(0) === hex.charAt(1) && hex.charAt(2) === hex.charAt(3) && hex.charAt(4) === hex.charAt(5))
+ this._short = hex.charAt(0) + hex.charAt(2) + hex.charAt(4);
+ else
+ this._short = hex;
+
+ return this._short;
+ },
+
+ get hex()
+ {
+ if (!this.simple)
+ return null;
+
+ return this._hex;
+ },
+
+ set hex(x)
+ {
+ this._hex = x;
+ },
+
+ get rgb()
+ {
+ if ("_rgb" in this)
+ return this._rgb;
+
+ if (this.simple)
+ this._rgb = this._hexToRGB(this.hex);
+ else {
+ var rgba = this.rgba;
+ this._rgb = [rgba[0], rgba[1], rgba[2]];
+ }
+
+ return this._rgb;
+ },
+
+ set rgb(x)
+ {
+ this._rgb = x;
+ },
+
+ get hsl()
+ {
+ if ("_hsl" in this)
+ return this._hsl;
+
+ this._hsl = this._rgbToHSL(this.rgb);
+ return this._hsl;
+ },
+
+ set hsl(x)
+ {
+ this._hsl = x;
+ },
+
+ get nickname()
+ {
+ if (typeof this._nickname !== "undefined") // would be set on parse if there was a nickname
+ return this._nickname;
+ else
+ return null;
+ },
+
+ set nickname(x)
+ {
+ this._nickname = x;
+ },
+
+ get rgba()
+ {
+ return this._rgba;
+ },
+
+ set rgba(x)
+ {
+ this._rgba = x;
+ },
+
+ get hsla()
+ {
+ return this._hsla;
+ },
+
+ set hsla(x)
+ {
+ this._hsla = x;
+ },
+
+ hasShortHex: function()
+ {
+ var shorthex = this.shorthex;
+ return (shorthex && shorthex.length === 3);
+ },
+
+ toString: function(format)
+ {
+ if (!format)
+ format = this.format;
+
+ switch (format) {
+ case "rgb":
+ return "rgb(" + this.rgb.join(", ") + ")";
+ case "rgba":
+ return "rgba(" + this.rgba.join(", ") + ")";
+ case "hsl":
+ var hsl = this.hsl;
+ return "hsl(" + hsl[0] + ", " + hsl[1] + "%, " + hsl[2] + "%)";
+ case "hsla":
+ var hsla = this.hsla;
+ return "hsla(" + hsla[0] + ", " + hsla[1] + "%, " + hsla[2] + "%, " + hsla[3] + ")";
+ case "hex":
+ return "#" + this.hex;
+ case "shorthex":
+ return "#" + this.shorthex;
+ case "nickname":
+ return this.nickname;
+ }
+
+ throw "invalid color format";
+ },
+
+ _rgbToHex: function(rgb)
+ {
+ var r = parseInt(rgb[0]).toString(16);
+ var g = parseInt(rgb[1]).toString(16);
+ var b = parseInt(rgb[2]).toString(16);
+ if (r.length === 1)
+ r = "0" + r;
+ if (g.length === 1)
+ g = "0" + g;
+ if (b.length === 1)
+ b = "0" + b;
+
+ return (r + g + b).toUpperCase();
+ },
+
+ _hexToRGB: function(hex)
+ {
+ var r = parseInt(hex.substring(0,2), 16);
+ var g = parseInt(hex.substring(2,4), 16);
+ var b = parseInt(hex.substring(4,6), 16);
+
+ return [r, g, b];
+ },
+
+ _rgbToHSL: function(rgb)
+ {
+ var r = parseInt(rgb[0]) / 255;
+ var g = parseInt(rgb[1]) / 255;
+ var b = parseInt(rgb[2]) / 255;
+ var max = Math.max(r, g, b);
+ var min = Math.min(r, g, b);
+ var diff = max - min;
+ var add = max + min;
+
+ if (min === max)
+ var h = 0;
+ else if (r === max)
+ var h = ((60 * (g - b) / diff) + 360) % 360;
+ else if (g === max)
+ var h = (60 * (b - r) / diff) + 120;
+ else
+ var h = (60 * (r - g) / diff) + 240;
+
+ var l = 0.5 * add;
+
+ if (l === 0)
+ var s = 0;
+ else if (l === 1)
+ var s = 1;
+ else if (l <= 0.5)
+ var s = diff / add;
+ else
+ var s = diff / (2 - add);
+
+ h = Math.round(h);
+ s = Math.round(s*100);
+ l = Math.round(l*100);
+
+ return [h, s, l];
+ },
+
+ _hslToRGB: function(hsl)
+ {
+ var h = parseFloat(hsl[0]) / 360;
+ var s = parseFloat(hsl[1]) / 100;
+ var l = parseFloat(hsl[2]) / 100;
+
+ if (l <= 0.5)
+ var q = l * (1 + s);
+ else
+ var q = l + s - (l * s);
+
+ var p = 2 * l - q;
+
+ var tr = h + (1 / 3);
+ var tg = h;
+ var tb = h - (1 / 3);
+
+ var r = Math.round(hueToRGB(p, q, tr) * 255);
+ var g = Math.round(hueToRGB(p, q, tg) * 255);
+ var b = Math.round(hueToRGB(p, q, tb) * 255);
+ return [r, g, b];
+
+ function hueToRGB(p, q, h) {
+ if (h < 0)
+ h += 1;
+ else if (h > 1)
+ h -= 1;
+
+ if ((h * 6) < 1)
+ return p + (q - p) * h * 6;
+ else if ((h * 2) < 1)
+ return q;
+ else if ((h * 3) < 2)
+ return p + (q - p) * ((2 / 3) - h) * 6;
+ else
+ return p;
+ }
+ },
+
+ _rgbaToHSLA: function(rgba)
+ {
+ var alpha = rgba[3];
+ var hsl = this._rgbToHSL(rgba)
+ hsl.push(alpha);
+ return hsl;
+ },
+
+ _hslaToRGBA: function(hsla)
+ {
+ var alpha = hsla[3];
+ var rgb = this._hslToRGB(hsla);
+ rgb.push(alpha);
+ return rgb;
+ },
+
+ _parse: function()
+ {
+ // Special Values - Advanced but Must Be Parsed First - transparent
+ var value = this.value.toLowerCase().replace(/%|\s+/g, "");
+ if (value in WebInspector.Color.AdvancedNickNames) {
+ this.format = "nickname";
+ var set = WebInspector.Color.AdvancedNickNames[value];
+ this.simple = false;
+ this.rgba = set[0];
+ this.hsla = set[1];
+ this.nickname = set[2];
+ this.alpha = set[0][3];
+ return;
+ }
+
+ // Simple - #hex, rgb(), nickname, hsl()
+ var simple = /^(?:#([0-9a-f]{3,6})|rgb\(([^)]+)\)|(\w+)|hsl\(([^)]+)\))$/i;
+ var match = this.value.match(simple);
+ if (match) {
+ this.simple = true;
+
+ if (match[1]) { // hex
+ var hex = match[1].toUpperCase();
+ if (hex.length === 3) {
+ this.format = "shorthex";
+ this.hex = hex.charAt(0) + hex.charAt(0) + hex.charAt(1) + hex.charAt(1) + hex.charAt(2) + hex.charAt(2);
+ } else {
+ this.format = "hex";
+ this.hex = hex;
+ }
+ } else if (match[2]) { // rgb
+ this.format = "rgb";
+ var rgb = match[2].split(/\s*,\s*/);
+ this.rgb = rgb;
+ this.hex = this._rgbToHex(rgb);
+ } else if (match[3]) { // nickname
+ var nickname = match[3].toLowerCase();
+ if (nickname in WebInspector.Color.Nicknames) {
+ this.format = "nickname";
+ this.hex = WebInspector.Color.Nicknames[nickname];
+ } else // unknown name
+ throw "unknown color name";
+ } else if (match[4]) { // hsl
+ this.format = "hsl";
+ var hsl = match[4].replace(/%g/, "").split(/\s*,\s*/);
+ this.hsl = hsl;
+ this.rgb = this._hslToRGB(hsl);
+ this.hex = this._rgbToHex(this.rgb);
+ }
+
+ // Fill in the values if this is a known hex color
+ var hex = this.hex;
+ if (hex && hex in WebInspector.Color.HexTable) {
+ var set = WebInspector.Color.HexTable[hex];
+ this.rgb = set[0];
+ this.hsl = set[1];
+ this.nickname = set[2];
+ }
+
+ return;
+ }
+
+ // Advanced - rgba(), hsla()
+ var advanced = /^(?:rgba\(([^)]+)\)|hsla\(([^)]+)\))$/;
+ match = this.value.match(advanced);
+ if (match) {
+ this.simple = false;
+ if (match[1]) { // rgba
+ this.format = "rgba";
+ this.rgba = match[1].split(/\s*,\s*/);
+ this.hsla = this._rgbaToHSLA(this.rgba);
+ this.alpha = this.rgba[3];
+ } else if (match[2]) { // hsla
+ this.format = "hsla";
+ this.hsla = match[2].replace(/%/g, "").split(/\s*,\s*/);
+ this.rgba = this._hslaToRGBA(this.hsla);
+ this.alpha = this.hsla[3];
+ }
+
+ return;
+ }
+
+ // Could not parse as a valid color
+ throw "could not parse color";
+ }
+}
+
+// Simple Values: [rgb, hsl, nickname]
+WebInspector.Color.HexTable = {
+ "000000": [[0, 0, 0], [0, 0, 0], "black"],
+ "000080": [[0, 0, 128], [240, 100, 25], "navy"],
+ "00008B": [[0, 0, 139], [240, 100, 27], "darkBlue"],
+ "0000CD": [[0, 0, 205], [240, 100, 40], "mediumBlue"],
+ "0000FF": [[0, 0, 255], [240, 100, 50], "blue"],
+ "006400": [[0, 100, 0], [120, 100, 20], "darkGreen"],
+ "008000": [[0, 128, 0], [120, 100, 25], "green"],
+ "008080": [[0, 128, 128], [180, 100, 25], "teal"],
+ "008B8B": [[0, 139, 139], [180, 100, 27], "darkCyan"],
+ "00BFFF": [[0, 191, 255], [195, 100, 50], "deepSkyBlue"],
+ "00CED1": [[0, 206, 209], [181, 100, 41], "darkTurquoise"],
+ "00FA9A": [[0, 250, 154], [157, 100, 49], "mediumSpringGreen"],
+ "00FF00": [[0, 255, 0], [120, 100, 50], "lime"],
+ "00FF7F": [[0, 255, 127], [150, 100, 50], "springGreen"],
+ "00FFFF": [[0, 255, 255], [180, 100, 50], "cyan"],
+ "191970": [[25, 25, 112], [240, 64, 27], "midnightBlue"],
+ "1E90FF": [[30, 144, 255], [210, 100, 56], "dodgerBlue"],
+ "20B2AA": [[32, 178, 170], [177, 70, 41], "lightSeaGreen"],
+ "228B22": [[34, 139, 34], [120, 61, 34], "forestGreen"],
+ "2E8B57": [[46, 139, 87], [146, 50, 36], "seaGreen"],
+ "2F4F4F": [[47, 79, 79], [180, 25, 25], "darkSlateGray"],
+ "32CD32": [[50, 205, 50], [120, 61, 50], "limeGreen"],
+ "3CB371": [[60, 179, 113], [147, 50, 47], "mediumSeaGreen"],
+ "40E0D0": [[64, 224, 208], [174, 72, 56], "turquoise"],
+ "4169E1": [[65, 105, 225], [225, 73, 57], "royalBlue"],
+ "4682B4": [[70, 130, 180], [207, 44, 49], "steelBlue"],
+ "483D8B": [[72, 61, 139], [248, 39, 39], "darkSlateBlue"],
+ "48D1CC": [[72, 209, 204], [178, 60, 55], "mediumTurquoise"],
+ "4B0082": [[75, 0, 130], [275, 100, 25], "indigo"],
+ "556B2F": [[85, 107, 47], [82, 39, 30], "darkOliveGreen"],
+ "5F9EA0": [[95, 158, 160], [182, 25, 50], "cadetBlue"],
+ "6495ED": [[100, 149, 237], [219, 79, 66], "cornflowerBlue"],
+ "66CDAA": [[102, 205, 170], [160, 51, 60], "mediumAquaMarine"],
+ "696969": [[105, 105, 105], [0, 0, 41], "dimGray"],
+ "6A5ACD": [[106, 90, 205], [248, 53, 58], "slateBlue"],
+ "6B8E23": [[107, 142, 35], [80, 60, 35], "oliveDrab"],
+ "708090": [[112, 128, 144], [210, 13, 50], "slateGray"],
+ "778899": [[119, 136, 153], [210, 14, 53], "lightSlateGray"],
+ "7B68EE": [[123, 104, 238], [249, 80, 67], "mediumSlateBlue"],
+ "7CFC00": [[124, 252, 0], [90, 100, 49], "lawnGreen"],
+ "7FFF00": [[127, 255, 0], [90, 100, 50], "chartreuse"],
+ "7FFFD4": [[127, 255, 212], [160, 100, 75], "aquamarine"],
+ "800000": [[128, 0, 0], [0, 100, 25], "maroon"],
+ "800080": [[128, 0, 128], [300, 100, 25], "purple"],
+ "808000": [[128, 128, 0], [60, 100, 25], "olive"],
+ "808080": [[128, 128, 128], [0, 0, 50], "gray"],
+ "87CEEB": [[135, 206, 235], [197, 71, 73], "skyBlue"],
+ "87CEFA": [[135, 206, 250], [203, 92, 75], "lightSkyBlue"],
+ "8A2BE2": [[138, 43, 226], [271, 76, 53], "blueViolet"],
+ "8B0000": [[139, 0, 0], [0, 100, 27], "darkRed"],
+ "8B008B": [[139, 0, 139], [300, 100, 27], "darkMagenta"],
+ "8B4513": [[139, 69, 19], [25, 76, 31], "saddleBrown"],
+ "8FBC8F": [[143, 188, 143], [120, 25, 65], "darkSeaGreen"],
+ "90EE90": [[144, 238, 144], [120, 73, 75], "lightGreen"],
+ "9370D8": [[147, 112, 219], [260, 60, 65], "mediumPurple"],
+ "9400D3": [[148, 0, 211], [282, 100, 41], "darkViolet"],
+ "98FB98": [[152, 251, 152], [120, 93, 79], "paleGreen"],
+ "9932CC": [[153, 50, 204], [280, 61, 50], "darkOrchid"],
+ "9ACD32": [[154, 205, 50], [80, 61, 50], "yellowGreen"],
+ "A0522D": [[160, 82, 45], [19, 56, 40], "sienna"],
+ "A52A2A": [[165, 42, 42], [0, 59, 41], "brown"],
+ "A9A9A9": [[169, 169, 169], [0, 0, 66], "darkGray"],
+ "ADD8E6": [[173, 216, 230], [195, 53, 79], "lightBlue"],
+ "ADFF2F": [[173, 255, 47], [84, 100, 59], "greenYellow"],
+ "AFEEEE": [[175, 238, 238], [180, 65, 81], "paleTurquoise"],
+ "B0C4DE": [[176, 196, 222], [214, 41, 78], "lightSteelBlue"],
+ "B0E0E6": [[176, 224, 230], [187, 52, 80], "powderBlue"],
+ "B22222": [[178, 34, 34], [0, 68, 42], "fireBrick"],
+ "B8860B": [[184, 134, 11], [43, 89, 38], "darkGoldenRod"],
+ "BA55D3": [[186, 85, 211], [288, 59, 58], "mediumOrchid"],
+ "BC8F8F": [[188, 143, 143], [0, 25, 65], "rosyBrown"],
+ "BDB76B": [[189, 183, 107], [56, 38, 58], "darkKhaki"],
+ "C0C0C0": [[192, 192, 192], [0, 0, 75], "silver"],
+ "C71585": [[199, 21, 133], [322, 81, 43], "mediumVioletRed"],
+ "CD5C5C": [[205, 92, 92], [0, 53, 58], "indianRed"],
+ "CD853F": [[205, 133, 63], [30, 59, 53], "peru"],
+ "D2691E": [[210, 105, 30], [25, 75, 47], "chocolate"],
+ "D2B48C": [[210, 180, 140], [34, 44, 69], "tan"],
+ "D3D3D3": [[211, 211, 211], [0, 0, 83], "lightGrey"],
+ "D87093": [[219, 112, 147], [340, 60, 65], "paleVioletRed"],
+ "D8BFD8": [[216, 191, 216], [300, 24, 80], "thistle"],
+ "DA70D6": [[218, 112, 214], [302, 59, 65], "orchid"],
+ "DAA520": [[218, 165, 32], [43, 74, 49], "goldenRod"],
+ "DC143C": [[237, 164, 61], [35, 83, 58], "crimson"],
+ "DCDCDC": [[220, 220, 220], [0, 0, 86], "gainsboro"],
+ "DDA0DD": [[221, 160, 221], [300, 47, 75], "plum"],
+ "DEB887": [[222, 184, 135], [34, 57, 70], "burlyWood"],
+ "E0FFFF": [[224, 255, 255], [180, 100, 94], "lightCyan"],
+ "E6E6FA": [[230, 230, 250], [240, 67, 94], "lavender"],
+ "E9967A": [[233, 150, 122], [15, 72, 70], "darkSalmon"],
+ "EE82EE": [[238, 130, 238], [300, 76, 72], "violet"],
+ "EEE8AA": [[238, 232, 170], [55, 67, 80], "paleGoldenRod"],
+ "F08080": [[240, 128, 128], [0, 79, 72], "lightCoral"],
+ "F0E68C": [[240, 230, 140], [54, 77, 75], "khaki"],
+ "F0F8FF": [[240, 248, 255], [208, 100, 97], "aliceBlue"],
+ "F0FFF0": [[240, 255, 240], [120, 100, 97], "honeyDew"],
+ "F0FFFF": [[240, 255, 255], [180, 100, 97], "azure"],
+ "F4A460": [[244, 164, 96], [28, 87, 67], "sandyBrown"],
+ "F5DEB3": [[245, 222, 179], [39, 77, 83], "wheat"],
+ "F5F5DC": [[245, 245, 220], [60, 56, 91], "beige"],
+ "F5F5F5": [[245, 245, 245], [0, 0, 96], "whiteSmoke"],
+ "F5FFFA": [[245, 255, 250], [150, 100, 98], "mintCream"],
+ "F8F8FF": [[248, 248, 255], [240, 100, 99], "ghostWhite"],
+ "FA8072": [[250, 128, 114], [6, 93, 71], "salmon"],
+ "FAEBD7": [[250, 235, 215], [34, 78, 91], "antiqueWhite"],
+ "FAF0E6": [[250, 240, 230], [30, 67, 94], "linen"],
+ "FAFAD2": [[250, 250, 210], [60, 80, 90], "lightGoldenRodYellow"],
+ "FDF5E6": [[253, 245, 230], [39, 85, 95], "oldLace"],
+ "FF0000": [[255, 0, 0], [0, 100, 50], "red"],
+ "FF00FF": [[255, 0, 255], [300, 100, 50], "magenta"],
+ "FF1493": [[255, 20, 147], [328, 100, 54], "deepPink"],
+ "FF4500": [[255, 69, 0], [16, 100, 50], "orangeRed"],
+ "FF6347": [[255, 99, 71], [9, 100, 64], "tomato"],
+ "FF69B4": [[255, 105, 180], [330, 100, 71], "hotPink"],
+ "FF7F50": [[255, 127, 80], [16, 100, 66], "coral"],
+ "FF8C00": [[255, 140, 0], [33, 100, 50], "darkOrange"],
+ "FFA07A": [[255, 160, 122], [17, 100, 74], "lightSalmon"],
+ "FFA500": [[255, 165, 0], [39, 100, 50], "orange"],
+ "FFB6C1": [[255, 182, 193], [351, 100, 86], "lightPink"],
+ "FFC0CB": [[255, 192, 203], [350, 100, 88], "pink"],
+ "FFD700": [[255, 215, 0], [51, 100, 50], "gold"],
+ "FFDAB9": [[255, 218, 185], [28, 100, 86], "peachPuff"],
+ "FFDEAD": [[255, 222, 173], [36, 100, 84], "navajoWhite"],
+ "FFE4B5": [[255, 228, 181], [38, 100, 85], "moccasin"],
+ "FFE4C4": [[255, 228, 196], [33, 100, 88], "bisque"],
+ "FFE4E1": [[255, 228, 225], [6, 100, 94], "mistyRose"],
+ "FFEBCD": [[255, 235, 205], [36, 100, 90], "blanchedAlmond"],
+ "FFEFD5": [[255, 239, 213], [37, 100, 92], "papayaWhip"],
+ "FFF0F5": [[255, 240, 245], [340, 100, 97], "lavenderBlush"],
+ "FFF5EE": [[255, 245, 238], [25, 100, 97], "seaShell"],
+ "FFF8DC": [[255, 248, 220], [48, 100, 93], "cornsilk"],
+ "FFFACD": [[255, 250, 205], [54, 100, 90], "lemonChiffon"],
+ "FFFAF0": [[255, 250, 240], [40, 100, 97], "floralWhite"],
+ "FFFAFA": [[255, 250, 250], [0, 100, 99], "snow"],
+ "FFFF00": [[255, 255, 0], [60, 100, 50], "yellow"],
+ "FFFFE0": [[255, 255, 224], [60, 100, 94], "lightYellow"],
+ "FFFFF0": [[255, 255, 240], [60, 100, 97], "ivory"],
+ "FFFFFF": [[255, 255, 255], [0, 100, 100], "white"]
+};
+
+// Simple Values
+WebInspector.Color.Nicknames = {
+ "aliceblue": "F0F8FF",
+ "antiquewhite": "FAEBD7",
+ "aqua": "00FFFF",
+ "aquamarine": "7FFFD4",
+ "azure": "F0FFFF",
+ "beige": "F5F5DC",
+ "bisque": "FFE4C4",
+ "black": "000000",
+ "blanchedalmond": "FFEBCD",
+ "blue": "0000FF",
+ "blueviolet": "8A2BE2",
+ "brown": "A52A2A",
+ "burlywood": "DEB887",
+ "cadetblue": "5F9EA0",
+ "chartreuse": "7FFF00",
+ "chocolate": "D2691E",
+ "coral": "FF7F50",
+ "cornflowerblue": "6495ED",
+ "cornsilk": "FFF8DC",
+ "crimson": "DC143C",
+ "cyan": "00FFFF",
+ "darkblue": "00008B",
+ "darkcyan": "008B8B",
+ "darkgoldenrod": "B8860B",
+ "darkgray": "A9A9A9",
+ "darkgreen": "006400",
+ "darkkhaki": "BDB76B",
+ "darkmagenta": "8B008B",
+ "darkolivegreen": "556B2F",
+ "darkorange": "FF8C00",
+ "darkorchid": "9932CC",
+ "darkred": "8B0000",
+ "darksalmon": "E9967A",
+ "darkseagreen": "8FBC8F",
+ "darkslateblue": "483D8B",
+ "darkslategray": "2F4F4F",
+ "darkturquoise": "00CED1",
+ "darkviolet": "9400D3",
+ "deeppink": "FF1493",
+ "deepskyblue": "00BFFF",
+ "dimgray": "696969",
+ "dodgerblue": "1E90FF",
+ "firebrick": "B22222",
+ "floralwhite": "FFFAF0",
+ "forestgreen": "228B22",
+ "fuchsia": "FF00FF",
+ "gainsboro": "DCDCDC",
+ "ghostwhite": "F8F8FF",
+ "gold": "FFD700",
+ "goldenrod": "DAA520",
+ "gray": "808080",
+ "green": "008000",
+ "greenyellow": "ADFF2F",
+ "honeydew": "F0FFF0",
+ "hotpink": "FF69B4",
+ "indianred": "CD5C5C",
+ "indigo": "4B0082",
+ "ivory": "FFFFF0",
+ "khaki": "F0E68C",
+ "lavender": "E6E6FA",
+ "lavenderblush": "FFF0F5",
+ "lawngreen": "7CFC00",
+ "lemonchiffon": "FFFACD",
+ "lightblue": "ADD8E6",
+ "lightcoral": "F08080",
+ "lightcyan": "E0FFFF",
+ "lightgoldenrodyellow": "FAFAD2",
+ "lightgreen": "90EE90",
+ "lightgrey": "D3D3D3",
+ "lightpink": "FFB6C1",
+ "lightsalmon": "FFA07A",
+ "lightseagreen": "20B2AA",
+ "lightskyblue": "87CEFA",
+ "lightslategray": "778899",
+ "lightsteelblue": "B0C4DE",
+ "lightyellow": "FFFFE0",
+ "lime": "00FF00",
+ "limegreen": "32CD32",
+ "linen": "FAF0E6",
+ "magenta": "FF00FF",
+ "maroon": "800000",
+ "mediumaquamarine": "66CDAA",
+ "mediumblue": "0000CD",
+ "mediumorchid": "BA55D3",
+ "mediumpurple": "9370D8",
+ "mediumseagreen": "3CB371",
+ "mediumslateblue": "7B68EE",
+ "mediumspringgreen": "00FA9A",
+ "mediumturquoise": "48D1CC",
+ "mediumvioletred": "C71585",
+ "midnightblue": "191970",
+ "mintcream": "F5FFFA",
+ "mistyrose": "FFE4E1",
+ "moccasin": "FFE4B5",
+ "navajowhite": "FFDEAD",
+ "navy": "000080",
+ "oldlace": "FDF5E6",
+ "olive": "808000",
+ "olivedrab": "6B8E23",
+ "orange": "FFA500",
+ "orangered": "FF4500",
+ "orchid": "DA70D6",
+ "palegoldenrod": "EEE8AA",
+ "palegreen": "98FB98",
+ "paleturquoise": "AFEEEE",
+ "palevioletred": "D87093",
+ "papayawhip": "FFEFD5",
+ "peachpuff": "FFDAB9",
+ "peru": "CD853F",
+ "pink": "FFC0CB",
+ "plum": "DDA0DD",
+ "powderblue": "B0E0E6",
+ "purple": "800080",
+ "red": "FF0000",
+ "rosybrown": "BC8F8F",
+ "royalblue": "4169E1",
+ "saddlebrown": "8B4513",
+ "salmon": "FA8072",
+ "sandybrown": "F4A460",
+ "seagreen": "2E8B57",
+ "seashell": "FFF5EE",
+ "sienna": "A0522D",
+ "silver": "C0C0C0",
+ "skyblue": "87CEEB",
+ "slateblue": "6A5ACD",
+ "slategray": "708090",
+ "snow": "FFFAFA",
+ "springgreen": "00FF7F",
+ "steelblue": "4682B4",
+ "tan": "D2B48C",
+ "teal": "008080",
+ "thistle": "D8BFD8",
+ "tomato": "FF6347",
+ "turquoise": "40E0D0",
+ "violet": "EE82EE",
+ "wheat": "F5DEB3",
+ "white": "FFFFFF",
+ "whitesmoke": "F5F5F5",
+ "yellow": "FFFF00",
+ "yellowgreen": "9ACD32"
+};
+
+// Advanced Values [rgba, hsla, nickname]
+WebInspector.Color.AdvancedNickNames = {
+ "transparent": [[0, 0, 0, 0], [0, 0, 0, 0], "transparent"],
+ "rgba(0,0,0,0)": [[0, 0, 0, 0], [0, 0, 0, 0], "transparent"],
+ "hsla(0,0,0,0)": [[0, 0, 0, 0], [0, 0, 0, 0], "transparent"],
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ConsoleView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ConsoleView.js
new file mode 100644
index 0000000..fa9a363
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ConsoleView.js
@@ -0,0 +1,970 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ConsoleView = function(drawer)
+{
+ WebInspector.View.call(this, document.getElementById("console-view"));
+
+ this.messages = [];
+ this.drawer = drawer;
+
+ this.clearButton = document.getElementById("clear-console-status-bar-item");
+ this.clearButton.title = WebInspector.UIString("Clear console log.");
+ this.clearButton.addEventListener("click", this._clearButtonClicked.bind(this), false);
+
+ this.messagesElement = document.getElementById("console-messages");
+ this.messagesElement.addEventListener("selectstart", this._messagesSelectStart.bind(this), false);
+ this.messagesElement.addEventListener("click", this._messagesClicked.bind(this), true);
+
+ this.promptElement = document.getElementById("console-prompt");
+ this.promptElement.handleKeyEvent = this._promptKeyDown.bind(this);
+ this.prompt = new WebInspector.TextPrompt(this.promptElement, this.completions.bind(this), " .=:[({;");
+
+ this.topGroup = new WebInspector.ConsoleGroup(null, 0);
+ this.messagesElement.insertBefore(this.topGroup.element, this.promptElement);
+ this.groupLevel = 0;
+ this.currentGroup = this.topGroup;
+
+ this.toggleConsoleButton = document.getElementById("console-status-bar-item");
+ this.toggleConsoleButton.title = WebInspector.UIString("Show console.");
+ this.toggleConsoleButton.addEventListener("click", this._toggleConsoleButtonClicked.bind(this), false);
+
+ var anchoredStatusBar = document.getElementById("anchored-status-bar-items");
+ anchoredStatusBar.appendChild(this.toggleConsoleButton);
+
+ // Will hold the list of filter elements
+ this.filterBarElement = document.getElementById("console-filter");
+
+ function createFilterElement(category) {
+ var categoryElement = document.createElement("li");
+ categoryElement.category = category;
+
+ categoryElement.addStyleClass(categoryElement.category);
+
+ var label = category.toString();
+ categoryElement.appendChild(document.createTextNode(label));
+
+ categoryElement.addEventListener("click", this._updateFilter.bind(this), false);
+
+ this.filterBarElement.appendChild(categoryElement);
+ return categoryElement;
+ }
+
+ this.allElement = createFilterElement.call(this, "All");
+ this.errorElement = createFilterElement.call(this, "Errors");
+ this.warningElement = createFilterElement.call(this, "Warnings");
+ this.logElement = createFilterElement.call(this, "Logs");
+
+ this.filter(this.allElement);
+}
+
+WebInspector.ConsoleView.prototype = {
+
+ _updateFilter: function(e)
+ {
+ this.filter(e.target);
+ },
+
+ filter: function(target)
+ {
+ if (target.category == "All") {
+ if (target.hasStyleClass("selected")) {
+ // We can't unselect all, so we break early here
+ return;
+ }
+
+ this.errorElement.removeStyleClass("selected");
+ this.warningElement.removeStyleClass("selected");
+ this.logElement.removeStyleClass("selected");
+
+ document.getElementById("console-messages").removeStyleClass("filter-errors");
+ document.getElementById("console-messages").removeStyleClass("filter-warnings");
+ document.getElementById("console-messages").removeStyleClass("filter-logs");
+ } else {
+ // Something other than all is being selected, so we want to unselect all
+ if (this.allElement.hasStyleClass("selected")) {
+ this.allElement.removeStyleClass("selected");
+ document.getElementById("console-messages").removeStyleClass("filter-all");
+ }
+ }
+
+ if (target.hasStyleClass("selected")) {
+ target.removeStyleClass("selected");
+ var newClass = "filter-" + target.category.toLowerCase();
+ var filterElement = document.getElementById("console-messages");
+ filterElement.removeStyleClass(newClass);
+ } else {
+ target.addStyleClass("selected");
+ var newClass = "filter-" + target.category.toLowerCase();
+ var filterElement = document.getElementById("console-messages");
+ filterElement.addStyleClass(newClass);
+ }
+ },
+
+ _toggleConsoleButtonClicked: function()
+ {
+ this.drawer.visibleView = this;
+ },
+
+ attach: function(mainElement, statusBarElement)
+ {
+ mainElement.appendChild(this.element);
+ statusBarElement.appendChild(this.clearButton);
+ statusBarElement.appendChild(this.filterBarElement);
+ },
+
+ show: function()
+ {
+ this.toggleConsoleButton.addStyleClass("toggled-on");
+ this.toggleConsoleButton.title = WebInspector.UIString("Hide console.");
+ if (!this.prompt.isCaretInsidePrompt())
+ this.prompt.moveCaretToEndOfPrompt();
+ },
+
+ afterShow: function()
+ {
+ WebInspector.currentFocusElement = this.promptElement;
+ },
+
+ hide: function()
+ {
+ this.toggleConsoleButton.removeStyleClass("toggled-on");
+ this.toggleConsoleButton.title = WebInspector.UIString("Show console.");
+ },
+
+ addMessage: function(msg)
+ {
+ if (msg instanceof WebInspector.ConsoleMessage && !(msg instanceof WebInspector.ConsoleCommandResult)) {
+ msg.totalRepeatCount = msg.repeatCount;
+ msg.repeatDelta = msg.repeatCount;
+
+ var messageRepeated = false;
+
+ if (msg.isEqual && msg.isEqual(this.previousMessage)) {
+ // Because sometimes we get a large number of repeated messages and sometimes
+ // we get them one at a time, we need to know the difference between how many
+ // repeats we used to have and how many we have now.
+ msg.repeatDelta -= this.previousMessage.totalRepeatCount;
+
+ if (!isNaN(this.repeatCountBeforeCommand))
+ msg.repeatCount -= this.repeatCountBeforeCommand;
+
+ if (!this.commandSincePreviousMessage) {
+ // Recreate the previous message element to reset the repeat count.
+ var messagesElement = this.currentGroup.messagesElement;
+ messagesElement.removeChild(messagesElement.lastChild);
+ messagesElement.appendChild(msg.toMessageElement());
+
+ messageRepeated = true;
+ }
+ } else
+ delete this.repeatCountBeforeCommand;
+
+ // Increment the error or warning count
+ switch (msg.level) {
+ case WebInspector.ConsoleMessage.MessageLevel.Warning:
+ WebInspector.warnings += msg.repeatDelta;
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Error:
+ WebInspector.errors += msg.repeatDelta;
+ break;
+ }
+
+ // Add message to the resource panel
+ if (msg.url in WebInspector.resourceURLMap) {
+ msg.resource = WebInspector.resourceURLMap[msg.url];
+ if (WebInspector.panels.resources)
+ WebInspector.panels.resources.addMessageToResource(msg.resource, msg);
+ }
+
+ this.commandSincePreviousMessage = false;
+ this.previousMessage = msg;
+
+ if (messageRepeated)
+ return;
+ } else if (msg instanceof WebInspector.ConsoleCommand) {
+ if (this.previousMessage) {
+ this.commandSincePreviousMessage = true;
+ this.repeatCountBeforeCommand = this.previousMessage.totalRepeatCount;
+ }
+ }
+
+ this.messages.push(msg);
+
+ if (msg.type === WebInspector.ConsoleMessage.MessageType.EndGroup) {
+ if (this.groupLevel < 1)
+ return;
+
+ this.groupLevel--;
+
+ this.currentGroup = this.currentGroup.parentGroup;
+ } else {
+ if (msg.type === WebInspector.ConsoleMessage.MessageType.StartGroup) {
+ this.groupLevel++;
+
+ var group = new WebInspector.ConsoleGroup(this.currentGroup, this.groupLevel);
+ this.currentGroup.messagesElement.appendChild(group.element);
+ this.currentGroup = group;
+ }
+
+ this.currentGroup.addMessage(msg);
+ }
+
+ this.promptElement.scrollIntoView(false);
+ },
+
+ clearMessages: function(clearInspectorController)
+ {
+ if (clearInspectorController)
+ InspectorController.clearMessages(false);
+ if (WebInspector.panels.resources)
+ WebInspector.panels.resources.clearMessages();
+
+ this.messages = [];
+
+ this.groupLevel = 0;
+ this.currentGroup = this.topGroup;
+ this.topGroup.messagesElement.removeChildren();
+
+ WebInspector.errors = 0;
+ WebInspector.warnings = 0;
+
+ delete this.commandSincePreviousMessage;
+ delete this.repeatCountBeforeCommand;
+ delete this.previousMessage;
+ },
+
+ completions: function(wordRange, bestMatchOnly, completionsReadyCallback)
+ {
+ // Pass less stop characters to rangeOfWord so the range will be a more complete expression.
+ const expressionStopCharacters = " =:{;";
+ var expressionRange = wordRange.startContainer.rangeOfWord(wordRange.startOffset, expressionStopCharacters, this.promptElement, "backward");
+ var expressionString = expressionRange.toString();
+ var lastIndex = expressionString.length - 1;
+
+ var dotNotation = (expressionString[lastIndex] === ".");
+ var bracketNotation = (expressionString[lastIndex] === "[");
+
+ if (dotNotation || bracketNotation)
+ expressionString = expressionString.substr(0, lastIndex);
+
+ var prefix = wordRange.toString();
+ if (!expressionString && !prefix)
+ return;
+
+ var reportCompletions = this._reportCompletions.bind(this, bestMatchOnly, completionsReadyCallback, dotNotation, bracketNotation, prefix);
+ // Collect comma separated object properties for the completion.
+
+ if (!expressionString) {
+ if (WebInspector.panels.scripts && WebInspector.panels.scripts.paused) {
+ // Evaluate into properties in scope of the selected call frame.
+ reportCompletions(WebInspector.panels.scripts.variablesInSelectedCallFrame());
+ return;
+ } else {
+ expressionString = "this";
+ }
+ }
+
+ var includeInspectorCommandLineAPI = (!dotNotation && !bracketNotation);
+ InjectedScriptAccess.getCompletions(expressionString, includeInspectorCommandLineAPI, reportCompletions);
+ },
+
+ _reportCompletions: function(bestMatchOnly, completionsReadyCallback, dotNotation, bracketNotation, prefix, result, isException) {
+ if (isException)
+ return;
+
+ if (bracketNotation) {
+ if (prefix.length && prefix[0] === "'")
+ var quoteUsed = "'";
+ else
+ var quoteUsed = "\"";
+ }
+
+ var results = [];
+ var properties = Object.sortedProperties(result);
+
+ for (var i = 0; i < properties.length; ++i) {
+ var property = properties[i];
+
+ if (dotNotation && !/^[a-zA-Z_$][a-zA-Z0-9_$]*$/.test(property))
+ continue;
+
+ if (bracketNotation) {
+ if (!/^[0-9]+$/.test(property))
+ property = quoteUsed + property.escapeCharacters(quoteUsed + "\\") + quoteUsed;
+ property += "]";
+ }
+
+ if (property.length < prefix.length)
+ continue;
+ if (property.indexOf(prefix) !== 0)
+ continue;
+
+ results.push(property);
+ if (bestMatchOnly)
+ break;
+ }
+ setTimeout(completionsReadyCallback, 0, results);
+ },
+
+ _clearButtonClicked: function()
+ {
+ this.clearMessages(true);
+ },
+
+ _messagesSelectStart: function(event)
+ {
+ if (this._selectionTimeout)
+ clearTimeout(this._selectionTimeout);
+
+ this.prompt.clearAutoComplete();
+
+ function moveBackIfOutside()
+ {
+ delete this._selectionTimeout;
+ if (!this.prompt.isCaretInsidePrompt() && window.getSelection().isCollapsed)
+ this.prompt.moveCaretToEndOfPrompt();
+ this.prompt.autoCompleteSoon();
+ }
+
+ this._selectionTimeout = setTimeout(moveBackIfOutside.bind(this), 100);
+ },
+
+ _messagesClicked: function(event)
+ {
+ var link = event.target.enclosingNodeOrSelfWithNodeName("a");
+ if (!link || !link.representedNode)
+ return;
+
+ WebInspector.updateFocusedNode(link.representedNode.id);
+ event.stopPropagation();
+ event.preventDefault();
+ },
+
+ _promptKeyDown: function(event)
+ {
+ switch (event.keyIdentifier) {
+ case "Enter":
+ this._enterKeyPressed(event);
+ return;
+ }
+
+ this.prompt.handleKeyEvent(event);
+ },
+
+ evalInInspectedWindow: function(expression, callback)
+ {
+ if (WebInspector.panels.scripts && WebInspector.panels.scripts.paused) {
+ WebInspector.panels.scripts.evaluateInSelectedCallFrame(expression, false, callback);
+ return;
+ }
+ this.doEvalInWindow(expression, callback);
+ },
+
+ doEvalInWindow: function(expression, callback)
+ {
+ if (!expression) {
+ // There is no expression, so the completion should happen against global properties.
+ expression = "this";
+ }
+
+ function evalCallback(result)
+ {
+ callback(result.value, result.isException);
+ };
+ InjectedScriptAccess.evaluate(expression, evalCallback);
+ },
+
+ _enterKeyPressed: function(event)
+ {
+ if (event.altKey)
+ return;
+
+ event.preventDefault();
+ event.stopPropagation();
+
+ this.prompt.clearAutoComplete(true);
+
+ var str = this.prompt.text;
+ if (!str.length)
+ return;
+
+ var commandMessage = new WebInspector.ConsoleCommand(str);
+ this.addMessage(commandMessage);
+
+ var self = this;
+ function printResult(result, exception)
+ {
+ self.prompt.history.push(str);
+ self.prompt.historyOffset = 0;
+ self.prompt.text = "";
+ self.addMessage(new WebInspector.ConsoleCommandResult(result, exception, commandMessage));
+ }
+ this.evalInInspectedWindow(str, printResult);
+ },
+
+ _format: function(output, forceObjectFormat)
+ {
+ var isProxy = (output != null && typeof output === "object");
+
+ if (forceObjectFormat)
+ var type = "object";
+ else
+ var type = Object.proxyType(output);
+
+ if (isProxy && type !== "object" && type !== "function" && type !== "array" && type !== "node") {
+ // Unwrap primitive value, skip decoration.
+ output = output.description;
+ type = "undecorated"
+ }
+
+ // We don't perform any special formatting on these types, so we just
+ // pass them through the simple _formatvalue function.
+ var undecoratedTypes = {
+ "undefined": 1,
+ "null": 1,
+ "boolean": 1,
+ "number": 1,
+ "undecorated": 1
+ };
+
+ var formatter;
+ if (forceObjectFormat)
+ formatter = "_formatobject";
+ else if (type in undecoratedTypes)
+ formatter = "_formatvalue";
+ else {
+ formatter = "_format" + type;
+ if (!(formatter in this)) {
+ formatter = "_formatobject";
+ type = "object";
+ }
+ }
+
+ var span = document.createElement("span");
+ span.addStyleClass("console-formatted-" + type);
+ this[formatter](output, span);
+ return span;
+ },
+
+ _formatvalue: function(val, elem)
+ {
+ elem.appendChild(document.createTextNode(val));
+ },
+
+ _formatfunction: function(func, elem)
+ {
+ elem.appendChild(document.createTextNode(func.description));
+ },
+
+ _formatdate: function(date, elem)
+ {
+ elem.appendChild(document.createTextNode(date));
+ },
+
+ _formatstring: function(str, elem)
+ {
+ elem.appendChild(document.createTextNode("\"" + str + "\""));
+ },
+
+ _formatregexp: function(re, elem)
+ {
+ var formatted = String(re.description).replace(/([\\\/])/g, "\\$1").replace(/\\(\/[gim]*)$/, "$1").substring(1);
+ elem.appendChild(document.createTextNode(formatted));
+ },
+
+ _formatarray: function(arr, elem)
+ {
+ var self = this;
+ function printResult(properties)
+ {
+ if (!properties)
+ return;
+ elem.appendChild(document.createTextNode("["));
+ for (var i = 0; i < properties.length; ++i) {
+ var property = properties[i].value;
+ elem.appendChild(self._format(property));
+ if (i < properties.length - 1)
+ elem.appendChild(document.createTextNode(", "));
+ }
+ elem.appendChild(document.createTextNode("]"));
+ }
+ InjectedScriptAccess.getProperties(arr, false, printResult);
+ },
+
+ _formatnode: function(object, elem)
+ {
+ function printNode(nodeId)
+ {
+ if (!nodeId)
+ return;
+ var treeOutline = new WebInspector.ElementsTreeOutline();
+ treeOutline.rootDOMNode = WebInspector.domAgent.nodeForId(nodeId);
+ treeOutline.element.addStyleClass("outline-disclosure");
+ if (!treeOutline.children[0].hasChildren)
+ treeOutline.element.addStyleClass("single-node");
+ elem.appendChild(treeOutline.element);
+ }
+ InjectedScriptAccess.pushNodeToFrontend(object, printNode);
+ },
+
+ _formatobject: function(obj, elem)
+ {
+ elem.appendChild(new WebInspector.ObjectPropertiesSection(obj, obj.description, null, true).element);
+ },
+
+ _formaterror: function(obj, elem)
+ {
+ var messageElement = document.createElement("span");
+ messageElement.className = "error-message";
+ messageElement.textContent = obj.name + ": " + obj.message;
+ elem.appendChild(messageElement);
+
+ if (obj.sourceURL) {
+ var urlElement = document.createElement("a");
+ urlElement.className = "webkit-html-resource-link";
+ urlElement.href = obj.sourceURL;
+ urlElement.lineNumber = obj.line;
+ urlElement.preferredPanel = "scripts";
+
+ if (obj.line > 0)
+ urlElement.textContent = WebInspector.displayNameForURL(obj.sourceURL) + ":" + obj.line;
+ else
+ urlElement.textContent = WebInspector.displayNameForURL(obj.sourceURL);
+
+ elem.appendChild(document.createTextNode(" ("));
+ elem.appendChild(urlElement);
+ elem.appendChild(document.createTextNode(")"));
+ }
+ }
+}
+
+WebInspector.ConsoleView.prototype.__proto__ = WebInspector.View.prototype;
+
+WebInspector.ConsoleMessage = function(source, type, level, line, url, groupLevel, repeatCount)
+{
+ this.source = source;
+ this.type = type;
+ this.level = level;
+ this.line = line;
+ this.url = url;
+ this.groupLevel = groupLevel;
+ this.repeatCount = repeatCount;
+ if (arguments.length > 7)
+ this.setMessageBody(Array.prototype.slice.call(arguments, 7));
+}
+
+WebInspector.ConsoleMessage.prototype = {
+ setMessageBody: function(args)
+ {
+ switch (this.type) {
+ case WebInspector.ConsoleMessage.MessageType.Trace:
+ var span = document.createElement("span");
+ span.addStyleClass("console-formatted-trace");
+ var stack = Array.prototype.slice.call(args);
+ var funcNames = stack.map(function(f) {
+ return f || WebInspector.UIString("(anonymous function)");
+ });
+ span.appendChild(document.createTextNode(funcNames.join("\n")));
+ this.formattedMessage = span;
+ break;
+ case WebInspector.ConsoleMessage.MessageType.Object:
+ this.formattedMessage = this._format(["%O", args[0]]);
+ break;
+ default:
+ this.formattedMessage = this._format(args);
+ break;
+ }
+
+ // This is used for inline message bubbles in SourceFrames, or other plain-text representations.
+ this.message = this.formattedMessage.textContent;
+ },
+
+ isErrorOrWarning: function()
+ {
+ return (this.level === WebInspector.ConsoleMessage.MessageLevel.Warning || this.level === WebInspector.ConsoleMessage.MessageLevel.Error);
+ },
+
+ _format: function(parameters)
+ {
+ var formattedResult = document.createElement("span");
+
+ if (!parameters.length)
+ return formattedResult;
+
+ function formatForConsole(obj)
+ {
+ return WebInspector.console._format(obj);
+ }
+
+ function formatAsObjectForConsole(obj)
+ {
+ return WebInspector.console._format(obj, true);
+ }
+
+ if (typeof parameters[0] === "string") {
+ var formatters = {}
+ for (var i in String.standardFormatters)
+ formatters[i] = String.standardFormatters[i];
+
+ // Firebug uses %o for formatting objects.
+ formatters.o = formatForConsole;
+ // Firebug allows both %i and %d for formatting integers.
+ formatters.i = formatters.d;
+ // Support %O to force object formating, instead of the type-based %o formatting.
+ formatters.O = formatAsObjectForConsole;
+
+ function append(a, b)
+ {
+ if (!(b instanceof Node))
+ a.appendChild(WebInspector.linkifyStringAsFragment(b.toString()));
+ else
+ a.appendChild(b);
+ return a;
+ }
+
+ var result = String.format(parameters[0], parameters.slice(1), formatters, formattedResult, append);
+ formattedResult = result.formattedResult;
+ parameters = result.unusedSubstitutions;
+ if (parameters.length)
+ formattedResult.appendChild(document.createTextNode(" "));
+ }
+
+ for (var i = 0; i < parameters.length; ++i) {
+ if (typeof parameters[i] === "string")
+ formattedResult.appendChild(WebInspector.linkifyStringAsFragment(parameters[i]));
+ else
+ formattedResult.appendChild(formatForConsole(parameters[i]));
+
+ if (i < parameters.length - 1)
+ formattedResult.appendChild(document.createTextNode(" "));
+ }
+
+ return formattedResult;
+ },
+
+ toMessageElement: function()
+ {
+ if (this.propertiesSection)
+ return this.propertiesSection.element;
+
+ var element = document.createElement("div");
+ element.message = this;
+ element.className = "console-message";
+
+ switch (this.source) {
+ case WebInspector.ConsoleMessage.MessageSource.HTML:
+ element.addStyleClass("console-html-source");
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.WML:
+ element.addStyleClass("console-wml-source");
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.XML:
+ element.addStyleClass("console-xml-source");
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.JS:
+ element.addStyleClass("console-js-source");
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.CSS:
+ element.addStyleClass("console-css-source");
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.Other:
+ element.addStyleClass("console-other-source");
+ break;
+ }
+
+ switch (this.level) {
+ case WebInspector.ConsoleMessage.MessageLevel.Tip:
+ element.addStyleClass("console-tip-level");
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Log:
+ element.addStyleClass("console-log-level");
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Debug:
+ element.addStyleClass("console-debug-level");
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Warning:
+ element.addStyleClass("console-warning-level");
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Error:
+ element.addStyleClass("console-error-level");
+ break;
+ }
+
+ if (this.type === WebInspector.ConsoleMessage.MessageType.StartGroup) {
+ element.addStyleClass("console-group-title");
+ }
+
+ if (this.elementsTreeOutline) {
+ element.addStyleClass("outline-disclosure");
+ element.appendChild(this.elementsTreeOutline.element);
+ return element;
+ }
+
+ if (this.repeatCount > 1) {
+ var messageRepeatCountElement = document.createElement("span");
+ messageRepeatCountElement.className = "bubble";
+ messageRepeatCountElement.textContent = this.repeatCount;
+
+ element.appendChild(messageRepeatCountElement);
+ element.addStyleClass("repeated-message");
+ }
+
+ if (this.url && this.url !== "undefined") {
+ var urlElement = document.createElement("a");
+ urlElement.className = "console-message-url webkit-html-resource-link";
+ urlElement.href = this.url;
+ urlElement.lineNumber = this.line;
+
+ if (this.source === WebInspector.ConsoleMessage.MessageSource.JS)
+ urlElement.preferredPanel = "scripts";
+
+ if (this.line > 0)
+ urlElement.textContent = WebInspector.displayNameForURL(this.url) + ":" + this.line;
+ else
+ urlElement.textContent = WebInspector.displayNameForURL(this.url);
+
+ element.appendChild(urlElement);
+ }
+
+ var messageTextElement = document.createElement("span");
+ messageTextElement.className = "console-message-text";
+ messageTextElement.appendChild(this.formattedMessage);
+ element.appendChild(messageTextElement);
+
+ return element;
+ },
+
+ toString: function()
+ {
+ var sourceString;
+ switch (this.source) {
+ case WebInspector.ConsoleMessage.MessageSource.HTML:
+ sourceString = "HTML";
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.WML:
+ sourceString = "WML";
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.XML:
+ sourceString = "XML";
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.JS:
+ sourceString = "JS";
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.CSS:
+ sourceString = "CSS";
+ break;
+ case WebInspector.ConsoleMessage.MessageSource.Other:
+ sourceString = "Other";
+ break;
+ }
+
+ var typeString;
+ switch (this.type) {
+ case WebInspector.ConsoleMessage.MessageType.Log:
+ typeString = "Log";
+ break;
+ case WebInspector.ConsoleMessage.MessageType.Object:
+ typeString = "Object";
+ break;
+ case WebInspector.ConsoleMessage.MessageType.Trace:
+ typeString = "Trace";
+ break;
+ case WebInspector.ConsoleMessage.MessageType.StartGroup:
+ typeString = "Start Group";
+ break;
+ case WebInspector.ConsoleMessage.MessageType.EndGroup:
+ typeString = "End Group";
+ break;
+ }
+
+ var levelString;
+ switch (this.level) {
+ case WebInspector.ConsoleMessage.MessageLevel.Tip:
+ levelString = "Tip";
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Log:
+ levelString = "Log";
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Warning:
+ levelString = "Warning";
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Debug:
+ levelString = "Debug";
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Error:
+ levelString = "Error";
+ break;
+ }
+
+ return sourceString + " " + typeString + " " + levelString + ": " + this.formattedMessage.textContent + "\n" + this.url + " line " + this.line;
+ },
+
+ isEqual: function(msg, disreguardGroup)
+ {
+ if (!msg)
+ return false;
+
+ var ret = (this.source == msg.source)
+ && (this.type == msg.type)
+ && (this.level == msg.level)
+ && (this.line == msg.line)
+ && (this.url == msg.url)
+ && (this.message == msg.message);
+
+ return (disreguardGroup ? ret : (ret && (this.groupLevel == msg.groupLevel)));
+ }
+}
+
+// Note: Keep these constants in sync with the ones in Console.h
+WebInspector.ConsoleMessage.MessageSource = {
+ HTML: 0,
+ WML: 1,
+ XML: 2,
+ JS: 3,
+ CSS: 4,
+ Other: 5
+}
+
+WebInspector.ConsoleMessage.MessageType = {
+ Log: 0,
+ Object: 1,
+ Trace: 2,
+ StartGroup: 3,
+ EndGroup: 4
+}
+
+WebInspector.ConsoleMessage.MessageLevel = {
+ Tip: 0,
+ Log: 1,
+ Warning: 2,
+ Error: 3,
+ Debug: 4
+}
+
+WebInspector.ConsoleCommand = function(command)
+{
+ this.command = command;
+}
+
+WebInspector.ConsoleCommand.prototype = {
+ toMessageElement: function()
+ {
+ var element = document.createElement("div");
+ element.command = this;
+ element.className = "console-user-command";
+
+ var commandTextElement = document.createElement("span");
+ commandTextElement.className = "console-message-text";
+ commandTextElement.textContent = this.command;
+ element.appendChild(commandTextElement);
+
+ return element;
+ }
+}
+
+WebInspector.ConsoleTextMessage = function(text, level)
+{
+ level = level || WebInspector.ConsoleMessage.MessageLevel.Log;
+ WebInspector.ConsoleMessage.call(this, WebInspector.ConsoleMessage.MessageSource.JS, WebInspector.ConsoleMessage.MessageType.Log, level, 0, null, null, 1, text);
+}
+
+WebInspector.ConsoleTextMessage.prototype.__proto__ = WebInspector.ConsoleMessage.prototype;
+
+WebInspector.ConsoleCommandResult = function(result, exception, originatingCommand)
+{
+ var level = (exception ? WebInspector.ConsoleMessage.MessageLevel.Error : WebInspector.ConsoleMessage.MessageLevel.Log);
+ var message = (exception ? String(result) : result);
+ var line = (exception ? result.line : -1);
+ var url = (exception ? result.sourceURL : null);
+
+ WebInspector.ConsoleMessage.call(this, WebInspector.ConsoleMessage.MessageSource.JS, WebInspector.ConsoleMessage.MessageType.Log, level, line, url, null, 1, message);
+
+ this.originatingCommand = originatingCommand;
+}
+
+WebInspector.ConsoleCommandResult.prototype = {
+ toMessageElement: function()
+ {
+ var element = WebInspector.ConsoleMessage.prototype.toMessageElement.call(this);
+ element.addStyleClass("console-user-command-result");
+ return element;
+ }
+}
+
+WebInspector.ConsoleCommandResult.prototype.__proto__ = WebInspector.ConsoleMessage.prototype;
+
+WebInspector.ConsoleGroup = function(parentGroup, level)
+{
+ this.parentGroup = parentGroup;
+ this.level = level;
+
+ var element = document.createElement("div");
+ element.className = "console-group";
+ element.group = this;
+ this.element = element;
+
+ var messagesElement = document.createElement("div");
+ messagesElement.className = "console-group-messages";
+ element.appendChild(messagesElement);
+ this.messagesElement = messagesElement;
+}
+
+WebInspector.ConsoleGroup.prototype = {
+ addMessage: function(msg)
+ {
+ var element = msg.toMessageElement();
+
+ if (msg.type === WebInspector.ConsoleMessage.MessageType.StartGroup) {
+ this.messagesElement.parentNode.insertBefore(element, this.messagesElement);
+ element.addEventListener("click", this._titleClicked.bind(this), true);
+ } else
+ this.messagesElement.appendChild(element);
+
+ if (element.previousSibling && msg.originatingCommand && element.previousSibling.command === msg.originatingCommand)
+ element.previousSibling.addStyleClass("console-adjacent-user-command-result");
+ },
+
+ _titleClicked: function(event)
+ {
+ var groupTitleElement = event.target.enclosingNodeOrSelfWithClass("console-group-title");
+ if (groupTitleElement) {
+ var groupElement = groupTitleElement.enclosingNodeOrSelfWithClass("console-group");
+ if (groupElement)
+ if (groupElement.hasStyleClass("collapsed"))
+ groupElement.removeStyleClass("collapsed");
+ else
+ groupElement.addStyleClass("collapsed");
+ groupTitleElement.scrollIntoViewIfNeeded(true);
+ }
+
+ event.stopPropagation();
+ event.preventDefault();
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/CookieItemsView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/CookieItemsView.js
new file mode 100644
index 0000000..f9604a4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/CookieItemsView.js
@@ -0,0 +1,266 @@
+/*
+ * Copyright (C) 2009 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.CookieItemsView = function()
+{
+ WebInspector.View.call(this);
+
+ this.element.addStyleClass("storage-view");
+ this.element.addStyleClass("table");
+
+ this.deleteButton = new WebInspector.StatusBarButton(WebInspector.UIString("Delete"), "delete-storage-status-bar-item");
+ this.deleteButton.visible = false;
+ this.deleteButton.addEventListener("click", this._deleteButtonClicked.bind(this), false);
+
+ this.refreshButton = new WebInspector.StatusBarButton(WebInspector.UIString("Refresh"), "refresh-storage-status-bar-item");
+ this.refreshButton.addEventListener("click", this._refreshButtonClicked.bind(this), false);
+}
+
+WebInspector.CookieItemsView.prototype = {
+ get statusBarItems()
+ {
+ return [this.refreshButton.element, this.deleteButton.element];
+ },
+
+ show: function(parentElement)
+ {
+ WebInspector.View.prototype.show.call(this, parentElement);
+ this.update();
+ },
+
+ hide: function()
+ {
+ WebInspector.View.prototype.hide.call(this);
+ this.deleteButton.visible = false;
+ },
+
+ update: function()
+ {
+ this.element.removeChildren();
+
+ var self = this;
+ function callback(cookies, isAdvanced) {
+ var dataGrid = (isAdvanced ? self.dataGridForCookies(cookies) : self.simpleDataGridForCookies(cookies));
+ if (dataGrid) {
+ self._dataGrid = dataGrid;
+ self.element.appendChild(dataGrid.element);
+ if (isAdvanced)
+ self.deleteButton.visible = true;
+ } else {
+ var emptyMsgElement = document.createElement("div");
+ emptyMsgElement.className = "storage-table-empty";
+ emptyMsgElement.textContent = WebInspector.UIString("This site has no cookies.");
+ self.element.appendChild(emptyMsgElement);
+ self._dataGrid = null;
+ self.deleteButton.visible = false;
+ }
+ }
+
+ WebInspector.Cookies.getCookiesAsync(callback);
+ },
+
+ dataGridForCookies: function(cookies)
+ {
+ if (!cookies.length)
+ return null;
+
+ for (var i = 0; i < cookies.length; ++i)
+ cookies[i].expires = new Date(cookies[i].expires);
+
+ var columns = { 0: {}, 1: {}, 2: {}, 3: {}, 4: {}, 5: {}, 6: {}, 7: {} };
+ columns[0].title = WebInspector.UIString("Name");
+ columns[0].width = columns[0].title.length;
+ columns[1].title = WebInspector.UIString("Value");
+ columns[1].width = columns[1].title.length;
+ columns[2].title = WebInspector.UIString("Domain");
+ columns[2].width = columns[2].title.length;
+ columns[3].title = WebInspector.UIString("Path");
+ columns[3].width = columns[3].title.length;
+ columns[4].title = WebInspector.UIString("Expires");
+ columns[4].width = columns[4].title.length;
+ columns[5].title = WebInspector.UIString("Size");
+ columns[5].width = columns[5].title.length;
+ columns[5].aligned = "right";
+ columns[6].title = WebInspector.UIString("HTTP");
+ columns[6].width = columns[6].title.length;
+ columns[6].aligned = "centered";
+ columns[7].title = WebInspector.UIString("Secure");
+ columns[7].width = columns[7].title.length;
+ columns[7].aligned = "centered";
+
+ function updateDataAndColumn(index, value) {
+ data[index] = value;
+ if (value.length > columns[index].width)
+ columns[index].width = value.length;
+ }
+
+ var data;
+ var nodes = [];
+ for (var i = 0; i < cookies.length; ++i) {
+ var cookie = cookies[i];
+ data = {};
+
+ updateDataAndColumn(0, cookie.name);
+ updateDataAndColumn(1, cookie.value);
+ updateDataAndColumn(2, cookie.domain);
+ updateDataAndColumn(3, cookie.path);
+ updateDataAndColumn(4, (cookie.session ? WebInspector.UIString("Session") : cookie.expires.toGMTString()));
+ updateDataAndColumn(5, Number.bytesToString(cookie.size, WebInspector.UIString));
+ updateDataAndColumn(6, (cookie.httpOnly ? "\u2713" : "")); // Checkmark
+ updateDataAndColumn(7, (cookie.secure ? "\u2713" : "")); // Checkmark
+
+ var node = new WebInspector.DataGridNode(data, false);
+ node.cookie = cookie;
+ node.selectable = true;
+ nodes.push(node);
+ }
+
+ var totalColumnWidths = 0;
+ for (var columnIdentifier in columns)
+ totalColumnWidths += columns[columnIdentifier].width;
+
+ // Enforce the Value column (the 2nd column) to be a max of 33%
+ // tweaking the raw total width because may massively outshadow the others
+ var valueColumnWidth = columns[1].width;
+ if (valueColumnWidth / totalColumnWidths > 0.33) {
+ totalColumnWidths -= valueColumnWidth;
+ totalColumnWidths *= 1.33;
+ columns[1].width = totalColumnWidths * 0.33;
+ }
+
+ // Calculate the percentage width for the columns.
+ const minimumPrecent = 6;
+ var recoupPercent = 0;
+ for (var columnIdentifier in columns) {
+ var width = columns[columnIdentifier].width;
+ width = Math.round((width / totalColumnWidths) * 100);
+ if (width < minimumPrecent) {
+ recoupPercent += (minimumPrecent - width);
+ width = minimumPrecent;
+ }
+ columns[columnIdentifier].width = width;
+ }
+
+ // Enforce the minimum percentage width. (need to narrow total percentage due to earlier additions)
+ while (recoupPercent > 0) {
+ for (var columnIdentifier in columns) {
+ if (columns[columnIdentifier].width > minimumPrecent) {
+ --columns[columnIdentifier].width;
+ --recoupPercent;
+ if (!recoupPercent)
+ break;
+ }
+ }
+ }
+
+ for (var columnIdentifier in columns)
+ columns[columnIdentifier].width += "%";
+
+ var dataGrid = new WebInspector.DataGrid(columns);
+ var length = nodes.length;
+ for (var i = 0; i < length; ++i)
+ dataGrid.appendChild(nodes[i]);
+ if (length > 0)
+ nodes[0].selected = true;
+
+ return dataGrid;
+ },
+
+ simpleDataGridForCookies: function(cookies)
+ {
+ if (!cookies.length)
+ return null;
+
+ var columns = {};
+ columns[0] = {};
+ columns[1] = {};
+ columns[0].title = WebInspector.UIString("Name");
+ columns[0].width = columns[0].title.length;
+ columns[1].title = WebInspector.UIString("Value");
+ columns[1].width = columns[1].title.length;
+
+ var nodes = [];
+ for (var i = 0; i < cookies.length; ++i) {
+ var cookie = cookies[i];
+ var data = {};
+
+ var name = cookie.name;
+ data[0] = name;
+ if (name.length > columns[0].width)
+ columns[0].width = name.length;
+
+ var value = cookie.value;
+ data[1] = value;
+ if (value.length > columns[1].width)
+ columns[1].width = value.length;
+
+ var node = new WebInspector.DataGridNode(data, false);
+ node.selectable = true;
+ nodes.push(node);
+ }
+
+ var totalColumnWidths = columns[0].width + columns[1].width;
+ var width = Math.round((columns[0].width * 100) / totalColumnWidths);
+ const minimumPrecent = 20;
+ if (width < minimumPrecent)
+ width = minimumPrecent;
+ if (width > 100 - minimumPrecent)
+ width = 100 - minimumPrecent;
+ columns[0].width = width;
+ columns[1].width = 100 - width;
+ columns[0].width += "%";
+ columns[1].width += "%";
+
+ var dataGrid = new WebInspector.DataGrid(columns);
+ var length = nodes.length;
+ for (var i = 0; i < length; ++i)
+ dataGrid.appendChild(nodes[i]);
+ if (length > 0)
+ nodes[0].selected = true;
+
+ return dataGrid;
+ },
+
+ _deleteButtonClicked: function(event)
+ {
+ if (!this._dataGrid)
+ return;
+
+ var cookie = this._dataGrid.selectedNode.cookie;
+ InspectorController.deleteCookie(cookie.name);
+ this.update();
+ },
+
+ _refreshButtonClicked: function(event)
+ {
+ this.update();
+ }
+}
+
+WebInspector.CookieItemsView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMAgent.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMAgent.js
new file mode 100644
index 0000000..47c8041
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMAgent.js
@@ -0,0 +1,650 @@
+/*
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.DOMNode = function(doc, payload) {
+ this.ownerDocument = doc;
+
+ this.id = payload.id;
+ this.nodeType = payload.nodeType;
+ this.nodeName = payload.nodeName;
+ this._nodeValue = payload.nodeValue;
+ this.textContent = this.nodeValue;
+
+ this.attributes = [];
+ this._attributesMap = {};
+ if (payload.attributes)
+ this._setAttributesPayload(payload.attributes);
+
+ this._childNodeCount = payload.childNodeCount;
+ this.children = null;
+
+ this.nextSibling = null;
+ this.prevSibling = null;
+ this.firstChild = null;
+ this.lastChild = null;
+ this.parentNode = null;
+
+ if (payload.children)
+ this._setChildrenPayload(payload.children);
+
+ this._computedStyle = null;
+ this.style = null;
+ this._matchedCSSRules = [];
+
+ if (this.nodeType == Node.ELEMENT_NODE) {
+ if (this.nodeName == "HTML")
+ this.ownerDocument.documentElement = this;
+ if (this.nodeName == "BODY")
+ this.ownerDocument.body = this;
+ }
+}
+
+WebInspector.DOMNode.prototype = {
+ hasAttributes: function()
+ {
+ return this.attributes.length > 0;
+ },
+
+ hasChildNodes: function() {
+ return this._childNodeCount > 0;
+ },
+
+ get nodeValue() {
+ return this._nodeValue;
+ },
+
+ set nodeValue(value) {
+ if (this.nodeType != Node.TEXT_NODE)
+ return;
+ var self = this;
+ var callback = function()
+ {
+ self._nodeValue = value;
+ self.textContent = value;
+ };
+ this.ownerDocument._domAgent.setTextNodeValueAsync(this, value, callback);
+ },
+
+ getAttribute: function(name)
+ {
+ var attr = this._attributesMap[name];
+ return attr ? attr.value : undefined;
+ },
+
+ setAttribute: function(name, value)
+ {
+ var self = this;
+ var callback = function()
+ {
+ var attr = self._attributesMap[name];
+ if (attr)
+ attr.value = value;
+ else
+ attr = self._addAttribute(name, value);
+ };
+ this.ownerDocument._domAgent.setAttributeAsync(this, name, value, callback);
+ },
+
+ removeAttribute: function(name)
+ {
+ var self = this;
+ var callback = function()
+ {
+ delete self._attributesMap[name];
+ for (var i = 0; i < self.attributes.length; ++i) {
+ if (self.attributes[i].name == name) {
+ self.attributes.splice(i, 1);
+ break;
+ }
+ }
+ };
+ this.ownerDocument._domAgent.removeAttributeAsync(this, name, callback);
+ },
+
+ _setAttributesPayload: function(attrs)
+ {
+ for (var i = 0; i < attrs.length; i += 2)
+ this._addAttribute(attrs[i], attrs[i + 1]);
+ },
+
+ _insertChild: function(prev, payload)
+ {
+ var node = new WebInspector.DOMNode(this.ownerDocument, payload);
+ if (!prev)
+ // First node
+ this.children = [ node ];
+ else
+ this.children.splice(this.children.indexOf(prev) + 1, 0, node);
+ this._renumber();
+ return node;
+ },
+
+ removeChild_: function(node)
+ {
+ this.children.splice(this.children.indexOf(node), 1);
+ node.parentNode = null;
+ this._renumber();
+ },
+
+ _setChildrenPayload: function(payloads)
+ {
+ this.children = [];
+ for (var i = 0; i < payloads.length; ++i) {
+ var payload = payloads[i];
+ var node = new WebInspector.DOMNode(this.ownerDocument, payload);
+ this.children.push(node);
+ }
+ this._renumber();
+ },
+
+ _renumber: function()
+ {
+ this._childNodeCount = this.children.length;
+ if (this._childNodeCount == 0) {
+ this.firstChild = null;
+ this.lastChild = null;
+ return;
+ }
+ this.firstChild = this.children[0];
+ this.lastChild = this.children[this._childNodeCount - 1];
+ for (var i = 0; i < this._childNodeCount; ++i) {
+ var child = this.children[i];
+ child.nextSibling = i + 1 < this._childNodeCount ? this.children[i + 1] : null;
+ child.prevSibling = i - 1 >= 0 ? this.children[i - 1] : null;
+ child.parentNode = this;
+ }
+ },
+
+ _addAttribute: function(name, value)
+ {
+ var attr = {
+ "name": name,
+ "value": value,
+ "_node": this
+ };
+ this._attributesMap[name] = attr;
+ this.attributes.push(attr);
+ },
+
+ _setStyles: function(computedStyle, inlineStyle, styleAttributes, matchedCSSRules)
+ {
+ this._computedStyle = new WebInspector.CSSStyleDeclaration(computedStyle);
+ this.style = new WebInspector.CSSStyleDeclaration(inlineStyle);
+
+ for (var name in styleAttributes) {
+ if (this._attributesMap[name])
+ this._attributesMap[name].style = new WebInspector.CSSStyleDeclaration(styleAttributes[name]);
+ }
+
+ this._matchedCSSRules = [];
+ for (var i = 0; i < matchedCSSRules.length; i++)
+ this._matchedCSSRules.push(WebInspector.CSSStyleDeclaration.parseRule(matchedCSSRules[i]));
+ },
+
+ _clearStyles: function()
+ {
+ this.computedStyle = null;
+ this.style = null;
+ for (var name in this._attributesMap)
+ this._attributesMap[name].style = null;
+ this._matchedCSSRules = null;
+ }
+}
+
+WebInspector.DOMDocument = function(domAgent, defaultView, payload)
+{
+ WebInspector.DOMNode.call(this, this, payload);
+ this._listeners = {};
+ this._domAgent = domAgent;
+ this.defaultView = defaultView;
+}
+
+WebInspector.DOMDocument.prototype = {
+
+ addEventListener: function(name, callback)
+ {
+ var listeners = this._listeners[name];
+ if (!listeners) {
+ listeners = [];
+ this._listeners[name] = listeners;
+ }
+ listeners.push(callback);
+ },
+
+ removeEventListener: function(name, callback)
+ {
+ var listeners = this._listeners[name];
+ if (!listeners)
+ return;
+
+ var index = listeners.indexOf(callback);
+ if (index != -1)
+ listeners.splice(index, 1);
+ },
+
+ _fireDomEvent: function(name, event)
+ {
+ var listeners = this._listeners[name];
+ if (!listeners)
+ return;
+
+ for (var i = 0; i < listeners.length; ++i) {
+ var listener = listeners[i];
+ listener.call(this, event);
+ }
+ }
+}
+
+WebInspector.DOMDocument.prototype.__proto__ = WebInspector.DOMNode.prototype;
+
+
+WebInspector.DOMWindow = function(domAgent)
+{
+ this._domAgent = domAgent;
+}
+
+WebInspector.DOMWindow.prototype = {
+ get document()
+ {
+ return this._domAgent.document;
+ },
+
+ get Node()
+ {
+ return WebInspector.DOMNode;
+ },
+
+ get Element()
+ {
+ return WebInspector.DOMNode;
+ },
+
+ Object: function()
+ {
+ },
+
+ getComputedStyle: function(node)
+ {
+ return node._computedStyle;
+ },
+
+ getMatchedCSSRules: function(node, pseudoElement, authorOnly)
+ {
+ return node._matchedCSSRules;
+ }
+}
+
+WebInspector.DOMAgent = function() {
+ this._window = new WebInspector.DOMWindow(this);
+ this._idToDOMNode = null;
+ this.document = null;
+
+ // TODO: update ElementsPanel to not track embedded iframes - it is already being handled
+ // in the agent backend.
+
+ // Whitespace is ignored in InspectorDOMAgent already -> no need to filter.
+ // TODO: Either remove all of its usages or push value into the agent backend.
+ Preferences.ignoreWhitespace = false;
+}
+
+WebInspector.DOMAgent.prototype = {
+ get domWindow()
+ {
+ return this._window;
+ },
+
+ getChildNodesAsync: function(parent, callback)
+ {
+ var children = parent.children;
+ if (children) {
+ callback(children);
+ return;
+ }
+ function mycallback() {
+ callback(parent.children);
+ }
+ var callId = WebInspector.Callback.wrap(mycallback);
+ InspectorController.getChildNodes(callId, parent.id);
+ },
+
+ setAttributeAsync: function(node, name, value, callback)
+ {
+ var mycallback = this._didApplyDomChange.bind(this, node, callback);
+ InspectorController.setAttribute(WebInspector.Callback.wrap(mycallback), node.id, name, value);
+ },
+
+ removeAttributeAsync: function(node, name, callback)
+ {
+ var mycallback = this._didApplyDomChange.bind(this, node, callback);
+ InspectorController.removeAttribute(WebInspector.Callback.wrap(mycallback), node.id, name);
+ },
+
+ setTextNodeValueAsync: function(node, text, callback)
+ {
+ var mycallback = this._didApplyDomChange.bind(this, node, callback);
+ InspectorController.setTextNodeValue(WebInspector.Callback.wrap(mycallback), node.id, text);
+ },
+
+ _didApplyDomChange: function(node, callback, success)
+ {
+ if (!success)
+ return;
+ callback();
+ // TODO(pfeldman): Fix this hack.
+ var elem = WebInspector.panels.elements.treeOutline.findTreeElement(node);
+ if (elem) {
+ elem._updateTitle();
+ }
+ },
+
+ _attributesUpdated: function(nodeId, attrsArray)
+ {
+ var node = this._idToDOMNode[nodeId];
+ node._setAttributesPayload(attrsArray);
+ },
+
+ nodeForId: function(nodeId) {
+ return this._idToDOMNode[nodeId];
+ },
+
+ _setDocument: function(payload)
+ {
+ this.document = new WebInspector.DOMDocument(this, this._window, payload);
+ this._idToDOMNode = {};
+ this._idToDOMNode[payload.id] = this.document;
+ this._bindNodes(this.document.children);
+ WebInspector.panels.elements.reset();
+ },
+
+ _setDetachedRoot: function(payload)
+ {
+ var root = new WebInspector.DOMNode(this.document, payload);
+ this._idToDOMNode[payload.id] = root;
+ },
+
+ _setChildNodes: function(parentId, payloads)
+ {
+ var parent = this._idToDOMNode[parentId];
+ parent._setChildrenPayload(payloads);
+ this._bindNodes(parent.children);
+ },
+
+ _bindNodes: function(children)
+ {
+ for (var i = 0; i < children.length; ++i) {
+ var child = children[i];
+ this._idToDOMNode[child.id] = child;
+ if (child.children)
+ this._bindNodes(child.children);
+ }
+ },
+
+ _childNodeCountUpdated: function(nodeId, newValue)
+ {
+ var node = this._idToDOMNode[nodeId];
+ node._childNodeCount = newValue;
+ var outline = WebInspector.panels.elements.treeOutline;
+ var treeElement = outline.findTreeElement(node);
+ if (treeElement) {
+ treeElement.hasChildren = newValue;
+ treeElement.whitespaceIgnored = Preferences.ignoreWhitespace;
+ }
+ },
+
+ _childNodeInserted: function(parentId, prevId, payload)
+ {
+ var parent = this._idToDOMNode[parentId];
+ var prev = this._idToDOMNode[prevId];
+ var node = parent._insertChild(prev, payload);
+ this._idToDOMNode[node.id] = node;
+ var event = { target : node, relatedNode : parent };
+ this.document._fireDomEvent("DOMNodeInserted", event);
+ },
+
+ _childNodeRemoved: function(parentId, nodeId)
+ {
+ var parent = this._idToDOMNode[parentId];
+ var node = this._idToDOMNode[nodeId];
+ parent.removeChild_(node);
+ var event = { target : node, relatedNode : parent };
+ this.document._fireDomEvent("DOMNodeRemoved", event);
+ delete this._idToDOMNode[nodeId];
+ }
+}
+
+WebInspector.Cookies = {}
+
+WebInspector.Cookies.getCookiesAsync = function(callback)
+{
+ function mycallback(cookies, cookiesString) {
+ if (cookiesString)
+ callback(WebInspector.Cookies.buildCookiesFromString(cookiesString), false);
+ else
+ callback(cookies, true);
+ }
+ var callId = WebInspector.Callback.wrap(mycallback);
+ InspectorController.getCookies(callId);
+}
+
+WebInspector.Cookies.buildCookiesFromString = function(rawCookieString)
+{
+ var rawCookies = rawCookieString.split(/;\s*/);
+ var cookies = [];
+
+ if (!(/^\s*$/.test(rawCookieString))) {
+ for (var i = 0; i < rawCookies.length; ++i) {
+ var cookie = rawCookies[i];
+ var delimIndex = cookie.indexOf("=");
+ var name = cookie.substring(0, delimIndex);
+ var value = cookie.substring(delimIndex + 1);
+ var size = name.length + value.length;
+ cookies.push({ name: name, value: value, size: size });
+ }
+ }
+
+ return cookies;
+}
+
+WebInspector.CSSStyleDeclaration = function(payload)
+{
+ this.id = payload.id;
+ this.width = payload.width;
+ this.height = payload.height;
+ this.__disabledProperties = payload.__disabledProperties;
+ this.__disabledPropertyValues = payload.__disabledPropertyValues;
+ this.__disabledPropertyPriorities = payload.__disabledPropertyPriorities;
+ this.uniqueStyleProperties = payload.uniqueStyleProperties;
+ this._shorthandValues = payload.shorthandValues;
+ this._propertyMap = {};
+ this._longhandProperties = {};
+ this.length = payload.properties.length;
+
+ for (var i = 0; i < this.length; ++i) {
+ var property = payload.properties[i];
+ var name = property.name;
+ this[i] = name;
+ this._propertyMap[name] = property;
+ }
+
+ // Index longhand properties.
+ for (var i = 0; i < this.uniqueStyleProperties.length; ++i) {
+ var name = this.uniqueStyleProperties[i];
+ var property = this._propertyMap[name];
+ if (property.shorthand) {
+ var longhands = this._longhandProperties[property.shorthand];
+ if (!longhands) {
+ longhands = [];
+ this._longhandProperties[property.shorthand] = longhands;
+ }
+ longhands.push(name);
+ }
+ }
+}
+
+WebInspector.CSSStyleDeclaration.parseStyle = function(payload)
+{
+ return new WebInspector.CSSStyleDeclaration(payload);
+}
+
+WebInspector.CSSStyleDeclaration.parseRule = function(payload)
+{
+ var rule = {};
+ rule.id = payload.id;
+ rule.selectorText = payload.selectorText;
+ rule.style = new WebInspector.CSSStyleDeclaration(payload.style);
+ rule.style.parentRule = rule;
+ rule.isUserAgent = payload.isUserAgent;
+ rule.isUser = payload.isUser;
+ rule.isViaInspector = payload.isViaInspector;
+ if (payload.parentStyleSheet)
+ rule.parentStyleSheet = { href: payload.parentStyleSheet.href };
+
+ return rule;
+}
+
+WebInspector.CSSStyleDeclaration.prototype = {
+ getPropertyValue: function(name)
+ {
+ var property = this._propertyMap[name];
+ return property ? property.value : "";
+ },
+
+ getPropertyPriority: function(name)
+ {
+ var property = this._propertyMap[name];
+ return property ? property.priority : "";
+ },
+
+ getPropertyShorthand: function(name)
+ {
+ var property = this._propertyMap[name];
+ return property ? property.shorthand : "";
+ },
+
+ isPropertyImplicit: function(name)
+ {
+ var property = this._propertyMap[name];
+ return property ? property.implicit : "";
+ },
+
+ styleTextWithShorthands: function()
+ {
+ var cssText = "";
+ var foundProperties = {};
+ for (var i = 0; i < this.length; ++i) {
+ var individualProperty = this[i];
+ var shorthandProperty = this.getPropertyShorthand(individualProperty);
+ var propertyName = (shorthandProperty || individualProperty);
+
+ if (propertyName in foundProperties)
+ continue;
+
+ if (shorthandProperty) {
+ var value = this.getPropertyValue(shorthandProperty);
+ var priority = this.getShorthandPriority(shorthandProperty);
+ } else {
+ var value = this.getPropertyValue(individualProperty);
+ var priority = this.getPropertyPriority(individualProperty);
+ }
+
+ foundProperties[propertyName] = true;
+
+ cssText += propertyName + ": " + value;
+ if (priority)
+ cssText += " !" + priority;
+ cssText += "; ";
+ }
+
+ return cssText;
+ },
+
+ getLonghandProperties: function(name)
+ {
+ return this._longhandProperties[name] || [];
+ },
+
+ getShorthandValue: function(shorthandProperty)
+ {
+ return this._shorthandValues[shorthandProperty];
+ },
+
+ getShorthandPriority: function(shorthandProperty)
+ {
+ var priority = this.getPropertyPriority(shorthandProperty);
+ if (priority)
+ return priority;
+
+ var longhands = this._longhandProperties[shorthandProperty];
+ return longhands ? this.getPropertyPriority(longhands[0]) : null;
+ }
+}
+
+WebInspector.attributesUpdated = function()
+{
+ this.domAgent._attributesUpdated.apply(this.domAgent, arguments);
+}
+
+WebInspector.setDocument = function()
+{
+ this.domAgent._setDocument.apply(this.domAgent, arguments);
+}
+
+WebInspector.setDetachedRoot = function()
+{
+ this.domAgent._setDetachedRoot.apply(this.domAgent, arguments);
+}
+
+WebInspector.setChildNodes = function()
+{
+ this.domAgent._setChildNodes.apply(this.domAgent, arguments);
+}
+
+WebInspector.childNodeCountUpdated = function()
+{
+ this.domAgent._childNodeCountUpdated.apply(this.domAgent, arguments);
+}
+
+WebInspector.childNodeInserted = function()
+{
+ this.domAgent._childNodeInserted.apply(this.domAgent, arguments);
+}
+
+WebInspector.childNodeRemoved = function()
+{
+ this.domAgent._childNodeRemoved.apply(this.domAgent, arguments);
+}
+
+WebInspector.didGetCookies = WebInspector.Callback.processCallback;
+WebInspector.didGetChildNodes = WebInspector.Callback.processCallback;
+WebInspector.didPerformSearch = WebInspector.Callback.processCallback;
+WebInspector.didApplyDomChange = WebInspector.Callback.processCallback;
+WebInspector.didRemoveAttribute = WebInspector.Callback.processCallback;
+WebInspector.didSetTextNodeValue = WebInspector.Callback.processCallback;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorage.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorage.js
new file mode 100644
index 0000000..5207b69
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorage.js
@@ -0,0 +1,72 @@
+/*
+ * Copyright (C) 2008 Nokia Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.DOMStorage = function(domStorage, domain, isLocalStorage)
+{
+ this.domStorage = domStorage;
+ this.domain = domain;
+ this.isLocalStorage = isLocalStorage;
+}
+
+WebInspector.DOMStorage.prototype = {
+ get domStorage()
+ {
+ return this._domStorage;
+ },
+
+ set domStorage(x)
+ {
+ if (this._domStorage === x)
+ return;
+ this._domStorage = x;
+ },
+
+ get domain()
+ {
+ return this._domain;
+ },
+
+ set domain(x)
+ {
+ if (this._domain === x)
+ return;
+ this._domain = x;
+ },
+
+ get isLocalStorage()
+ {
+ return this._isLocalStorage;
+ },
+
+ set isLocalStorage(x)
+ {
+ if (this._isLocalStorage === x)
+ return;
+ this._isLocalStorage = x;
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageDataGrid.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageDataGrid.js
new file mode 100644
index 0000000..efdd090
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageDataGrid.js
@@ -0,0 +1,165 @@
+/*
+ * Copyright (C) 2009 Nokia Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.DOMStorageDataGrid = function(columns)
+{
+ WebInspector.DataGrid.call(this, columns);
+ this.dataTableBody.addEventListener("dblclick", this._ondblclick.bind(this), false);
+}
+
+WebInspector.DOMStorageDataGrid.prototype = {
+ _ondblclick: function(event)
+ {
+ if (this._editing)
+ return;
+ if (this._editingNode)
+ return;
+ this._startEditing(event);
+ },
+
+ _startEditingColumnOfDataGridNode: function(node, column)
+ {
+ this._editing = true;
+ this._editingNode = node;
+ this._editingNode.select();
+ WebInspector.panels.storage._unregisterStorageEventListener();
+
+ var element = this._editingNode._element.children[column];
+ WebInspector.startEditing(element, this._editingCommitted.bind(this), this._editingCancelled.bind(this), element.textContent);
+ window.getSelection().setBaseAndExtent(element, 0, element, 1);
+ },
+
+ _startEditing: function(event)
+ {
+ var element = event.target.enclosingNodeOrSelfWithNodeName("td");
+ if (!element)
+ return;
+
+ this._editingNode = this.dataGridNodeFromEvent(event);
+ if (!this._editingNode) {
+ if (!this.creationNode)
+ return;
+ this._editingNode = this.creationNode;
+ }
+
+ // Force editing the "Key" column when editing the creation node
+ if (this._editingNode.isCreationNode)
+ return this._startEditingColumnOfDataGridNode(this._editingNode, 0);
+
+ this._editing = true;
+ WebInspector.panels.storage._unregisterStorageEventListener();
+ WebInspector.startEditing(element, this._editingCommitted.bind(this), this._editingCancelled.bind(this), element.textContent);
+ window.getSelection().setBaseAndExtent(element, 0, element, 1);
+ },
+
+ _editingCommitted: function(element, newText, oldText, context, moveDirection)
+ {
+ var columnIdentifier = (element.hasStyleClass("0-column") ? 0 : 1);
+ var textBeforeEditing = this._editingNode.data[columnIdentifier];
+ var currentEditingNode = this._editingNode;
+
+ function moveToNextIfNeeded(wasChange) {
+ if (!moveDirection)
+ return;
+
+ if (moveDirection === "forward") {
+ if (currentEditingNode.isCreationNode && columnIdentifier === 0 && !wasChange)
+ return;
+
+ if (columnIdentifier === 0)
+ return this._startEditingColumnOfDataGridNode(currentEditingNode, 1);
+
+ var nextDataGridNode = currentEditingNode.traverseNextNode(true, null, true);
+ if (nextDataGridNode)
+ return this._startEditingColumnOfDataGridNode(nextDataGridNode, 0);
+ if (currentEditingNode.isCreationNode && wasChange) {
+ addCreationNode(false);
+ return this._startEditingColumnOfDataGridNode(this.creationNode, 0);
+ }
+ return;
+ }
+
+ if (moveDirection === "backward") {
+ if (columnIdentifier === 1)
+ return this._startEditingColumnOfDataGridNode(currentEditingNode, 0);
+ var nextDataGridNode = currentEditingNode.traversePreviousNode(true, null, true);
+
+ if (nextDataGridNode)
+ return this._startEditingColumnOfDataGridNode(nextDataGridNode, 1);
+ return;
+ }
+ }
+
+ if (textBeforeEditing == newText) {
+ this._editingCancelled(element);
+ moveToNextIfNeeded.call(this, false);
+ return;
+ }
+
+ var domStorage = WebInspector.panels.storage.visibleView.domStorage.domStorage;
+ if (domStorage) {
+ if (columnIdentifier == 0) {
+ if (domStorage.getItem(newText) != null) {
+ element.textContent = this._editingNode.data[0];
+ this._editingCancelled(element);
+ moveToNextIfNeeded.call(this, false);
+ return;
+ }
+ domStorage.removeItem(this._editingNode.data[0]);
+ domStorage.setItem(newText, this._editingNode.data[1]);
+ this._editingNode.data[0] = newText;
+ } else {
+ domStorage.setItem(this._editingNode.data[0], newText);
+ this._editingNode.data[1] = newText;
+ }
+ }
+
+ if (this._editingNode.isCreationNode)
+ this.addCreationNode(false);
+
+ this._editingCancelled(element);
+ moveToNextIfNeeded.call(this, true);
+ },
+
+ _editingCancelled: function(element, context)
+ {
+ delete this._editing;
+ this._editingNode = null;
+ WebInspector.panels.storage._registerStorageEventListener();
+ },
+
+ deleteSelectedRow: function()
+ {
+ var node = this.selectedNode;
+ if (this.selectedNode.isCreationNode)
+ return;
+
+ var domStorage = WebInspector.panels.storage.visibleView.domStorage.domStorage;
+ if (node && domStorage)
+ domStorage.removeItem(node.data[0]);
+ }
+}
+
+WebInspector.DOMStorageDataGrid.prototype.__proto__ = WebInspector.DataGrid.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageItemsView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageItemsView.js
new file mode 100644
index 0000000..8617d60
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DOMStorageItemsView.js
@@ -0,0 +1,112 @@
+/*
+ * Copyright (C) 2008 Nokia Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.DOMStorageItemsView = function(domStorage)
+{
+ WebInspector.View.call(this);
+
+ this.domStorage = domStorage;
+
+ this.element.addStyleClass("storage-view");
+ this.element.addStyleClass("table");
+
+ this.deleteButton = new WebInspector.StatusBarButton(WebInspector.UIString("Delete"), "delete-storage-status-bar-item");
+ this.deleteButton.visible = false;
+ this.deleteButton.addEventListener("click", this._deleteButtonClicked.bind(this), false);
+
+ this.refreshButton = new WebInspector.StatusBarButton(WebInspector.UIString("Refresh"), "refresh-storage-status-bar-item");
+ this.refreshButton.addEventListener("click", this._refreshButtonClicked.bind(this), false);
+}
+
+WebInspector.DOMStorageItemsView.prototype = {
+ get statusBarItems()
+ {
+ return [this.refreshButton.element, this.deleteButton.element];
+ },
+
+ show: function(parentElement)
+ {
+ WebInspector.View.prototype.show.call(this, parentElement);
+ this.update();
+ },
+
+ hide: function()
+ {
+ WebInspector.View.prototype.hide.call(this);
+ this.deleteButton.visible = false;
+ },
+
+ update: function()
+ {
+ this.element.removeChildren();
+ var hasDOMStorage = this.domStorage;
+ if (hasDOMStorage)
+ hasDOMStorage = this.domStorage.domStorage;
+
+ if (hasDOMStorage) {
+ var dataGrid = WebInspector.panels.storage.dataGridForDOMStorage(this.domStorage.domStorage);
+ if (!dataGrid)
+ hasDOMStorage = 0;
+ else {
+ this._dataGrid = dataGrid;
+ this.element.appendChild(dataGrid.element);
+ this._dataGrid.updateWidths();
+ this.deleteButton.visible = true;
+ }
+ }
+
+ if (!hasDOMStorage) {
+ var emptyMsgElement = document.createElement("div");
+ emptyMsgElement.className = "storage-table-empty";
+ if (this.domStorage)
+ emptyMsgElement.textContent = WebInspector.UIString("This storage is empty.");
+ this.element.appendChild(emptyMsgElement);
+ this._dataGrid = null;
+ this.deleteButton.visible = false;
+ }
+ },
+
+ resize: function()
+ {
+ if (this._dataGrid)
+ this._dataGrid.updateWidths();
+ },
+
+ _deleteButtonClicked: function(event)
+ {
+ if (this._dataGrid) {
+ this._dataGrid.deleteSelectedRow();
+
+ this.show();
+ }
+ },
+
+ _refreshButtonClicked: function(event)
+ {
+ this.update();
+ }
+}
+
+WebInspector.DOMStorageItemsView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DataGrid.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DataGrid.js
new file mode 100644
index 0000000..ce61548
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DataGrid.js
@@ -0,0 +1,1041 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.DataGrid = function(columns)
+{
+ this.element = document.createElement("div");
+ this.element.className = "data-grid";
+ this.element.tabIndex = 0;
+ this.element.addEventListener("keydown", this._keyDown.bind(this), false);
+
+ this._headerTable = document.createElement("table");
+ this._headerTable.className = "header";
+
+ this._dataTable = document.createElement("table");
+ this._dataTable.className = "data";
+
+ this._dataTable.addEventListener("mousedown", this._mouseDownInDataTable.bind(this), true);
+ this._dataTable.addEventListener("click", this._clickInDataTable.bind(this), true);
+
+ this.aligned = {};
+
+ var scrollContainer = document.createElement("div");
+ scrollContainer.className = "data-container";
+ scrollContainer.appendChild(this._dataTable);
+
+ this.element.appendChild(this._headerTable);
+ this.element.appendChild(scrollContainer);
+
+ var headerRow = document.createElement("tr");
+ var columnGroup = document.createElement("colgroup");
+ var columnCount = 0;
+
+ for (var columnIdentifier in columns) {
+ var column = columns[columnIdentifier];
+ if (column.disclosure)
+ this.disclosureColumnIdentifier = columnIdentifier;
+
+ var col = document.createElement("col");
+ if (column.width)
+ col.style.width = column.width;
+ columnGroup.appendChild(col);
+
+ var cell = document.createElement("th");
+ cell.className = columnIdentifier + "-column";
+ cell.columnIdentifier = columnIdentifier;
+
+ var div = document.createElement("div");
+ div.textContent = column.title;
+ cell.appendChild(div);
+
+ if (column.sort) {
+ cell.addStyleClass("sort-" + column.sort);
+ this._sortColumnCell = cell;
+ }
+
+ if (column.sortable) {
+ cell.addEventListener("click", this._clickInHeaderCell.bind(this), false);
+ cell.addStyleClass("sortable");
+ }
+
+ if (column.aligned) {
+ cell.addStyleClass(column.aligned);
+ this.aligned[columnIdentifier] = column.aligned;
+ }
+
+ headerRow.appendChild(cell);
+
+ ++columnCount;
+ }
+
+ columnGroup.span = columnCount;
+
+ var cell = document.createElement("th");
+ cell.className = "corner";
+ headerRow.appendChild(cell);
+
+ this._headerTableColumnGroup = columnGroup;
+ this._headerTable.appendChild(this._headerTableColumnGroup);
+ this.headerTableBody.appendChild(headerRow);
+
+ var fillerRow = document.createElement("tr");
+ fillerRow.className = "filler";
+
+ for (var i = 0; i < columnCount; ++i) {
+ var cell = document.createElement("td");
+ fillerRow.appendChild(cell);
+ }
+
+ this._dataTableColumnGroup = columnGroup.cloneNode(true);
+ this._dataTable.appendChild(this._dataTableColumnGroup);
+ this.dataTableBody.appendChild(fillerRow);
+
+ this.columns = columns || {};
+ this.children = [];
+ this.selectedNode = null;
+ this.expandNodesWhenArrowing = false;
+ this.root = true;
+ this.hasChildren = false;
+ this.expanded = true;
+ this.revealed = true;
+ this.selected = false;
+ this.dataGrid = this;
+ this.indentWidth = 15;
+ this.resizers = [];
+ this.columnWidthsInitialized = false;
+}
+
+WebInspector.DataGrid.prototype = {
+ get sortColumnIdentifier()
+ {
+ if (!this._sortColumnCell)
+ return null;
+ return this._sortColumnCell.columnIdentifier;
+ },
+
+ get sortOrder()
+ {
+ if (!this._sortColumnCell || this._sortColumnCell.hasStyleClass("sort-ascending"))
+ return "ascending";
+ if (this._sortColumnCell.hasStyleClass("sort-descending"))
+ return "descending";
+ return null;
+ },
+
+ get headerTableBody()
+ {
+ if ("_headerTableBody" in this)
+ return this._headerTableBody;
+
+ this._headerTableBody = this._headerTable.getElementsByTagName("tbody")[0];
+ if (!this._headerTableBody) {
+ this._headerTableBody = this.element.ownerDocument.createElement("tbody");
+ this._headerTable.insertBefore(this._headerTableBody, this._headerTable.tFoot);
+ }
+
+ return this._headerTableBody;
+ },
+
+ get dataTableBody()
+ {
+ if ("_dataTableBody" in this)
+ return this._dataTableBody;
+
+ this._dataTableBody = this._dataTable.getElementsByTagName("tbody")[0];
+ if (!this._dataTableBody) {
+ this._dataTableBody = this.element.ownerDocument.createElement("tbody");
+ this._dataTable.insertBefore(this._dataTableBody, this._dataTable.tFoot);
+ }
+
+ return this._dataTableBody;
+ },
+
+ // Updates the widths of the table, including the positions of the column
+ // resizers.
+ //
+ // IMPORTANT: This function MUST be called once after the element of the
+ // DataGrid is attached to its parent element and every subsequent time the
+ // width of the parent element is changed in order to make it possible to
+ // resize the columns.
+ //
+ // If this function is not called after the DataGrid is attached to its
+ // parent element, then the DataGrid's columns will not be resizable.
+ updateWidths: function()
+ {
+ var headerTableColumns = this._headerTableColumnGroup.children;
+
+ var left = 0;
+ var tableWidth = this._dataTable.offsetWidth;
+ var numColumns = headerTableColumns.length;
+
+ if (!this.columnWidthsInitialized) {
+ // Give all the columns initial widths now so that during a resize,
+ // when the two columns that get resized get a percent value for
+ // their widths, all the other columns already have percent values
+ // for their widths.
+ for (var i = 0; i < numColumns; i++) {
+ var columnWidth = this.headerTableBody.rows[0].cells[i].offsetWidth;
+ var percentWidth = ((columnWidth / tableWidth) * 100) + "%";
+ this._headerTableColumnGroup.children[i].style.width = percentWidth;
+ this._dataTableColumnGroup.children[i].style.width = percentWidth;
+ }
+ this.columnWidthsInitialized = true;
+ }
+
+ // Make n - 1 resizers for n columns.
+ for (var i = 0; i < numColumns - 1; i++) {
+ var resizer = this.resizers[i];
+
+ if (!resizer) {
+ // This is the first call to updateWidth, so the resizers need
+ // to be created.
+ resizer = document.createElement("div");
+ resizer.addStyleClass("data-grid-resizer");
+ // This resizer is associated with the column to its right.
+ resizer.rightNeighboringColumnID = i + 1;
+ resizer.addEventListener("mousedown", this._startResizerDragging.bind(this), false);
+ this.element.appendChild(resizer);
+ this.resizers[i] = resizer;
+ }
+
+ // Get the width of the cell in the first (and only) row of the
+ // header table in order to determine the width of the column, since
+ // it is not possible to query a column for its width.
+ left += this.headerTableBody.rows[0].cells[i].offsetWidth;
+
+ resizer.style.left = left + "px";
+ }
+ },
+
+ addCreationNode: function(hasChildren)
+ {
+ if (this.creationNode)
+ this.creationNode.makeNormal();
+
+ var emptyData = {};
+ for (var column in this.columns)
+ emptyData[column] = '';
+ this.creationNode = new WebInspector.CreationDataGridNode(emptyData, hasChildren);
+ this.appendChild(this.creationNode);
+ },
+
+ appendChild: function(child)
+ {
+ this.insertChild(child, this.children.length);
+ },
+
+ insertChild: function(child, index)
+ {
+ if (!child)
+ throw("insertChild: Node can't be undefined or null.");
+ if (child.parent === this)
+ throw("insertChild: Node is already a child of this node.");
+
+ if (child.parent)
+ child.parent.removeChild(child);
+
+ this.children.splice(index, 0, child);
+ this.hasChildren = true;
+
+ child.parent = this;
+ child.dataGrid = this.dataGrid;
+ child._recalculateSiblings(index);
+
+ delete child._depth;
+ delete child._revealed;
+ delete child._attached;
+
+ var current = child.children[0];
+ while (current) {
+ current.dataGrid = this.dataGrid;
+ delete current._depth;
+ delete current._revealed;
+ delete current._attached;
+ current = current.traverseNextNode(false, child, true);
+ }
+
+ if (this.expanded)
+ child._attach();
+ },
+
+ removeChild: function(child)
+ {
+ if (!child)
+ throw("removeChild: Node can't be undefined or null.");
+ if (child.parent !== this)
+ throw("removeChild: Node is not a child of this node.");
+
+ child.deselect();
+
+ this.children.remove(child, true);
+
+ if (child.previousSibling)
+ child.previousSibling.nextSibling = child.nextSibling;
+ if (child.nextSibling)
+ child.nextSibling.previousSibling = child.previousSibling;
+
+ child.dataGrid = null;
+ child.parent = null;
+ child.nextSibling = null;
+ child.previousSibling = null;
+
+ if (this.children.length <= 0)
+ this.hasChildren = false;
+ },
+
+ removeChildren: function()
+ {
+ for (var i = 0; i < this.children.length; ++i) {
+ var child = this.children[i];
+ child.deselect();
+ child._detach();
+
+ child.dataGrid = null;
+ child.parent = null;
+ child.nextSibling = null;
+ child.previousSibling = null;
+ }
+
+ this.children = [];
+ this.hasChildren = false;
+ },
+
+ removeChildrenRecursive: function()
+ {
+ var childrenToRemove = this.children;
+
+ var child = this.children[0];
+ while (child) {
+ if (child.children.length)
+ childrenToRemove = childrenToRemove.concat(child.children);
+ child = child.traverseNextNode(false, this, true);
+ }
+
+ for (var i = 0; i < childrenToRemove.length; ++i) {
+ var child = childrenToRemove[i];
+ child.deselect();
+ child._detach();
+
+ child.children = [];
+ child.dataGrid = null;
+ child.parent = null;
+ child.nextSibling = null;
+ child.previousSibling = null;
+ }
+
+ this.children = [];
+ },
+
+ handleKeyEvent: function(event)
+ {
+ if (!this.selectedNode || event.shiftKey || event.metaKey || event.ctrlKey)
+ return false;
+
+ var handled = false;
+ var nextSelectedNode;
+ if (event.keyIdentifier === "Up" && !event.altKey) {
+ nextSelectedNode = this.selectedNode.traversePreviousNode(true);
+ while (nextSelectedNode && !nextSelectedNode.selectable)
+ nextSelectedNode = nextSelectedNode.traversePreviousNode(!this.expandTreeNodesWhenArrowing);
+ handled = nextSelectedNode ? true : false;
+ } else if (event.keyIdentifier === "Down" && !event.altKey) {
+ nextSelectedNode = this.selectedNode.traverseNextNode(true);
+ while (nextSelectedNode && !nextSelectedNode.selectable)
+ nextSelectedNode = nextSelectedNode.traverseNextNode(!this.expandTreeNodesWhenArrowing);
+ handled = nextSelectedNode ? true : false;
+ } else if (event.keyIdentifier === "Left") {
+ if (this.selectedNode.expanded) {
+ if (event.altKey)
+ this.selectedNode.collapseRecursively();
+ else
+ this.selectedNode.collapse();
+ handled = true;
+ } else if (this.selectedNode.parent && !this.selectedNode.parent.root) {
+ handled = true;
+ if (this.selectedNode.parent.selectable) {
+ nextSelectedNode = this.selectedNode.parent;
+ handled = nextSelectedNode ? true : false;
+ } else if (this.selectedNode.parent)
+ this.selectedNode.parent.collapse();
+ }
+ } else if (event.keyIdentifier === "Right") {
+ if (!this.selectedNode.revealed) {
+ this.selectedNode.reveal();
+ handled = true;
+ } else if (this.selectedNode.hasChildren) {
+ handled = true;
+ if (this.selectedNode.expanded) {
+ nextSelectedNode = this.selectedNode.children[0];
+ handled = nextSelectedNode ? true : false;
+ } else {
+ if (event.altKey)
+ this.selectedNode.expandRecursively();
+ else
+ this.selectedNode.expand();
+ }
+ }
+ }
+
+ if (nextSelectedNode) {
+ nextSelectedNode.reveal();
+ nextSelectedNode.select();
+ }
+
+ if (handled) {
+ event.preventDefault();
+ event.stopPropagation();
+ }
+
+ return handled;
+ },
+
+ expand: function()
+ {
+ // This is the root, do nothing.
+ },
+
+ collapse: function()
+ {
+ // This is the root, do nothing.
+ },
+
+ reveal: function()
+ {
+ // This is the root, do nothing.
+ },
+
+ dataGridNodeFromEvent: function(event)
+ {
+ var rowElement = event.target.enclosingNodeOrSelfWithNodeName("tr");
+ return rowElement._dataGridNode;
+ },
+
+ dataGridNodeFromPoint: function(x, y)
+ {
+ var node = this._dataTable.ownerDocument.elementFromPoint(x, y);
+ var rowElement = node.enclosingNodeOrSelfWithNodeName("tr");
+ return rowElement._dataGridNode;
+ },
+
+ _keyDown: function(event)
+ {
+ this.handleKeyEvent(event);
+ },
+
+ _clickInHeaderCell: function(event)
+ {
+ var cell = event.target.enclosingNodeOrSelfWithNodeName("th");
+ if (!cell || !cell.columnIdentifier || !cell.hasStyleClass("sortable"))
+ return;
+
+ var sortOrder = this.sortOrder;
+
+ if (this._sortColumnCell) {
+ this._sortColumnCell.removeStyleClass("sort-ascending");
+ this._sortColumnCell.removeStyleClass("sort-descending");
+ }
+
+ if (cell == this._sortColumnCell) {
+ if (sortOrder == "ascending")
+ sortOrder = "descending";
+ else
+ sortOrder = "ascending";
+ }
+
+ this._sortColumnCell = cell;
+
+ cell.addStyleClass("sort-" + sortOrder);
+
+ this.dispatchEventToListeners("sorting changed");
+ },
+
+ _mouseDownInDataTable: function(event)
+ {
+ var gridNode = this.dataGridNodeFromEvent(event);
+ if (!gridNode || !gridNode.selectable)
+ return;
+
+ if (gridNode.isEventWithinDisclosureTriangle(event))
+ return;
+
+ if (event.metaKey) {
+ if (gridNode.selected)
+ gridNode.deselect();
+ else
+ gridNode.select();
+ } else
+ gridNode.select();
+ },
+
+ _clickInDataTable: function(event)
+ {
+ var gridNode = this.dataGridNodeFromEvent(event);
+ if (!gridNode || !gridNode.hasChildren)
+ return;
+
+ if (!gridNode.isEventWithinDisclosureTriangle(event))
+ return;
+
+ if (gridNode.expanded) {
+ if (event.altKey)
+ gridNode.collapseRecursively();
+ else
+ gridNode.collapse();
+ } else {
+ if (event.altKey)
+ gridNode.expandRecursively();
+ else
+ gridNode.expand();
+ }
+ },
+
+ _startResizerDragging: function(event)
+ {
+ this.currentResizer = event.target;
+ if (!this.currentResizer.rightNeighboringColumnID)
+ return;
+ WebInspector.elementDragStart(this.lastResizer, this._resizerDragging.bind(this),
+ this._endResizerDragging.bind(this), event, "col-resize");
+ },
+
+ _resizerDragging: function(event)
+ {
+ var resizer = this.currentResizer;
+ if (!resizer)
+ return;
+
+ // Constrain the dragpoint to be within the containing div of the
+ // datagrid.
+ var dragPoint = event.clientX - this.element.totalOffsetLeft;
+ // Constrain the dragpoint to be within the space made up by the
+ // column directly to the left and the column directly to the right.
+ var leftEdgeOfPreviousColumn = 0;
+ var firstRowCells = this.headerTableBody.rows[0].cells;
+ for (var i = 0; i < resizer.rightNeighboringColumnID - 1; i++)
+ leftEdgeOfPreviousColumn += firstRowCells[i].offsetWidth;
+
+ var rightEdgeOfNextColumn = leftEdgeOfPreviousColumn + firstRowCells[resizer.rightNeighboringColumnID - 1].offsetWidth + firstRowCells[resizer.rightNeighboringColumnID].offsetWidth;
+
+ // Give each column some padding so that they don't disappear.
+ var leftMinimum = leftEdgeOfPreviousColumn + this.ColumnResizePadding;
+ var rightMaximum = rightEdgeOfNextColumn - this.ColumnResizePadding;
+
+ dragPoint = Number.constrain(dragPoint, leftMinimum, rightMaximum);
+
+ resizer.style.left = (dragPoint - this.CenterResizerOverBorderAdjustment) + "px";
+
+ var percentLeftColumn = (((dragPoint - leftEdgeOfPreviousColumn) / this._dataTable.offsetWidth) * 100) + "%";
+ this._headerTableColumnGroup.children[resizer.rightNeighboringColumnID - 1].style.width = percentLeftColumn;
+ this._dataTableColumnGroup.children[resizer.rightNeighboringColumnID - 1].style.width = percentLeftColumn;
+
+ var percentRightColumn = (((rightEdgeOfNextColumn - dragPoint) / this._dataTable.offsetWidth) * 100) + "%";
+ this._headerTableColumnGroup.children[resizer.rightNeighboringColumnID].style.width = percentRightColumn;
+ this._dataTableColumnGroup.children[resizer.rightNeighboringColumnID].style.width = percentRightColumn;
+
+ event.preventDefault();
+ },
+
+ _endResizerDragging: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+ this.currentResizer = null;
+ },
+
+ ColumnResizePadding: 10,
+
+ CenterResizerOverBorderAdjustment: 3,
+}
+
+WebInspector.DataGrid.prototype.__proto__ = WebInspector.Object.prototype;
+
+WebInspector.DataGridNode = function(data, hasChildren)
+{
+ this._expanded = false;
+ this._selected = false;
+ this._shouldRefreshChildren = true;
+ this._data = data || {};
+ this.hasChildren = hasChildren || false;
+ this.children = [];
+ this.dataGrid = null;
+ this.parent = null;
+ this.previousSibling = null;
+ this.nextSibling = null;
+ this.disclosureToggleWidth = 10;
+}
+
+WebInspector.DataGridNode.prototype = {
+ selectable: true,
+
+ get element()
+ {
+ if (this._element)
+ return this._element;
+
+ if (!this.dataGrid)
+ return null;
+
+ this._element = document.createElement("tr");
+ this._element._dataGridNode = this;
+
+ if (this.hasChildren)
+ this._element.addStyleClass("parent");
+ if (this.expanded)
+ this._element.addStyleClass("expanded");
+ if (this.selected)
+ this._element.addStyleClass("selected");
+ if (this.revealed)
+ this._element.addStyleClass("revealed");
+
+ for (var columnIdentifier in this.dataGrid.columns) {
+ var cell = this.createCell(columnIdentifier);
+ this._element.appendChild(cell);
+ }
+
+ return this._element;
+ },
+
+ get data()
+ {
+ return this._data;
+ },
+
+ set data(x)
+ {
+ this._data = x || {};
+ this.refresh();
+ },
+
+ get revealed()
+ {
+ if ("_revealed" in this)
+ return this._revealed;
+
+ var currentAncestor = this.parent;
+ while (currentAncestor && !currentAncestor.root) {
+ if (!currentAncestor.expanded) {
+ this._revealed = false;
+ return false;
+ }
+
+ currentAncestor = currentAncestor.parent;
+ }
+
+ this._revealed = true;
+ return true;
+ },
+
+ set hasChildren(x)
+ {
+ if (this._hasChildren === x)
+ return;
+
+ this._hasChildren = x;
+
+ if (!this._element)
+ return;
+
+ if (this._hasChildren)
+ {
+ this._element.addStyleClass("parent");
+ if (this.expanded)
+ this._element.addStyleClass("expanded");
+ }
+ else
+ {
+ this._element.removeStyleClass("parent");
+ this._element.removeStyleClass("expanded");
+ }
+ },
+
+ get hasChildren()
+ {
+ return this._hasChildren;
+ },
+
+ set revealed(x)
+ {
+ if (this._revealed === x)
+ return;
+
+ this._revealed = x;
+
+ if (this._element) {
+ if (this._revealed)
+ this._element.addStyleClass("revealed");
+ else
+ this._element.removeStyleClass("revealed");
+ }
+
+ for (var i = 0; i < this.children.length; ++i)
+ this.children[i].revealed = x && this.expanded;
+ },
+
+ get depth()
+ {
+ if ("_depth" in this)
+ return this._depth;
+ if (this.parent && !this.parent.root)
+ this._depth = this.parent.depth + 1;
+ else
+ this._depth = 0;
+ return this._depth;
+ },
+
+ get shouldRefreshChildren()
+ {
+ return this._shouldRefreshChildren;
+ },
+
+ set shouldRefreshChildren(x)
+ {
+ this._shouldRefreshChildren = x;
+ if (x && this.expanded)
+ this.expand();
+ },
+
+ get selected()
+ {
+ return this._selected;
+ },
+
+ set selected(x)
+ {
+ if (x)
+ this.select();
+ else
+ this.deselect();
+ },
+
+ get expanded()
+ {
+ return this._expanded;
+ },
+
+ set expanded(x)
+ {
+ if (x)
+ this.expand();
+ else
+ this.collapse();
+ },
+
+ refresh: function()
+ {
+ if (!this._element || !this.dataGrid)
+ return;
+
+ this._element.removeChildren();
+
+ for (var columnIdentifier in this.dataGrid.columns) {
+ var cell = this.createCell(columnIdentifier);
+ this._element.appendChild(cell);
+ }
+ },
+
+ createCell: function(columnIdentifier)
+ {
+ var cell = document.createElement("td");
+ cell.className = columnIdentifier + "-column";
+
+ var alignment = this.dataGrid.aligned[columnIdentifier];
+ if (alignment)
+ cell.addStyleClass(alignment);
+
+ var div = document.createElement("div");
+ div.textContent = this.data[columnIdentifier];
+ cell.appendChild(div);
+
+ if (columnIdentifier === this.dataGrid.disclosureColumnIdentifier) {
+ cell.addStyleClass("disclosure");
+ if (this.depth)
+ cell.style.setProperty("padding-left", (this.depth * this.dataGrid.indentWidth) + "px");
+ }
+
+ return cell;
+ },
+
+ // Share these functions with DataGrid. They are written to work with a DataGridNode this object.
+ appendChild: WebInspector.DataGrid.prototype.appendChild,
+ insertChild: WebInspector.DataGrid.prototype.insertChild,
+ removeChild: WebInspector.DataGrid.prototype.removeChild,
+ removeChildren: WebInspector.DataGrid.prototype.removeChildren,
+ removeChildrenRecursive: WebInspector.DataGrid.prototype.removeChildrenRecursive,
+
+ _recalculateSiblings: function(myIndex)
+ {
+ if (!this.parent)
+ return;
+
+ var previousChild = (myIndex > 0 ? this.parent.children[myIndex - 1] : null);
+
+ if (previousChild) {
+ previousChild.nextSibling = this;
+ this.previousSibling = previousChild;
+ } else
+ this.previousSibling = null;
+
+ var nextChild = this.parent.children[myIndex + 1];
+
+ if (nextChild) {
+ nextChild.previousSibling = this;
+ this.nextSibling = nextChild;
+ } else
+ this.nextSibling = null;
+ },
+
+ collapse: function()
+ {
+ if (this._element)
+ this._element.removeStyleClass("expanded");
+
+ this._expanded = false;
+
+ for (var i = 0; i < this.children.length; ++i)
+ this.children[i].revealed = false;
+
+ this.dispatchEventToListeners("collapsed");
+ },
+
+ collapseRecursively: function()
+ {
+ var item = this;
+ while (item) {
+ if (item.expanded)
+ item.collapse();
+ item = item.traverseNextNode(false, this, true);
+ }
+ },
+
+ expand: function()
+ {
+ if (!this.hasChildren || this.expanded)
+ return;
+
+ if (this.revealed && !this._shouldRefreshChildren)
+ for (var i = 0; i < this.children.length; ++i)
+ this.children[i].revealed = true;
+
+ if (this._shouldRefreshChildren) {
+ for (var i = 0; i < this.children.length; ++i)
+ this.children[i]._detach();
+
+ this.dispatchEventToListeners("populate");
+
+ if (this._attached) {
+ for (var i = 0; i < this.children.length; ++i) {
+ var child = this.children[i];
+ if (this.revealed)
+ child.revealed = true;
+ child._attach();
+ }
+ }
+
+ delete this._shouldRefreshChildren;
+ }
+
+ if (this._element)
+ this._element.addStyleClass("expanded");
+
+ this._expanded = true;
+
+ this.dispatchEventToListeners("expanded");
+ },
+
+ expandRecursively: function()
+ {
+ var item = this;
+ while (item) {
+ item.expand();
+ item = item.traverseNextNode(false, this);
+ }
+ },
+
+ reveal: function()
+ {
+ var currentAncestor = this.parent;
+ while (currentAncestor && !currentAncestor.root) {
+ if (!currentAncestor.expanded)
+ currentAncestor.expand();
+ currentAncestor = currentAncestor.parent;
+ }
+
+ this.element.scrollIntoViewIfNeeded(false);
+
+ this.dispatchEventToListeners("revealed");
+ },
+
+ select: function(supressSelectedEvent)
+ {
+ if (!this.dataGrid || !this.selectable || this.selected)
+ return;
+
+ if (this.dataGrid.selectedNode)
+ this.dataGrid.selectedNode.deselect();
+
+ this._selected = true;
+ this.dataGrid.selectedNode = this;
+
+ if (this._element)
+ this._element.addStyleClass("selected");
+
+ if (!supressSelectedEvent)
+ this.dispatchEventToListeners("selected");
+ },
+
+ deselect: function(supressDeselectedEvent)
+ {
+ if (!this.dataGrid || this.dataGrid.selectedNode !== this || !this.selected)
+ return;
+
+ this._selected = false;
+ this.dataGrid.selectedNode = null;
+
+ if (this._element)
+ this._element.removeStyleClass("selected");
+
+ if (!supressDeselectedEvent)
+ this.dispatchEventToListeners("deselected");
+ },
+
+ traverseNextNode: function(skipHidden, stayWithin, dontPopulate, info)
+ {
+ if (!dontPopulate && this.hasChildren)
+ this.dispatchEventToListeners("populate");
+
+ if (info)
+ info.depthChange = 0;
+
+ var node = (!skipHidden || this.revealed) ? this.children[0] : null;
+ if (node && (!skipHidden || this.expanded)) {
+ if (info)
+ info.depthChange = 1;
+ return node;
+ }
+
+ if (this === stayWithin)
+ return null;
+
+ node = (!skipHidden || this.revealed) ? this.nextSibling : null;
+ if (node)
+ return node;
+
+ node = this;
+ while (node && !node.root && !((!skipHidden || node.revealed) ? node.nextSibling : null) && node.parent !== stayWithin) {
+ if (info)
+ info.depthChange -= 1;
+ node = node.parent;
+ }
+
+ if (!node)
+ return null;
+
+ return (!skipHidden || node.revealed) ? node.nextSibling : null;
+ },
+
+ traversePreviousNode: function(skipHidden, dontPopulate)
+ {
+ var node = (!skipHidden || this.revealed) ? this.previousSibling : null;
+ if (!dontPopulate && node && node.hasChildren)
+ node.dispatchEventToListeners("populate");
+
+ while (node && ((!skipHidden || (node.revealed && node.expanded)) ? node.children[node.children.length - 1] : null)) {
+ if (!dontPopulate && node.hasChildren)
+ node.dispatchEventToListeners("populate");
+ node = ((!skipHidden || (node.revealed && node.expanded)) ? node.children[node.children.length - 1] : null);
+ }
+
+ if (node)
+ return node;
+
+ if (!this.parent || this.parent.root)
+ return null;
+
+ return this.parent;
+ },
+
+ isEventWithinDisclosureTriangle: function(event)
+ {
+ if (!this.hasChildren)
+ return false;
+ var cell = event.target.enclosingNodeOrSelfWithNodeName("td");
+ if (!cell.hasStyleClass("disclosure"))
+ return false;
+ var computedLeftPadding = window.getComputedStyle(cell).getPropertyCSSValue("padding-left").getFloatValue(CSSPrimitiveValue.CSS_PX);
+ var left = cell.totalOffsetLeft + computedLeftPadding;
+ return event.pageX >= left && event.pageX <= left + this.disclosureToggleWidth;
+ },
+
+ _attach: function()
+ {
+ if (!this.dataGrid || this._attached)
+ return;
+
+ this._attached = true;
+
+ var nextNode = null;
+ var previousNode = this.traversePreviousNode(true, true);
+ if (previousNode && previousNode.element.parentNode && previousNode.element.nextSibling)
+ var nextNode = previousNode.element.nextSibling;
+ if (!nextNode)
+ nextNode = this.dataGrid.dataTableBody.lastChild;
+ this.dataGrid.dataTableBody.insertBefore(this.element, nextNode);
+
+ if (this.expanded)
+ for (var i = 0; i < this.children.length; ++i)
+ this.children[i]._attach();
+ },
+
+ _detach: function()
+ {
+ if (!this._attached)
+ return;
+
+ this._attached = false;
+
+ if (this._element && this._element.parentNode)
+ this._element.parentNode.removeChild(this._element);
+
+ for (var i = 0; i < this.children.length; ++i)
+ this.children[i]._detach();
+ }
+}
+
+WebInspector.DataGridNode.prototype.__proto__ = WebInspector.Object.prototype;
+
+WebInspector.CreationDataGridNode = function(data, hasChildren)
+{
+ WebInspector.DataGridNode.call(this, data, hasChildren);
+ this.isCreationNode = true;
+}
+
+WebInspector.CreationDataGridNode.prototype = {
+ makeNormal: function()
+ {
+ delete this.isCreationNode;
+ delete this.makeNormal;
+ }
+}
+
+WebInspector.CreationDataGridNode.prototype.__proto__ = WebInspector.DataGridNode.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Database.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Database.js
new file mode 100644
index 0000000..dcab2ab
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Database.js
@@ -0,0 +1,113 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Database = function(database, domain, name, version)
+{
+ this._database = database;
+ this.domain = domain;
+ this.name = name;
+ this.version = version;
+}
+
+WebInspector.Database.prototype = {
+ isDatabase: function(db)
+ {
+ return this._database === db;
+ },
+
+ get name()
+ {
+ return this._name;
+ },
+
+ set name(x)
+ {
+ if (this._name === x)
+ return;
+ this._name = x;
+ },
+
+ get version()
+ {
+ return this._version;
+ },
+
+ set version(x)
+ {
+ if (this._version === x)
+ return;
+ this._version = x;
+ },
+
+ get domain()
+ {
+ return this._domain;
+ },
+
+ set domain(x)
+ {
+ if (this._domain === x)
+ return;
+ this._domain = x;
+ },
+
+ get displayDomain()
+ {
+ return WebInspector.Resource.prototype.__lookupGetter__("displayDomain").call(this);
+ },
+
+ getTableNames: function(callback)
+ {
+ var names = InspectorController.databaseTableNames(this._database);
+ function sortingCallback()
+ {
+ callback(names.sort());
+ }
+ setTimeout(sortingCallback, 0);
+ },
+
+ executeSql: function(query, onSuccess, onError)
+ {
+ function successCallback(tx, result)
+ {
+ onSuccess(result);
+ }
+
+ function errorCallback(tx, error)
+ {
+ onError(error);
+ }
+
+ var self = this;
+ function queryTransaction(tx)
+ {
+ tx.executeSql(query, null, InspectorController.wrapCallback(successCallback.bind(self)), InspectorController.wrapCallback(errorCallback.bind(self)));
+ }
+ this._database.transaction(InspectorController.wrapCallback(queryTransaction.bind(this)), InspectorController.wrapCallback(errorCallback.bind(this)));
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseQueryView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseQueryView.js
new file mode 100644
index 0000000..6c5fa02
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseQueryView.js
@@ -0,0 +1,195 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.DatabaseQueryView = function(database)
+{
+ WebInspector.View.call(this);
+
+ this.database = database;
+
+ this.element.addStyleClass("storage-view");
+ this.element.addStyleClass("query");
+ this.element.tabIndex = 0;
+
+ this.element.addEventListener("selectstart", this._selectStart.bind(this), false);
+
+ this.promptElement = document.createElement("div");
+ this.promptElement.className = "database-query-prompt";
+ this.promptElement.appendChild(document.createElement("br"));
+ this.promptElement.handleKeyEvent = this._promptKeyDown.bind(this);
+ this.element.appendChild(this.promptElement);
+
+ this.prompt = new WebInspector.TextPrompt(this.promptElement, this.completions.bind(this), " ");
+}
+
+WebInspector.DatabaseQueryView.prototype = {
+ show: function(parentElement)
+ {
+ WebInspector.View.prototype.show.call(this, parentElement);
+
+ function moveBackIfOutside()
+ {
+ if (!this.prompt.isCaretInsidePrompt() && window.getSelection().isCollapsed)
+ this.prompt.moveCaretToEndOfPrompt();
+ }
+
+ setTimeout(moveBackIfOutside.bind(this), 0);
+ },
+
+ completions: function(wordRange, bestMatchOnly, completionsReadyCallback)
+ {
+ var prefix = wordRange.toString().toLowerCase();
+ if (!prefix.length)
+ return;
+
+ var results = [];
+
+ function accumulateMatches(textArray)
+ {
+ if (bestMatchOnly && results.length)
+ return;
+ for (var i = 0; i < textArray.length; ++i) {
+ var text = textArray[i].toLowerCase();
+ if (text.length < prefix.length)
+ continue;
+ if (text.indexOf(prefix) !== 0)
+ continue;
+ results.push(textArray[i]);
+ if (bestMatchOnly)
+ return;
+ }
+ }
+
+ function tableNamesCallback(tableNames)
+ {
+ accumulateMatches(tableNames.map(function(name) { return name + " " }));
+ accumulateMatches(["SELECT ", "FROM ", "WHERE ", "LIMIT ", "DELETE FROM ", "CREATE ", "DROP ", "TABLE ", "INDEX ", "UPDATE ", "INSERT INTO ", "VALUES ("]);
+
+ completionsReadyCallback(results);
+ }
+ this.database.getTableNames(tableNamesCallback);
+ },
+
+ _promptKeyDown: function(event)
+ {
+ switch (event.keyIdentifier) {
+ case "Enter":
+ this._enterKeyPressed(event);
+ return;
+ }
+
+ this.prompt.handleKeyEvent(event);
+ },
+
+ _selectStart: function(event)
+ {
+ if (this._selectionTimeout)
+ clearTimeout(this._selectionTimeout);
+
+ this.prompt.clearAutoComplete();
+
+ function moveBackIfOutside()
+ {
+ delete this._selectionTimeout;
+ if (!this.prompt.isCaretInsidePrompt() && window.getSelection().isCollapsed)
+ this.prompt.moveCaretToEndOfPrompt();
+ this.prompt.autoCompleteSoon();
+ }
+
+ this._selectionTimeout = setTimeout(moveBackIfOutside.bind(this), 100);
+ },
+
+ _enterKeyPressed: function(event)
+ {
+ event.preventDefault();
+ event.stopPropagation();
+
+ this.prompt.clearAutoComplete(true);
+
+ var query = this.prompt.text;
+ if (!query.length)
+ return;
+
+ this.prompt.history.push(query);
+ this.prompt.historyOffset = 0;
+ this.prompt.text = "";
+
+ this.database.executeSql(query, this._queryFinished.bind(this, query), this._queryError.bind(this, query));
+ },
+
+ _queryFinished: function(query, result)
+ {
+ var dataGrid = WebInspector.panels.storage.dataGridForResult(result);
+ if (!dataGrid)
+ return;
+ dataGrid.element.addStyleClass("inline");
+ this._appendQueryResult(query, dataGrid.element);
+
+ if (query.match(/^create /i) || query.match(/^drop table /i))
+ WebInspector.panels.storage.updateDatabaseTables(this.database);
+ },
+
+ _queryError: function(query, error)
+ {
+ if (error.code == 1)
+ var message = error.message;
+ else if (error.code == 2)
+ var message = WebInspector.UIString("Database no longer has expected version.");
+ else
+ var message = WebInspector.UIString("An unexpected error %s occurred.", error.code);
+
+ this._appendQueryResult(query, message, "error");
+ },
+
+ _appendQueryResult: function(query, result, resultClassName)
+ {
+ var element = document.createElement("div");
+ element.className = "database-user-query";
+
+ var commandTextElement = document.createElement("span");
+ commandTextElement.className = "database-query-text";
+ commandTextElement.textContent = query;
+ element.appendChild(commandTextElement);
+
+ var resultElement = document.createElement("div");
+ resultElement.className = "database-query-result";
+
+ if (resultClassName)
+ resultElement.addStyleClass(resultClassName);
+
+ if (typeof result === "string" || result instanceof String)
+ resultElement.textContent = result;
+ else if (result && result.nodeName)
+ resultElement.appendChild(result);
+
+ if (resultElement.childNodes.length)
+ element.appendChild(resultElement);
+
+ this.element.insertBefore(element, this.promptElement);
+ this.promptElement.scrollIntoView(false);
+ }
+}
+
+WebInspector.DatabaseQueryView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseTableView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseTableView.js
new file mode 100644
index 0000000..aa76794f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/DatabaseTableView.js
@@ -0,0 +1,89 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.DatabaseTableView = function(database, tableName)
+{
+ WebInspector.View.call(this);
+
+ this.database = database;
+ this.tableName = tableName;
+
+ this.element.addStyleClass("storage-view");
+ this.element.addStyleClass("table");
+
+ this.refreshButton = new WebInspector.StatusBarButton(WebInspector.UIString("Refresh"), "refresh-storage-status-bar-item");
+ this.refreshButton.addEventListener("click", this._refreshButtonClicked.bind(this), false);
+}
+
+WebInspector.DatabaseTableView.prototype = {
+ show: function(parentElement)
+ {
+ WebInspector.View.prototype.show.call(this, parentElement);
+ this.update();
+ },
+
+ get statusBarItems()
+ {
+ return [this.refreshButton];
+ },
+
+ update: function()
+ {
+ this.database.executeSql("SELECT * FROM " + this.tableName, this._queryFinished.bind(this), this._queryError.bind(this));
+ },
+
+ _queryFinished: function(result)
+ {
+ this.element.removeChildren();
+
+ var dataGrid = WebInspector.panels.storage.dataGridForResult(result);
+ if (!dataGrid) {
+ var emptyMsgElement = document.createElement("div");
+ emptyMsgElement.className = "storage-table-empty";
+ emptyMsgElement.textContent = WebInspector.UIString("The “%sâ€\ntable is empty.", this.tableName);
+ this.element.appendChild(emptyMsgElement);
+ return;
+ }
+
+ this.element.appendChild(dataGrid.element);
+ },
+
+ _queryError: function(error)
+ {
+ this.element.removeChildren();
+
+ var errorMsgElement = document.createElement("div");
+ errorMsgElement.className = "storage-table-error";
+ errorMsgElement.textContent = WebInspector.UIString("An error occurred trying to\nread the “%s†table.", this.tableName);
+ this.element.appendChild(errorMsgElement);
+ },
+
+ _refreshButtonClicked: function(event)
+ {
+ this.update();
+ }
+}
+
+WebInspector.DatabaseTableView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Drawer.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Drawer.js
new file mode 100644
index 0000000..1b50f91
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Drawer.js
@@ -0,0 +1,202 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Drawer = function()
+{
+ WebInspector.View.call(this, document.getElementById("drawer"));
+
+ this.mainStatusBar = document.getElementById("main-status-bar");
+ this.mainStatusBar.addEventListener("mousedown", this._startStatusBarDragging.bind(this), true);
+ this.viewStatusBar = document.getElementById("other-drawer-status-bar-items");
+}
+
+WebInspector.Drawer.prototype = {
+ get visibleView()
+ {
+ return this._visibleView;
+ },
+
+ set visibleView(x)
+ {
+ if (this._visibleView === x) {
+ this.visible = !this.visible;
+ return;
+ }
+
+ var firstTime = !this._visibleView;
+ if (this._visibleView)
+ this._visibleView.hide();
+
+ this._visibleView = x;
+
+ if (x && !firstTime) {
+ this._safelyRemoveChildren();
+ this.viewStatusBar.removeChildren(); // optimize this? call old.detach()
+ x.attach(this.element, this.viewStatusBar);
+ x.show();
+ this.visible = true;
+ }
+ },
+
+ showView: function(view)
+ {
+ if (!this.visible || this.visibleView !== view)
+ this.visibleView = view;
+ },
+
+ show: function()
+ {
+ if (this._animating || this.visible)
+ return;
+
+ if (this.visibleView)
+ this.visibleView.show();
+
+ WebInspector.View.prototype.show.call(this);
+
+ this._animating = true;
+
+ document.body.addStyleClass("drawer-visible");
+
+ var anchoredItems = document.getElementById("anchored-status-bar-items");
+
+ var animations = [
+ {element: document.getElementById("main"), end: {bottom: this.element.offsetHeight}},
+ {element: document.getElementById("main-status-bar"), start: {"padding-left": anchoredItems.offsetWidth - 1}, end: {"padding-left": 0}},
+ {element: document.getElementById("other-drawer-status-bar-items"), start: {opacity: 0}, end: {opacity: 1}}
+ ];
+
+ var consoleStatusBar = document.getElementById("drawer-status-bar");
+ consoleStatusBar.insertBefore(anchoredItems, consoleStatusBar.firstChild);
+
+ function animationFinished()
+ {
+ if ("updateStatusBarItems" in WebInspector.currentPanel)
+ WebInspector.currentPanel.updateStatusBarItems();
+ if (this.visibleView.afterShow)
+ this.visibleView.afterShow();
+ delete this._animating;
+ }
+
+ WebInspector.animateStyle(animations, window.event && window.event.shiftKey ? 2000 : 250, animationFinished.bind(this));
+ },
+
+ hide: function()
+ {
+ if (this._animating || !this.visible)
+ return;
+
+ WebInspector.View.prototype.hide.call(this);
+
+ if (this.visibleView)
+ this.visibleView.hide();
+
+ this._animating = true;
+
+ if (this.element === WebInspector.currentFocusElement || this.element.isAncestor(WebInspector.currentFocusElement))
+ WebInspector.currentFocusElement = WebInspector.previousFocusElement;
+
+ var anchoredItems = document.getElementById("anchored-status-bar-items");
+
+ // Temporally set properties and classes to mimic the post-animation values so panels
+ // like Elements in their updateStatusBarItems call will size things to fit the final location.
+ document.getElementById("main-status-bar").style.setProperty("padding-left", (anchoredItems.offsetWidth - 1) + "px");
+ document.body.removeStyleClass("drawer-visible");
+ if ("updateStatusBarItems" in WebInspector.currentPanel)
+ WebInspector.currentPanel.updateStatusBarItems();
+ document.body.addStyleClass("drawer-visible");
+
+ var animations = [
+ {element: document.getElementById("main"), end: {bottom: 0}},
+ {element: document.getElementById("main-status-bar"), start: {"padding-left": 0}, end: {"padding-left": anchoredItems.offsetWidth - 1}},
+ {element: document.getElementById("other-drawer-status-bar-items"), start: {opacity: 1}, end: {opacity: 0}}
+ ];
+
+ function animationFinished()
+ {
+ var mainStatusBar = document.getElementById("main-status-bar");
+ mainStatusBar.insertBefore(anchoredItems, mainStatusBar.firstChild);
+ mainStatusBar.style.removeProperty("padding-left");
+ document.body.removeStyleClass("drawer-visible");
+ delete this._animating;
+ }
+
+ WebInspector.animateStyle(animations, window.event && window.event.shiftKey ? 2000 : 250, animationFinished.bind(this));
+ },
+
+ _safelyRemoveChildren: function()
+ {
+ var child = this.element.firstChild;
+ while (child) {
+ if (child.id !== "drawer-status-bar") {
+ var moveTo = child.nextSibling;
+ this.element.removeChild(child);
+ child = moveTo;
+ } else
+ child = child.nextSibling;
+ }
+ },
+
+ _startStatusBarDragging: function(event)
+ {
+ if (!this.visible || event.target !== document.getElementById("main-status-bar"))
+ return;
+
+ WebInspector.elementDragStart(document.getElementById("main-status-bar"), this._statusBarDragging.bind(this), this._endStatusBarDragging.bind(this), event, "row-resize");
+
+ this._statusBarDragOffset = event.pageY - this.element.totalOffsetTop;
+
+ event.stopPropagation();
+ },
+
+ _statusBarDragging: function(event)
+ {
+ var mainElement = document.getElementById("main");
+
+ var height = window.innerHeight - event.pageY + this._statusBarDragOffset;
+ height = Number.constrain(height, Preferences.minConsoleHeight, window.innerHeight - mainElement.totalOffsetTop - Preferences.minConsoleHeight);
+
+ mainElement.style.bottom = height + "px";
+ this.element.style.height = height + "px";
+
+ event.preventDefault();
+ event.stopPropagation();
+ },
+
+ _endStatusBarDragging: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+
+ delete this._statusBarDragOffset;
+
+ event.stopPropagation();
+ }
+}
+
+WebInspector.Drawer.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsPanel.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsPanel.js
new file mode 100644
index 0000000..49a1188
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsPanel.js
@@ -0,0 +1,1045 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com>
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ElementsPanel = function()
+{
+ WebInspector.Panel.call(this);
+
+ this.element.addStyleClass("elements");
+
+ this.contentElement = document.createElement("div");
+ this.contentElement.id = "elements-content";
+ this.contentElement.className = "outline-disclosure";
+
+ this.treeOutline = new WebInspector.ElementsTreeOutline();
+ this.treeOutline.panel = this;
+ this.treeOutline.includeRootDOMNode = false;
+ this.treeOutline.selectEnabled = true;
+
+ this.treeOutline.focusedNodeChanged = function(forceUpdate)
+ {
+ if (this.panel.visible && WebInspector.currentFocusElement !== document.getElementById("search"))
+ WebInspector.currentFocusElement = document.getElementById("main-panels");
+
+ this.panel.updateBreadcrumb(forceUpdate);
+
+ for (var pane in this.panel.sidebarPanes)
+ this.panel.sidebarPanes[pane].needsUpdate = true;
+
+ this.panel.updateStyles(true);
+ this.panel.updateMetrics();
+ this.panel.updateProperties();
+
+ if (InspectorController.searchingForNode()) {
+ InspectorController.toggleNodeSearch();
+ this.panel.nodeSearchButton.removeStyleClass("toggled-on");
+ }
+ if (this._focusedDOMNode)
+ InjectedScriptAccess.addInspectedNode(this._focusedDOMNode.id, function() {});
+ };
+
+ this.contentElement.appendChild(this.treeOutline.element);
+
+ this.crumbsElement = document.createElement("div");
+ this.crumbsElement.className = "crumbs";
+ this.crumbsElement.addEventListener("mousemove", this._mouseMovedInCrumbs.bind(this), false);
+ this.crumbsElement.addEventListener("mouseout", this._mouseMovedOutOfCrumbs.bind(this), false);
+
+ this.sidebarPanes = {};
+ this.sidebarPanes.styles = new WebInspector.StylesSidebarPane();
+ this.sidebarPanes.metrics = new WebInspector.MetricsSidebarPane();
+ this.sidebarPanes.properties = new WebInspector.PropertiesSidebarPane();
+
+ this.sidebarPanes.styles.onexpand = this.updateStyles.bind(this);
+ this.sidebarPanes.metrics.onexpand = this.updateMetrics.bind(this);
+ this.sidebarPanes.properties.onexpand = this.updateProperties.bind(this);
+
+ this.sidebarPanes.styles.expanded = true;
+
+ this.sidebarPanes.styles.addEventListener("style edited", this._stylesPaneEdited, this);
+ this.sidebarPanes.styles.addEventListener("style property toggled", this._stylesPaneEdited, this);
+ this.sidebarPanes.metrics.addEventListener("metrics edited", this._metricsPaneEdited, this);
+
+ this.sidebarElement = document.createElement("div");
+ this.sidebarElement.id = "elements-sidebar";
+
+ this.sidebarElement.appendChild(this.sidebarPanes.styles.element);
+ this.sidebarElement.appendChild(this.sidebarPanes.metrics.element);
+ this.sidebarElement.appendChild(this.sidebarPanes.properties.element);
+
+ this.sidebarResizeElement = document.createElement("div");
+ this.sidebarResizeElement.className = "sidebar-resizer-vertical";
+ this.sidebarResizeElement.addEventListener("mousedown", this.rightSidebarResizerDragStart.bind(this), false);
+
+ this.nodeSearchButton = new WebInspector.StatusBarButton(WebInspector.UIString("Select an element in the page to inspect it."), "node-search-status-bar-item");
+ this.nodeSearchButton.addEventListener("click", this._nodeSearchButtonClicked.bind(this), false);
+
+ this.searchingForNode = false;
+
+ this.element.appendChild(this.contentElement);
+ this.element.appendChild(this.sidebarElement);
+ this.element.appendChild(this.sidebarResizeElement);
+
+ this._changedStyles = {};
+
+ this.reset();
+}
+
+WebInspector.ElementsPanel.prototype = {
+ toolbarItemClass: "elements",
+
+ get toolbarItemLabel()
+ {
+ return WebInspector.UIString("Elements");
+ },
+
+ get statusBarItems()
+ {
+ return [this.nodeSearchButton.element, this.crumbsElement];
+ },
+
+ updateStatusBarItems: function()
+ {
+ this.updateBreadcrumbSizes();
+ },
+
+ show: function()
+ {
+ WebInspector.Panel.prototype.show.call(this);
+ this.sidebarResizeElement.style.right = (this.sidebarElement.offsetWidth - 3) + "px";
+ this.updateBreadcrumb();
+ this.treeOutline.updateSelection();
+ if (this.recentlyModifiedNodes.length)
+ this._updateModifiedNodes();
+ },
+
+ hide: function()
+ {
+ WebInspector.Panel.prototype.hide.call(this);
+
+ WebInspector.hoveredDOMNode = null;
+
+ if (InspectorController.searchingForNode()) {
+ InspectorController.toggleNodeSearch();
+ this.nodeSearchButton.toggled = false;
+ }
+ },
+
+ resize: function()
+ {
+ this.treeOutline.updateSelection();
+ this.updateBreadcrumbSizes();
+ },
+
+ reset: function()
+ {
+ this.rootDOMNode = null;
+ this.focusedDOMNode = null;
+
+ WebInspector.hoveredDOMNode = null;
+
+ if (InspectorController.searchingForNode()) {
+ InspectorController.toggleNodeSearch();
+ this.nodeSearchButton.toggled = false;
+ }
+
+ this.recentlyModifiedNodes = [];
+
+ delete this.currentQuery;
+ this.searchCanceled();
+
+ var domWindow = WebInspector.domAgent.domWindow;
+ if (!domWindow || !domWindow.document || !domWindow.document.firstChild)
+ return;
+
+ // If the window isn't visible, return early so the DOM tree isn't built
+ // and mutation event listeners are not added.
+ if (!InspectorController.isWindowVisible())
+ return;
+
+ var inspectedRootDocument = domWindow.document;
+ inspectedRootDocument.addEventListener("DOMNodeInserted", this._nodeInserted.bind(this));
+ inspectedRootDocument.addEventListener("DOMNodeRemoved", this._nodeRemoved.bind(this));
+
+ this.rootDOMNode = inspectedRootDocument;
+
+ var canidateFocusNode = inspectedRootDocument.body || inspectedRootDocument.documentElement;
+ if (canidateFocusNode) {
+ this.treeOutline.suppressSelectHighlight = true;
+ this.focusedDOMNode = canidateFocusNode;
+ this.treeOutline.suppressSelectHighlight = false;
+
+ if (this.treeOutline.selectedTreeElement)
+ this.treeOutline.selectedTreeElement.expand();
+ }
+ },
+
+ searchCanceled: function()
+ {
+ if (this._searchResults) {
+ for (var i = 0; i < this._searchResults.length; ++i) {
+ var treeElement = this.treeOutline.findTreeElement(this._searchResults[i]);
+ if (treeElement)
+ treeElement.highlighted = false;
+ }
+ }
+
+ WebInspector.updateSearchMatchesCount(0, this);
+
+ this._currentSearchResultIndex = 0;
+ this._searchResults = [];
+ InjectedScriptAccess.searchCanceled(function() {});
+ },
+
+ performSearch: function(query)
+ {
+ // Call searchCanceled since it will reset everything we need before doing a new search.
+ this.searchCanceled();
+
+ const whitespaceTrimmedQuery = query.trimWhitespace();
+ if (!whitespaceTrimmedQuery.length)
+ return;
+
+ this._updatedMatchCountOnce = false;
+ this._matchesCountUpdateTimeout = null;
+
+ InjectedScriptAccess.performSearch(whitespaceTrimmedQuery, function() {});
+ },
+
+ _updateMatchesCount: function()
+ {
+ WebInspector.updateSearchMatchesCount(this._searchResults.length, this);
+ this._matchesCountUpdateTimeout = null;
+ this._updatedMatchCountOnce = true;
+ },
+
+ _updateMatchesCountSoon: function()
+ {
+ if (!this._updatedMatchCountOnce)
+ return this._updateMatchesCount();
+ if (this._matchesCountUpdateTimeout)
+ return;
+ // Update the matches count every half-second so it doesn't feel twitchy.
+ this._matchesCountUpdateTimeout = setTimeout(this._updateMatchesCount.bind(this), 500);
+ },
+
+ addNodesToSearchResult: function(nodeIds)
+ {
+ if (!nodeIds)
+ return;
+
+ var nodeIdsArray = nodeIds.split(",");
+ for (var i = 0; i < nodeIdsArray.length; ++i) {
+ var nodeId = nodeIdsArray[i];
+ var node = WebInspector.domAgent.nodeForId(nodeId);
+ if (!node)
+ continue;
+
+ if (!this._searchResults.length) {
+ this._currentSearchResultIndex = 0;
+ this.focusedDOMNode = node;
+ }
+
+ this._searchResults.push(node);
+
+ // Highlight the tree element to show it matched the search.
+ // FIXME: highlight the substrings in text nodes and attributes.
+ var treeElement = this.treeOutline.findTreeElement(node);
+ if (treeElement)
+ treeElement.highlighted = true;
+ }
+
+ this._updateMatchesCountSoon();
+ },
+
+ jumpToNextSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ if (++this._currentSearchResultIndex >= this._searchResults.length)
+ this._currentSearchResultIndex = 0;
+ this.focusedDOMNode = this._searchResults[this._currentSearchResultIndex];
+ },
+
+ jumpToPreviousSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ if (--this._currentSearchResultIndex < 0)
+ this._currentSearchResultIndex = (this._searchResults.length - 1);
+ this.focusedDOMNode = this._searchResults[this._currentSearchResultIndex];
+ },
+
+ renameSelector: function(oldIdentifier, newIdentifier, oldSelector, newSelector)
+ {
+ // TODO: Implement Shifting the oldSelector, and its contents to a newSelector
+ },
+
+ addStyleChange: function(identifier, style, property)
+ {
+ if (!style.parentRule)
+ return;
+
+ var selector = style.parentRule.selectorText;
+ if (!this._changedStyles[identifier])
+ this._changedStyles[identifier] = {};
+
+ if (!this._changedStyles[identifier][selector])
+ this._changedStyles[identifier][selector] = {};
+
+ if (!this._changedStyles[identifier][selector][property])
+ WebInspector.styleChanges += 1;
+
+ this._changedStyles[identifier][selector][property] = style.getPropertyValue(property);
+ },
+
+ removeStyleChange: function(identifier, style, property)
+ {
+ if (!style.parentRule)
+ return;
+
+ var selector = style.parentRule.selectorText;
+ if (!this._changedStyles[identifier] || !this._changedStyles[identifier][selector])
+ return;
+
+ if (this._changedStyles[identifier][selector][property]) {
+ delete this._changedStyles[identifier][selector][property];
+ WebInspector.styleChanges -= 1;
+ }
+ },
+
+ generateStylesheet: function()
+ {
+ if (!WebInspector.styleChanges)
+ return;
+
+ // Merge Down to Just Selectors
+ var mergedSelectors = {};
+ for (var identifier in this._changedStyles) {
+ for (var selector in this._changedStyles[identifier]) {
+ if (!mergedSelectors[selector])
+ mergedSelectors[selector] = this._changedStyles[identifier][selector];
+ else { // merge on selector
+ var merge = {};
+ for (var property in mergedSelectors[selector])
+ merge[property] = mergedSelectors[selector][property];
+ for (var property in this._changedStyles[identifier][selector]) {
+ if (!merge[property])
+ merge[property] = this._changedStyles[identifier][selector][property];
+ else { // merge on property within a selector, include comment to notify user
+ var value1 = merge[property];
+ var value2 = this._changedStyles[identifier][selector][property];
+
+ if (value1 === value2)
+ merge[property] = [value1];
+ else if (value1 instanceof Array)
+ merge[property].push(value2);
+ else
+ merge[property] = [value1, value2];
+ }
+ }
+ mergedSelectors[selector] = merge;
+ }
+ }
+ }
+
+ var builder = [];
+ builder.push("/**");
+ builder.push(" * Inspector Generated Stylesheet"); // UIString?
+ builder.push(" */\n");
+
+ var indent = " ";
+ function displayProperty(property, value, comment) {
+ if (comment)
+ return indent + "/* " + property + ": " + value + "; */";
+ else
+ return indent + property + ": " + value + ";";
+ }
+
+ for (var selector in mergedSelectors) {
+ var psuedoStyle = mergedSelectors[selector];
+ var properties = Object.properties(psuedoStyle);
+ if (properties.length) {
+ builder.push(selector + " {");
+ for (var i = 0; i < properties.length; ++i) {
+ var property = properties[i];
+ var value = psuedoStyle[property];
+ if (!(value instanceof Array))
+ builder.push(displayProperty(property, value));
+ else {
+ if (value.length === 1)
+ builder.push(displayProperty(property, value) + " /* merged from equivalent edits */"); // UIString?
+ else {
+ builder.push(indent + "/* There was a Conflict... There were Multiple Edits for '" + property + "' */"); // UIString?
+ for (var j = 0; j < value.length; ++j)
+ builder.push(displayProperty(property, value, true));
+ }
+ }
+ }
+ builder.push("}\n");
+ }
+ }
+
+ WebInspector.showConsole();
+ WebInspector.console.addMessage(new WebInspector.ConsoleTextMessage(builder.join("\n")));
+ },
+
+ get rootDOMNode()
+ {
+ return this.treeOutline.rootDOMNode;
+ },
+
+ set rootDOMNode(x)
+ {
+ this.treeOutline.rootDOMNode = x;
+ },
+
+ get focusedDOMNode()
+ {
+ return this.treeOutline.focusedDOMNode;
+ },
+
+ set focusedDOMNode(x)
+ {
+ this.treeOutline.focusedDOMNode = x;
+ },
+
+ _nodeInserted: function(event)
+ {
+ this.recentlyModifiedNodes.push({node: event.target, parent: event.relatedNode, inserted: true});
+ if (this.visible)
+ this._updateModifiedNodesSoon();
+ },
+
+ _nodeRemoved: function(event)
+ {
+ this.recentlyModifiedNodes.push({node: event.target, parent: event.relatedNode, removed: true});
+ if (this.visible)
+ this._updateModifiedNodesSoon();
+ },
+
+ _updateModifiedNodesSoon: function()
+ {
+ if ("_updateModifiedNodesTimeout" in this)
+ return;
+ this._updateModifiedNodesTimeout = setTimeout(this._updateModifiedNodes.bind(this), 0);
+ },
+
+ _updateModifiedNodes: function()
+ {
+ if ("_updateModifiedNodesTimeout" in this) {
+ clearTimeout(this._updateModifiedNodesTimeout);
+ delete this._updateModifiedNodesTimeout;
+ }
+
+ var updatedParentTreeElements = [];
+ var updateBreadcrumbs = false;
+
+ for (var i = 0; i < this.recentlyModifiedNodes.length; ++i) {
+ var replaced = this.recentlyModifiedNodes[i].replaced;
+ var parent = this.recentlyModifiedNodes[i].parent;
+ if (!parent)
+ continue;
+
+ var parentNodeItem = this.treeOutline.findTreeElement(parent);
+ if (parentNodeItem && !parentNodeItem.alreadyUpdatedChildren) {
+ parentNodeItem.updateChildren(replaced);
+ parentNodeItem.alreadyUpdatedChildren = true;
+ updatedParentTreeElements.push(parentNodeItem);
+ }
+
+ if (!updateBreadcrumbs && (this.focusedDOMNode === parent || isAncestor(this.focusedDOMNode, parent)))
+ updateBreadcrumbs = true;
+ }
+
+ for (var i = 0; i < updatedParentTreeElements.length; ++i)
+ delete updatedParentTreeElements[i].alreadyUpdatedChildren;
+
+ this.recentlyModifiedNodes = [];
+
+ if (updateBreadcrumbs)
+ this.updateBreadcrumb(true);
+ },
+
+ _stylesPaneEdited: function()
+ {
+ this.sidebarPanes.metrics.needsUpdate = true;
+ this.updateMetrics();
+ },
+
+ _metricsPaneEdited: function()
+ {
+ this.sidebarPanes.styles.needsUpdate = true;
+ this.updateStyles(true);
+ },
+
+ _mouseMovedInCrumbs: function(event)
+ {
+ var nodeUnderMouse = document.elementFromPoint(event.pageX, event.pageY);
+ var crumbElement = nodeUnderMouse.enclosingNodeOrSelfWithClass("crumb");
+
+ WebInspector.hoveredDOMNode = (crumbElement ? crumbElement.representedObject : null);
+
+ if ("_mouseOutOfCrumbsTimeout" in this) {
+ clearTimeout(this._mouseOutOfCrumbsTimeout);
+ delete this._mouseOutOfCrumbsTimeout;
+ }
+ },
+
+ _mouseMovedOutOfCrumbs: function(event)
+ {
+ var nodeUnderMouse = document.elementFromPoint(event.pageX, event.pageY);
+ if (nodeUnderMouse.isDescendant(this.crumbsElement))
+ return;
+
+ WebInspector.hoveredDOMNode = null;
+
+ this._mouseOutOfCrumbsTimeout = setTimeout(this.updateBreadcrumbSizes.bind(this), 1000);
+ },
+
+ updateBreadcrumb: function(forceUpdate)
+ {
+ if (!this.visible)
+ return;
+
+ var crumbs = this.crumbsElement;
+
+ var handled = false;
+ var foundRoot = false;
+ var crumb = crumbs.firstChild;
+ while (crumb) {
+ if (crumb.representedObject === this.rootDOMNode)
+ foundRoot = true;
+
+ if (foundRoot)
+ crumb.addStyleClass("dimmed");
+ else
+ crumb.removeStyleClass("dimmed");
+
+ if (crumb.representedObject === this.focusedDOMNode) {
+ crumb.addStyleClass("selected");
+ handled = true;
+ } else {
+ crumb.removeStyleClass("selected");
+ }
+
+ crumb = crumb.nextSibling;
+ }
+
+ if (handled && !forceUpdate) {
+ // We don't need to rebuild the crumbs, but we need to adjust sizes
+ // to reflect the new focused or root node.
+ this.updateBreadcrumbSizes();
+ return;
+ }
+
+ crumbs.removeChildren();
+
+ var panel = this;
+
+ function selectCrumbFunction(event)
+ {
+ var crumb = event.currentTarget;
+ if (crumb.hasStyleClass("collapsed")) {
+ // Clicking a collapsed crumb will expose the hidden crumbs.
+ if (crumb === panel.crumbsElement.firstChild) {
+ // If the focused crumb is the first child, pick the farthest crumb
+ // that is still hidden. This allows the user to expose every crumb.
+ var currentCrumb = crumb;
+ while (currentCrumb) {
+ var hidden = currentCrumb.hasStyleClass("hidden");
+ var collapsed = currentCrumb.hasStyleClass("collapsed");
+ if (!hidden && !collapsed)
+ break;
+ crumb = currentCrumb;
+ currentCrumb = currentCrumb.nextSibling;
+ }
+ }
+
+ panel.updateBreadcrumbSizes(crumb);
+ } else {
+ // Clicking a dimmed crumb or double clicking (event.detail >= 2)
+ // will change the root node in addition to the focused node.
+ if (event.detail >= 2 || crumb.hasStyleClass("dimmed"))
+ panel.rootDOMNode = crumb.representedObject.parentNode;
+ panel.focusedDOMNode = crumb.representedObject;
+ }
+
+ event.preventDefault();
+ }
+
+ foundRoot = false;
+ for (var current = this.focusedDOMNode; current; current = current.parentNode) {
+ if (current.nodeType === Node.DOCUMENT_NODE)
+ continue;
+
+ if (current === this.rootDOMNode)
+ foundRoot = true;
+
+ var crumb = document.createElement("span");
+ crumb.className = "crumb";
+ crumb.representedObject = current;
+ crumb.addEventListener("mousedown", selectCrumbFunction, false);
+
+ var crumbTitle;
+ switch (current.nodeType) {
+ case Node.ELEMENT_NODE:
+ crumbTitle = current.nodeName.toLowerCase();
+
+ var nameElement = document.createElement("span");
+ nameElement.textContent = crumbTitle;
+ crumb.appendChild(nameElement);
+
+ var idAttribute = current.getAttribute("id");
+ if (idAttribute) {
+ var idElement = document.createElement("span");
+ crumb.appendChild(idElement);
+
+ var part = "#" + idAttribute;
+ crumbTitle += part;
+ idElement.appendChild(document.createTextNode(part));
+
+ // Mark the name as extra, since the ID is more important.
+ nameElement.className = "extra";
+ }
+
+ var classAttribute = current.getAttribute("class");
+ if (classAttribute) {
+ var classes = classAttribute.split(/\s+/);
+ var foundClasses = {};
+
+ if (classes.length) {
+ var classesElement = document.createElement("span");
+ classesElement.className = "extra";
+ crumb.appendChild(classesElement);
+
+ for (var i = 0; i < classes.length; ++i) {
+ var className = classes[i];
+ if (className && !(className in foundClasses)) {
+ var part = "." + className;
+ crumbTitle += part;
+ classesElement.appendChild(document.createTextNode(part));
+ foundClasses[className] = true;
+ }
+ }
+ }
+ }
+
+ break;
+
+ case Node.TEXT_NODE:
+ if (isNodeWhitespace.call(current))
+ crumbTitle = WebInspector.UIString("(whitespace)");
+ else
+ crumbTitle = WebInspector.UIString("(text)");
+ break
+
+ case Node.COMMENT_NODE:
+ crumbTitle = "<!-->";
+ break;
+
+ case Node.DOCUMENT_TYPE_NODE:
+ crumbTitle = "<!DOCTYPE>";
+ break;
+
+ default:
+ crumbTitle = current.nodeName.toLowerCase();
+ }
+
+ if (!crumb.childNodes.length) {
+ var nameElement = document.createElement("span");
+ nameElement.textContent = crumbTitle;
+ crumb.appendChild(nameElement);
+ }
+
+ crumb.title = crumbTitle;
+
+ if (foundRoot)
+ crumb.addStyleClass("dimmed");
+ if (current === this.focusedDOMNode)
+ crumb.addStyleClass("selected");
+ if (!crumbs.childNodes.length)
+ crumb.addStyleClass("end");
+
+ crumbs.appendChild(crumb);
+ }
+
+ if (crumbs.hasChildNodes())
+ crumbs.lastChild.addStyleClass("start");
+
+ this.updateBreadcrumbSizes();
+ },
+
+ updateBreadcrumbSizes: function(focusedCrumb)
+ {
+ if (!this.visible)
+ return;
+
+ if (document.body.offsetWidth <= 0) {
+ // The stylesheet hasn't loaded yet or the window is closed,
+ // so we can't calculate what is need. Return early.
+ return;
+ }
+
+ var crumbs = this.crumbsElement;
+ if (!crumbs.childNodes.length || crumbs.offsetWidth <= 0)
+ return; // No crumbs, do nothing.
+
+ // A Zero index is the right most child crumb in the breadcrumb.
+ var selectedIndex = 0;
+ var focusedIndex = 0;
+ var selectedCrumb;
+
+ var i = 0;
+ var crumb = crumbs.firstChild;
+ while (crumb) {
+ // Find the selected crumb and index.
+ if (!selectedCrumb && crumb.hasStyleClass("selected")) {
+ selectedCrumb = crumb;
+ selectedIndex = i;
+ }
+
+ // Find the focused crumb index.
+ if (crumb === focusedCrumb)
+ focusedIndex = i;
+
+ // Remove any styles that affect size before
+ // deciding to shorten any crumbs.
+ if (crumb !== crumbs.lastChild)
+ crumb.removeStyleClass("start");
+ if (crumb !== crumbs.firstChild)
+ crumb.removeStyleClass("end");
+
+ crumb.removeStyleClass("compact");
+ crumb.removeStyleClass("collapsed");
+ crumb.removeStyleClass("hidden");
+
+ crumb = crumb.nextSibling;
+ ++i;
+ }
+
+ // Restore the start and end crumb classes in case they got removed in coalesceCollapsedCrumbs().
+ // The order of the crumbs in the document is opposite of the visual order.
+ crumbs.firstChild.addStyleClass("end");
+ crumbs.lastChild.addStyleClass("start");
+
+ function crumbsAreSmallerThanContainer()
+ {
+ var rightPadding = 20;
+ var errorWarningElement = document.getElementById("error-warning-count");
+ if (!WebInspector.drawer.visible && errorWarningElement)
+ rightPadding += errorWarningElement.offsetWidth;
+ return ((crumbs.totalOffsetLeft + crumbs.offsetWidth + rightPadding) < window.innerWidth);
+ }
+
+ if (crumbsAreSmallerThanContainer())
+ return; // No need to compact the crumbs, they all fit at full size.
+
+ var BothSides = 0;
+ var AncestorSide = -1;
+ var ChildSide = 1;
+
+ function makeCrumbsSmaller(shrinkingFunction, direction, significantCrumb)
+ {
+ if (!significantCrumb)
+ significantCrumb = (focusedCrumb || selectedCrumb);
+
+ if (significantCrumb === selectedCrumb)
+ var significantIndex = selectedIndex;
+ else if (significantCrumb === focusedCrumb)
+ var significantIndex = focusedIndex;
+ else {
+ var significantIndex = 0;
+ for (var i = 0; i < crumbs.childNodes.length; ++i) {
+ if (crumbs.childNodes[i] === significantCrumb) {
+ significantIndex = i;
+ break;
+ }
+ }
+ }
+
+ function shrinkCrumbAtIndex(index)
+ {
+ var shrinkCrumb = crumbs.childNodes[index];
+ if (shrinkCrumb && shrinkCrumb !== significantCrumb)
+ shrinkingFunction(shrinkCrumb);
+ if (crumbsAreSmallerThanContainer())
+ return true; // No need to compact the crumbs more.
+ return false;
+ }
+
+ // Shrink crumbs one at a time by applying the shrinkingFunction until the crumbs
+ // fit in the container or we run out of crumbs to shrink.
+ if (direction) {
+ // Crumbs are shrunk on only one side (based on direction) of the signifcant crumb.
+ var index = (direction > 0 ? 0 : crumbs.childNodes.length - 1);
+ while (index !== significantIndex) {
+ if (shrinkCrumbAtIndex(index))
+ return true;
+ index += (direction > 0 ? 1 : -1);
+ }
+ } else {
+ // Crumbs are shrunk in order of descending distance from the signifcant crumb,
+ // with a tie going to child crumbs.
+ var startIndex = 0;
+ var endIndex = crumbs.childNodes.length - 1;
+ while (startIndex != significantIndex || endIndex != significantIndex) {
+ var startDistance = significantIndex - startIndex;
+ var endDistance = endIndex - significantIndex;
+ if (startDistance >= endDistance)
+ var index = startIndex++;
+ else
+ var index = endIndex--;
+ if (shrinkCrumbAtIndex(index))
+ return true;
+ }
+ }
+
+ // We are not small enough yet, return false so the caller knows.
+ return false;
+ }
+
+ function coalesceCollapsedCrumbs()
+ {
+ var crumb = crumbs.firstChild;
+ var collapsedRun = false;
+ var newStartNeeded = false;
+ var newEndNeeded = false;
+ while (crumb) {
+ var hidden = crumb.hasStyleClass("hidden");
+ if (!hidden) {
+ var collapsed = crumb.hasStyleClass("collapsed");
+ if (collapsedRun && collapsed) {
+ crumb.addStyleClass("hidden");
+ crumb.removeStyleClass("compact");
+ crumb.removeStyleClass("collapsed");
+
+ if (crumb.hasStyleClass("start")) {
+ crumb.removeStyleClass("start");
+ newStartNeeded = true;
+ }
+
+ if (crumb.hasStyleClass("end")) {
+ crumb.removeStyleClass("end");
+ newEndNeeded = true;
+ }
+
+ continue;
+ }
+
+ collapsedRun = collapsed;
+
+ if (newEndNeeded) {
+ newEndNeeded = false;
+ crumb.addStyleClass("end");
+ }
+ } else
+ collapsedRun = true;
+ crumb = crumb.nextSibling;
+ }
+
+ if (newStartNeeded) {
+ crumb = crumbs.lastChild;
+ while (crumb) {
+ if (!crumb.hasStyleClass("hidden")) {
+ crumb.addStyleClass("start");
+ break;
+ }
+ crumb = crumb.previousSibling;
+ }
+ }
+ }
+
+ function compact(crumb)
+ {
+ if (crumb.hasStyleClass("hidden"))
+ return;
+ crumb.addStyleClass("compact");
+ }
+
+ function collapse(crumb, dontCoalesce)
+ {
+ if (crumb.hasStyleClass("hidden"))
+ return;
+ crumb.addStyleClass("collapsed");
+ crumb.removeStyleClass("compact");
+ if (!dontCoalesce)
+ coalesceCollapsedCrumbs();
+ }
+
+ function compactDimmed(crumb)
+ {
+ if (crumb.hasStyleClass("dimmed"))
+ compact(crumb);
+ }
+
+ function collapseDimmed(crumb)
+ {
+ if (crumb.hasStyleClass("dimmed"))
+ collapse(crumb);
+ }
+
+ if (!focusedCrumb) {
+ // When not focused on a crumb we can be biased and collapse less important
+ // crumbs that the user might not care much about.
+
+ // Compact child crumbs.
+ if (makeCrumbsSmaller(compact, ChildSide))
+ return;
+
+ // Collapse child crumbs.
+ if (makeCrumbsSmaller(collapse, ChildSide))
+ return;
+
+ // Compact dimmed ancestor crumbs.
+ if (makeCrumbsSmaller(compactDimmed, AncestorSide))
+ return;
+
+ // Collapse dimmed ancestor crumbs.
+ if (makeCrumbsSmaller(collapseDimmed, AncestorSide))
+ return;
+ }
+
+ // Compact ancestor crumbs, or from both sides if focused.
+ if (makeCrumbsSmaller(compact, (focusedCrumb ? BothSides : AncestorSide)))
+ return;
+
+ // Collapse ancestor crumbs, or from both sides if focused.
+ if (makeCrumbsSmaller(collapse, (focusedCrumb ? BothSides : AncestorSide)))
+ return;
+
+ if (!selectedCrumb)
+ return;
+
+ // Compact the selected crumb.
+ compact(selectedCrumb);
+ if (crumbsAreSmallerThanContainer())
+ return;
+
+ // Collapse the selected crumb as a last resort. Pass true to prevent coalescing.
+ collapse(selectedCrumb, true);
+ },
+
+ updateStyles: function(forceUpdate)
+ {
+ var stylesSidebarPane = this.sidebarPanes.styles;
+ if (!stylesSidebarPane.expanded || !stylesSidebarPane.needsUpdate)
+ return;
+
+ stylesSidebarPane.update(this.focusedDOMNode, null, forceUpdate);
+ stylesSidebarPane.needsUpdate = false;
+ },
+
+ updateMetrics: function()
+ {
+ var metricsSidebarPane = this.sidebarPanes.metrics;
+ if (!metricsSidebarPane.expanded || !metricsSidebarPane.needsUpdate)
+ return;
+
+ metricsSidebarPane.update(this.focusedDOMNode);
+ metricsSidebarPane.needsUpdate = false;
+ },
+
+ updateProperties: function()
+ {
+ var propertiesSidebarPane = this.sidebarPanes.properties;
+ if (!propertiesSidebarPane.expanded || !propertiesSidebarPane.needsUpdate)
+ return;
+
+ propertiesSidebarPane.update(this.focusedDOMNode);
+ propertiesSidebarPane.needsUpdate = false;
+ },
+
+ handleKeyEvent: function(event)
+ {
+ this.treeOutline.handleKeyEvent(event);
+ },
+
+ handleCopyEvent: function(event)
+ {
+ // Don't prevent the normal copy if the user has a selection.
+ if (!window.getSelection().isCollapsed)
+ return;
+
+ switch (this.focusedDOMNode.nodeType) {
+ case Node.ELEMENT_NODE:
+ // TODO: Introduce InspectorController.copyEvent that pushes appropriate markup into the clipboard.
+ var data = null;
+ break;
+
+ case Node.COMMENT_NODE:
+ var data = "<!--" + this.focusedDOMNode.nodeValue + "-->";
+ break;
+
+ default:
+ case Node.TEXT_NODE:
+ var data = this.focusedDOMNode.nodeValue;
+ }
+
+ event.clipboardData.clearData();
+ event.preventDefault();
+
+ if (data)
+ event.clipboardData.setData("text/plain", data);
+ },
+
+ rightSidebarResizerDragStart: function(event)
+ {
+ WebInspector.elementDragStart(this.sidebarElement, this.rightSidebarResizerDrag.bind(this), this.rightSidebarResizerDragEnd.bind(this), event, "col-resize");
+ },
+
+ rightSidebarResizerDragEnd: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+ },
+
+ rightSidebarResizerDrag: function(event)
+ {
+ var x = event.pageX;
+ var newWidth = Number.constrain(window.innerWidth - x, Preferences.minElementsSidebarWidth, window.innerWidth * 0.66);
+
+ this.sidebarElement.style.width = newWidth + "px";
+ this.contentElement.style.right = newWidth + "px";
+ this.sidebarResizeElement.style.right = (newWidth - 3) + "px";
+
+ this.treeOutline.updateSelection();
+
+ event.preventDefault();
+ },
+
+ _nodeSearchButtonClicked: function(event)
+ {
+ InspectorController.toggleNodeSearch();
+
+ this.nodeSearchButton.toggled = InspectorController.searchingForNode();
+ }
+}
+
+WebInspector.ElementsPanel.prototype.__proto__ = WebInspector.Panel.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsTreeOutline.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsTreeOutline.js
new file mode 100644
index 0000000..08ba1c2
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ElementsTreeOutline.js
@@ -0,0 +1,742 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com>
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ElementsTreeOutline = function() {
+ this.element = document.createElement("ol");
+ this.element.addEventListener("mousedown", this._onmousedown.bind(this), false);
+ this.element.addEventListener("dblclick", this._ondblclick.bind(this), false);
+ this.element.addEventListener("mousemove", this._onmousemove.bind(this), false);
+ this.element.addEventListener("mouseout", this._onmouseout.bind(this), false);
+
+ TreeOutline.call(this, this.element);
+
+ this.includeRootDOMNode = true;
+ this.selectEnabled = false;
+ this.rootDOMNode = null;
+ this.focusedDOMNode = null;
+}
+
+WebInspector.ElementsTreeOutline.prototype = {
+ get rootDOMNode()
+ {
+ return this._rootDOMNode;
+ },
+
+ set rootDOMNode(x)
+ {
+ if (this._rootDOMNode === x)
+ return;
+
+ this._rootDOMNode = x;
+
+ this.update();
+ },
+
+ get focusedDOMNode()
+ {
+ return this._focusedDOMNode;
+ },
+
+ set focusedDOMNode(x)
+ {
+ if (this._focusedDOMNode === x) {
+ this.revealAndSelectNode(x);
+ return;
+ }
+
+ this._focusedDOMNode = x;
+
+ this.revealAndSelectNode(x);
+
+ // The revealAndSelectNode() method might find a different element if there is inlined text,
+ // and the select() call would change the focusedDOMNode and reenter this setter. So to
+ // avoid calling focusedNodeChanged() twice, first check if _focusedDOMNode is the same
+ // node as the one passed in.
+ if (this._focusedDOMNode === x) {
+ this.focusedNodeChanged();
+
+ if (x && !this.suppressSelectHighlight) {
+ InspectorController.highlightDOMNode(x.id);
+
+ if ("_restorePreviousHighlightNodeTimeout" in this)
+ clearTimeout(this._restorePreviousHighlightNodeTimeout);
+
+ function restoreHighlightToHoveredNode()
+ {
+ var hoveredNode = WebInspector.hoveredDOMNode;
+ if (hoveredNode)
+ InspectorController.highlightDOMNode(hoveredNode.id);
+ else
+ InspectorController.hideDOMNodeHighlight();
+ }
+
+ this._restorePreviousHighlightNodeTimeout = setTimeout(restoreHighlightToHoveredNode, 2000);
+ }
+ }
+ },
+
+ update: function()
+ {
+ this.removeChildren();
+
+ if (!this.rootDOMNode)
+ return;
+
+ var treeElement;
+ if (this.includeRootDOMNode) {
+ treeElement = new WebInspector.ElementsTreeElement(this.rootDOMNode);
+ treeElement.selectable = this.selectEnabled;
+ this.appendChild(treeElement);
+ } else {
+ // FIXME: this could use findTreeElement to reuse a tree element if it already exists
+ var node = (Preferences.ignoreWhitespace ? firstChildSkippingWhitespace.call(this.rootDOMNode) : this.rootDOMNode.firstChild);
+ while (node) {
+ treeElement = new WebInspector.ElementsTreeElement(node);
+ treeElement.selectable = this.selectEnabled;
+ this.appendChild(treeElement);
+ node = Preferences.ignoreWhitespace ? nextSiblingSkippingWhitespace.call(node) : node.nextSibling;
+ }
+ }
+
+ this.updateSelection();
+ },
+
+ updateSelection: function()
+ {
+ if (!this.selectedTreeElement)
+ return;
+ var element = this.treeOutline.selectedTreeElement;
+ element.updateSelection();
+ },
+
+ focusedNodeChanged: function(forceUpdate) {},
+
+ findTreeElement: function(node)
+ {
+ var treeElement = TreeOutline.prototype.findTreeElement.call(this, node, isAncestorNode, parentNode);
+ if (!treeElement && node.nodeType === Node.TEXT_NODE) {
+ // The text node might have been inlined if it was short, so try to find the parent element.
+ treeElement = TreeOutline.prototype.findTreeElement.call(this, node.parentNode, isAncestorNode, parentNode);
+ }
+
+ return treeElement;
+ },
+
+ revealAndSelectNode: function(node)
+ {
+ if (!node)
+ return;
+
+ var treeElement = this.findTreeElement(node);
+ if (!treeElement)
+ return;
+
+ treeElement.reveal();
+ treeElement.select();
+ },
+
+ _treeElementFromEvent: function(event)
+ {
+ var root = this.element;
+
+ // We choose this X coordinate based on the knowledge that our list
+ // items extend nearly to the right edge of the outer <ol>.
+ var x = root.totalOffsetLeft + root.offsetWidth - 20;
+
+ var y = event.pageY;
+
+ // Our list items have 1-pixel cracks between them vertically. We avoid
+ // the cracks by checking slightly above and slightly below the mouse
+ // and seeing if we hit the same element each time.
+ var elementUnderMouse = this.treeElementFromPoint(x, y);
+ var elementAboveMouse = this.treeElementFromPoint(x, y - 2);
+ var element;
+ if (elementUnderMouse === elementAboveMouse)
+ element = elementUnderMouse;
+ else
+ element = this.treeElementFromPoint(x, y + 2);
+
+ return element;
+ },
+
+ _ondblclick: function(event)
+ {
+ var element = this._treeElementFromEvent(event);
+
+ if (!element || !element.ondblclick)
+ return;
+
+ element.ondblclick(element, event);
+ },
+
+ _onmousedown: function(event)
+ {
+ var element = this._treeElementFromEvent(event);
+
+ if (!element || element.isEventWithinDisclosureTriangle(event))
+ return;
+
+ element.select();
+ },
+
+ _onmousemove: function(event)
+ {
+ if (this._previousHoveredElement) {
+ this._previousHoveredElement.hovered = false;
+ delete this._previousHoveredElement;
+ }
+
+ var element = this._treeElementFromEvent(event);
+ if (element && !element.elementCloseTag) {
+ element.hovered = true;
+ this._previousHoveredElement = element;
+ }
+
+ WebInspector.hoveredDOMNode = (element && !element.elementCloseTag ? element.representedObject : null);
+ },
+
+ _onmouseout: function(event)
+ {
+ var nodeUnderMouse = document.elementFromPoint(event.pageX, event.pageY);
+ if (nodeUnderMouse && nodeUnderMouse.isDescendant(this.element))
+ return;
+
+ if (this._previousHoveredElement) {
+ this._previousHoveredElement.hovered = false;
+ delete this._previousHoveredElement;
+ }
+
+ WebInspector.hoveredDOMNode = null;
+ }
+}
+
+WebInspector.ElementsTreeOutline.prototype.__proto__ = TreeOutline.prototype;
+
+WebInspector.ElementsTreeElement = function(node)
+{
+ var hasChildren = Preferences.ignoreWhitespace ? (firstChildSkippingWhitespace.call(node) ? true : false) : node.hasChildNodes();
+ var titleInfo = nodeTitleInfo.call(node, hasChildren, WebInspector.linkifyURL);
+
+ if (titleInfo.hasChildren)
+ this.whitespaceIgnored = Preferences.ignoreWhitespace;
+
+ // The title will be updated in onattach.
+ TreeElement.call(this, "", node, titleInfo.hasChildren);
+
+ if (this.representedObject.nodeType == Node.ELEMENT_NODE)
+ this._canAddAttributes = true;
+}
+
+WebInspector.ElementsTreeElement.prototype = {
+ get highlighted()
+ {
+ return this._highlighted;
+ },
+
+ set highlighted(x)
+ {
+ if (this._highlighted === x)
+ return;
+
+ this._highlighted = x;
+
+ if (this.listItemElement) {
+ if (x)
+ this.listItemElement.addStyleClass("highlighted");
+ else
+ this.listItemElement.removeStyleClass("highlighted");
+ }
+ },
+
+ get hovered()
+ {
+ return this._hovered;
+ },
+
+ set hovered(x)
+ {
+ if (this._hovered === x)
+ return;
+
+ this._hovered = x;
+
+ if (this.listItemElement) {
+ if (x) {
+ this.updateSelection();
+ this.listItemElement.addStyleClass("hovered");
+ } else
+ this.listItemElement.removeStyleClass("hovered");
+ if (this._canAddAttributes)
+ this.toggleNewAttributeButton();
+ }
+ },
+
+ toggleNewAttributeButton: function()
+ {
+ function removeWhenEditing(event)
+ {
+ if (this._addAttributeElement && this._addAttributeElement.parentNode)
+ this._addAttributeElement.parentNode.removeChild(this._addAttributeElement);
+ delete this._addAttributeElement;
+ }
+
+ if (!this._addAttributeElement && this._hovered && !this._editing) {
+ var span = document.createElement("span");
+ span.className = "add-attribute";
+ span.textContent = "\u2026";
+ span.addEventListener("dblclick", removeWhenEditing.bind(this), false);
+ this._addAttributeElement = span;
+
+ var tag = this.listItemElement.getElementsByClassName("webkit-html-tag")[0];
+ this._insertInLastAttributePosition(tag, span);
+ } else if (!this._hovered && this._addAttributeElement) {
+ if (this._addAttributeElement.parentNode)
+ this._addAttributeElement.parentNode.removeChild(this._addAttributeElement);
+ delete this._addAttributeElement;
+ }
+ },
+
+ updateSelection: function()
+ {
+ var listItemElement = this.listItemElement;
+ if (!listItemElement)
+ return;
+
+ if (document.body.offsetWidth <= 0) {
+ // The stylesheet hasn't loaded yet or the window is closed,
+ // so we can't calculate what is need. Return early.
+ return;
+ }
+
+ if (!this.selectionElement) {
+ this.selectionElement = document.createElement("div");
+ this.selectionElement.className = "selection selected";
+ listItemElement.insertBefore(this.selectionElement, listItemElement.firstChild);
+ }
+
+ this.selectionElement.style.height = listItemElement.offsetHeight + "px";
+ },
+
+ onattach: function()
+ {
+ this.listItemElement.addEventListener("mousedown", this.onmousedown.bind(this), false);
+
+ if (this._highlighted)
+ this.listItemElement.addStyleClass("highlighted");
+
+ if (this._hovered) {
+ this.updateSelection();
+ this.listItemElement.addStyleClass("hovered");
+ }
+
+ this._updateTitle();
+
+ this._preventFollowingLinksOnDoubleClick();
+ },
+
+ _preventFollowingLinksOnDoubleClick: function()
+ {
+ var links = this.listItemElement.querySelectorAll("li > .webkit-html-tag > .webkit-html-attribute > .webkit-html-external-link, li > .webkit-html-tag > .webkit-html-attribute > .webkit-html-resource-link");
+ if (!links)
+ return;
+
+ for (var i = 0; i < links.length; ++i)
+ links[i].preventFollowOnDoubleClick = true;
+ },
+
+ onpopulate: function()
+ {
+ if (this.children.length || this.whitespaceIgnored !== Preferences.ignoreWhitespace)
+ return;
+
+ this.whitespaceIgnored = Preferences.ignoreWhitespace;
+
+ this.updateChildren();
+ },
+
+ updateChildren: function(fullRefresh)
+ {
+ WebInspector.domAgent.getChildNodesAsync(this.representedObject, this._updateChildren.bind(this, fullRefresh));
+ },
+
+ _updateChildren: function(fullRefresh)
+ {
+ if (fullRefresh) {
+ var selectedTreeElement = this.treeOutline.selectedTreeElement;
+ if (selectedTreeElement && selectedTreeElement.hasAncestor(this))
+ this.select();
+ this.removeChildren();
+ }
+
+ var treeElement = this;
+ var treeChildIndex = 0;
+
+ function updateChildrenOfNode(node)
+ {
+ var treeOutline = treeElement.treeOutline;
+ var child = (Preferences.ignoreWhitespace ? firstChildSkippingWhitespace.call(node) : node.firstChild);
+ while (child) {
+ var currentTreeElement = treeElement.children[treeChildIndex];
+ if (!currentTreeElement || currentTreeElement.representedObject !== child) {
+ // Find any existing element that is later in the children list.
+ var existingTreeElement = null;
+ for (var i = (treeChildIndex + 1); i < treeElement.children.length; ++i) {
+ if (treeElement.children[i].representedObject === child) {
+ existingTreeElement = treeElement.children[i];
+ break;
+ }
+ }
+
+ if (existingTreeElement && existingTreeElement.parent === treeElement) {
+ // If an existing element was found and it has the same parent, just move it.
+ var wasSelected = existingTreeElement.selected;
+ treeElement.removeChild(existingTreeElement);
+ treeElement.insertChild(existingTreeElement, treeChildIndex);
+ if (wasSelected)
+ existingTreeElement.select();
+ } else {
+ // No existing element found, insert a new element.
+ var newElement = new WebInspector.ElementsTreeElement(child);
+ newElement.selectable = treeOutline.selectEnabled;
+ treeElement.insertChild(newElement, treeChildIndex);
+ }
+ }
+
+ child = Preferences.ignoreWhitespace ? nextSiblingSkippingWhitespace.call(child) : child.nextSibling;
+ ++treeChildIndex;
+ }
+ }
+
+ // Remove any tree elements that no longer have this node (or this node's contentDocument) as their parent.
+ for (var i = (this.children.length - 1); i >= 0; --i) {
+ if ("elementCloseTag" in this.children[i])
+ continue;
+
+ var currentChild = this.children[i];
+ var currentNode = currentChild.representedObject;
+ var currentParentNode = currentNode.parentNode;
+
+ if (currentParentNode === this.representedObject)
+ continue;
+
+ var selectedTreeElement = this.treeOutline.selectedTreeElement;
+ if (selectedTreeElement && (selectedTreeElement === currentChild || selectedTreeElement.hasAncestor(currentChild)))
+ this.select();
+
+ this.removeChildAtIndex(i);
+ }
+
+ updateChildrenOfNode(this.representedObject);
+
+ var lastChild = this.children[this.children.length - 1];
+ if (this.representedObject.nodeType == Node.ELEMENT_NODE && (!lastChild || !lastChild.elementCloseTag)) {
+ var title = "<span class=\"webkit-html-tag close\">&lt;/" + this.representedObject.nodeName.toLowerCase().escapeHTML() + "&gt;</span>";
+ var item = new TreeElement(title, null, false);
+ item.selectable = false;
+ item.elementCloseTag = true;
+ this.appendChild(item);
+ }
+ },
+
+ onexpand: function()
+ {
+ this.treeOutline.updateSelection();
+ },
+
+ oncollapse: function()
+ {
+ this.treeOutline.updateSelection();
+ },
+
+ onreveal: function()
+ {
+ if (this.listItemElement)
+ this.listItemElement.scrollIntoViewIfNeeded(false);
+ },
+
+ onselect: function()
+ {
+ this.treeOutline.focusedDOMNode = this.representedObject;
+ this.updateSelection();
+ },
+
+ onmousedown: function(event)
+ {
+ if (this._editing)
+ return;
+
+ // Prevent selecting the nearest word on double click.
+ if (event.detail >= 2)
+ event.preventDefault();
+ },
+
+ ondblclick: function(treeElement, event)
+ {
+ if (this._editing)
+ return;
+
+ if (this._startEditing(event, treeElement))
+ return;
+
+ if (this.treeOutline.panel) {
+ this.treeOutline.rootDOMNode = this.representedObject.parentNode;
+ this.treeOutline.focusedDOMNode = this.representedObject;
+ }
+
+ if (this.hasChildren && !this.expanded)
+ this.expand();
+ },
+
+ _insertInLastAttributePosition: function(tag, node)
+ {
+ if (tag.getElementsByClassName("webkit-html-attribute").length > 0)
+ tag.insertBefore(node, tag.lastChild);
+ else {
+ var nodeName = tag.textContent.match(/^<(.*?)>$/)[1];
+ tag.textContent = '';
+ tag.appendChild(document.createTextNode('<'+nodeName));
+ tag.appendChild(node);
+ tag.appendChild(document.createTextNode('>'));
+ }
+ },
+
+ _startEditing: function(event, treeElement)
+ {
+ if (this.treeOutline.focusedDOMNode != this.representedObject)
+ return;
+
+ if (this.representedObject.nodeType != Node.ELEMENT_NODE && this.representedObject.nodeType != Node.TEXT_NODE)
+ return false;
+
+ var textNode = event.target.enclosingNodeOrSelfWithClass("webkit-html-text-node");
+ if (textNode)
+ return this._startEditingTextNode(textNode);
+
+ var attribute = event.target.enclosingNodeOrSelfWithClass("webkit-html-attribute");
+ if (attribute)
+ return this._startEditingAttribute(attribute, event.target);
+
+ var newAttribute = event.target.enclosingNodeOrSelfWithClass("add-attribute");
+ if (newAttribute)
+ return this._addNewAttribute(treeElement.listItemElement);
+
+ return false;
+ },
+
+ _addNewAttribute: function(listItemElement)
+ {
+ var attr = document.createElement("span");
+ attr.className = "webkit-html-attribute";
+ attr.style.marginLeft = "2px"; // overrides the .editing margin rule
+ attr.style.marginRight = "2px"; // overrides the .editing margin rule
+ var name = document.createElement("span");
+ name.className = "webkit-html-attribute-name new-attribute";
+ name.textContent = " ";
+ var value = document.createElement("span");
+ value.className = "webkit-html-attribute-value";
+ attr.appendChild(name);
+ attr.appendChild(value);
+
+ var tag = listItemElement.getElementsByClassName("webkit-html-tag")[0];
+ this._insertInLastAttributePosition(tag, attr);
+ return this._startEditingAttribute(attr, attr);
+ },
+
+ _triggerEditAttribute: function(attributeName)
+ {
+ var attributeElements = this.listItemElement.getElementsByClassName("webkit-html-attribute-name");
+ for (var i = 0, len = attributeElements.length; i < len; ++i) {
+ if (attributeElements[i].textContent === attributeName) {
+ for (var elem = attributeElements[i].nextSibling; elem; elem = elem.nextSibling) {
+ if (elem.nodeType !== Node.ELEMENT_NODE)
+ continue;
+
+ if (elem.hasStyleClass("webkit-html-attribute-value"))
+ return this._startEditingAttribute(attributeElements[i].parentNode, elem);
+ }
+ }
+ }
+ },
+
+ _startEditingAttribute: function(attribute, elementForSelection)
+ {
+ if (WebInspector.isBeingEdited(attribute))
+ return true;
+
+ var attributeNameElement = attribute.getElementsByClassName("webkit-html-attribute-name")[0];
+ if (!attributeNameElement)
+ return false;
+
+ var attributeName = attributeNameElement.innerText;
+
+ function removeZeroWidthSpaceRecursive(node)
+ {
+ if (node.nodeType === Node.TEXT_NODE) {
+ node.nodeValue = node.nodeValue.replace(/\u200B/g, "");
+ return;
+ }
+
+ if (node.nodeType !== Node.ELEMENT_NODE)
+ return;
+
+ for (var child = node.firstChild; child; child = child.nextSibling)
+ removeZeroWidthSpaceRecursive(child);
+ }
+
+ // Remove zero-width spaces that were added by nodeTitleInfo.
+ removeZeroWidthSpaceRecursive(attribute);
+
+ this._editing = true;
+
+ WebInspector.startEditing(attribute, this._attributeEditingCommitted.bind(this), this._editingCancelled.bind(this), attributeName);
+ window.getSelection().setBaseAndExtent(elementForSelection, 0, elementForSelection, 1);
+
+ return true;
+ },
+
+ _startEditingTextNode: function(textNode)
+ {
+ if (WebInspector.isBeingEdited(textNode))
+ return true;
+
+ this._editing = true;
+
+ WebInspector.startEditing(textNode, this._textNodeEditingCommitted.bind(this), this._editingCancelled.bind(this));
+ window.getSelection().setBaseAndExtent(textNode, 0, textNode, 1);
+
+ return true;
+ },
+
+ _attributeEditingCommitted: function(element, newText, oldText, attributeName, moveDirection)
+ {
+ delete this._editing;
+
+ // Before we do anything, determine where we should move
+ // next based on the current element's settings
+ var moveToAttribute;
+ var newAttribute;
+ if (moveDirection) {
+ var found = false;
+ var attributes = this.representedObject.attributes;
+ for (var i = 0, len = attributes.length; i < len; ++i) {
+ if (attributes[i].name === attributeName) {
+ found = true;
+ if (moveDirection === "backward" && i > 0)
+ moveToAttribute = attributes[i - 1].name;
+ else if (moveDirection === "forward" && i < attributes.length - 1)
+ moveToAttribute = attributes[i + 1].name;
+ else if (moveDirection === "forward" && i === attributes.length - 1)
+ newAttribute = true;
+ }
+ }
+
+ if (!found && moveDirection === "backward")
+ moveToAttribute = attributes[attributes.length - 1].name;
+ else if (!found && moveDirection === "forward" && !/^\s*$/.test(newText))
+ newAttribute = true;
+ }
+
+ function moveToNextAttributeIfNeeded() {
+ if (moveToAttribute)
+ this._triggerEditAttribute(moveToAttribute);
+ else if (newAttribute)
+ this._addNewAttribute(this.listItemElement);
+ }
+
+ var parseContainerElement = document.createElement("span");
+ parseContainerElement.innerHTML = "<span " + newText + "></span>";
+ var parseElement = parseContainerElement.firstChild;
+
+ if (!parseElement) {
+ this._editingCancelled(element, attributeName);
+ moveToNextAttributeIfNeeded.call(this);
+ return;
+ }
+
+ if (!parseElement.hasAttributes()) {
+ this.representedObject.removeAttribute(attributeName);
+ this._updateTitle();
+ moveToNextAttributeIfNeeded.call(this);
+ return;
+ }
+
+ var foundOriginalAttribute = false;
+ for (var i = 0; i < parseElement.attributes.length; ++i) {
+ var attr = parseElement.attributes[i];
+ foundOriginalAttribute = foundOriginalAttribute || attr.name === attributeName;
+ try {
+ this.representedObject.setAttribute(attr.name, attr.value);
+ } catch(e) {} // ignore invalid attribute (innerHTML doesn't throw errors, but this can)
+ }
+
+ if (!foundOriginalAttribute)
+ this.representedObject.removeAttribute(attributeName);
+
+ this._updateTitle();
+
+ this.treeOutline.focusedNodeChanged(true);
+
+ moveToNextAttributeIfNeeded.call(this);
+ },
+
+ _textNodeEditingCommitted: function(element, newText)
+ {
+ delete this._editing;
+
+ var textNode;
+ if (this.representedObject.nodeType == Node.ELEMENT_NODE) {
+ // We only show text nodes inline in elements if the element only
+ // has a single child, and that child is a text node.
+ textNode = this.representedObject.firstChild;
+ } else if (this.representedObject.nodeType == Node.TEXT_NODE)
+ textNode = this.representedObject;
+
+ textNode.nodeValue = newText;
+ this._updateTitle();
+ },
+
+ _editingCancelled: function(element, context)
+ {
+ delete this._editing;
+
+ this._updateTitle();
+ },
+
+ _updateTitle: function()
+ {
+ var title = nodeTitleInfo.call(this.representedObject, this.hasChildren, WebInspector.linkifyURL).title;
+ this.title = "<span class=\"highlight\">" + title + "</span>";
+ delete this.selectionElement;
+ this.updateSelection();
+ this._preventFollowingLinksOnDoubleClick();
+ },
+}
+
+WebInspector.ElementsTreeElement.prototype.__proto__ = TreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/FontView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/FontView.js
new file mode 100644
index 0000000..4e1c931
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/FontView.js
@@ -0,0 +1,104 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.FontView = function(resource)
+{
+ WebInspector.ResourceView.call(this, resource);
+
+ this.element.addStyleClass("font");
+
+ var uniqueFontName = "WebInspectorFontPreview" + this.resource.identifier;
+
+ this.fontStyleElement = document.createElement("style");
+ this.fontStyleElement.textContent = "@font-face { font-family: \"" + uniqueFontName + "\"; src: url(" + this.resource.url + "); }";
+ document.getElementsByTagName("head").item(0).appendChild(this.fontStyleElement);
+
+ this.fontPreviewElement = document.createElement("div");
+ this.fontPreviewElement.className = "preview";
+ this.contentElement.appendChild(this.fontPreviewElement);
+
+ this.fontPreviewElement.style.setProperty("font-family", uniqueFontName, null);
+ this.fontPreviewElement.innerHTML = "ABCDEFGHIJKLM<br>NOPQRSTUVWXYZ<br>abcdefghijklm<br>nopqrstuvwxyz<br>1234567890";
+
+ this.updateFontPreviewSize();
+}
+
+WebInspector.FontView.prototype = {
+ show: function(parentElement)
+ {
+ WebInspector.ResourceView.prototype.show.call(this, parentElement);
+ this.updateFontPreviewSize();
+ },
+
+ resize: function()
+ {
+ this.updateFontPreviewSize();
+ },
+
+ updateFontPreviewSize: function ()
+ {
+ if (!this.fontPreviewElement || !this.visible)
+ return;
+
+ this.fontPreviewElement.removeStyleClass("preview");
+
+ var measureFontSize = 50;
+ this.fontPreviewElement.style.setProperty("position", "absolute", null);
+ this.fontPreviewElement.style.setProperty("font-size", measureFontSize + "px", null);
+ this.fontPreviewElement.style.removeProperty("height");
+
+ var height = this.fontPreviewElement.offsetHeight;
+ var width = this.fontPreviewElement.offsetWidth;
+
+ var containerWidth = this.contentElement.offsetWidth;
+
+ // Subtract some padding. This should match the padding in the CSS plus room for the scrollbar.
+ containerWidth -= 40;
+
+ if (!height || !width || !containerWidth) {
+ this.fontPreviewElement.style.removeProperty("font-size");
+ this.fontPreviewElement.style.removeProperty("position");
+ this.fontPreviewElement.addStyleClass("preview");
+ return;
+ }
+
+ var lineCount = this.fontPreviewElement.getElementsByTagName("br").length + 1;
+ var realLineHeight = Math.floor(height / lineCount);
+ var fontSizeLineRatio = measureFontSize / realLineHeight;
+ var widthRatio = containerWidth / width;
+ var finalFontSize = Math.floor(realLineHeight * widthRatio * fontSizeLineRatio) - 1;
+
+ this.fontPreviewElement.style.setProperty("font-size", finalFontSize + "px", null);
+ this.fontPreviewElement.style.setProperty("height", this.fontPreviewElement.offsetHeight + "px", null);
+ this.fontPreviewElement.style.removeProperty("position");
+
+ this.fontPreviewElement.addStyleClass("preview");
+ }
+}
+
+WebInspector.FontView.prototype.__proto__ = WebInspector.ResourceView.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ImageView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ImageView.js
new file mode 100644
index 0000000..001ffdd
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ImageView.js
@@ -0,0 +1,74 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ImageView = function(resource)
+{
+ WebInspector.ResourceView.call(this, resource);
+
+ this.element.addStyleClass("image");
+
+ var container = document.createElement("div");
+ container.className = "image";
+ this.contentElement.appendChild(container);
+
+ this.imagePreviewElement = document.createElement("img");
+ this.imagePreviewElement.setAttribute("src", this.resource.url);
+
+ container.appendChild(this.imagePreviewElement);
+
+ container = document.createElement("div");
+ container.className = "info";
+ this.contentElement.appendChild(container);
+
+ var imageNameElement = document.createElement("h1");
+ imageNameElement.className = "title";
+ imageNameElement.textContent = this.resource.displayName;
+ container.appendChild(imageNameElement);
+
+ var infoListElement = document.createElement("dl");
+ infoListElement.className = "infoList";
+
+ var imageProperties = [
+ { name: WebInspector.UIString("Dimensions"), value: WebInspector.UIString("%d × %d", this.imagePreviewElement.naturalWidth, this.imagePreviewElement.height) },
+ { name: WebInspector.UIString("File size"), value: Number.bytesToString(this.resource.contentLength, WebInspector.UIString.bind(WebInspector)) },
+ { name: WebInspector.UIString("MIME type"), value: this.resource.mimeType }
+ ];
+
+ var listHTML = '';
+ for (var i = 0; i < imageProperties.length; ++i)
+ listHTML += "<dt>" + imageProperties[i].name + "</dt><dd>" + imageProperties[i].value + "</dd>";
+
+ infoListElement.innerHTML = listHTML;
+ container.appendChild(infoListElement);
+}
+
+WebInspector.ImageView.prototype = {
+
+}
+
+WebInspector.ImageView.prototype.__proto__ = WebInspector.ResourceView.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScript.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScript.js
new file mode 100644
index 0000000..b3d9566
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScript.js
@@ -0,0 +1,1133 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+var InjectedScript = {};
+
+// Called from within InspectorController on the 'inspected page' side.
+InjectedScript.reset = function()
+{
+ InjectedScript._styles = {};
+ InjectedScript._styleRules = {};
+ InjectedScript._lastStyleId = 0;
+ InjectedScript._lastStyleRuleId = 0;
+ InjectedScript._searchResults = [];
+ InjectedScript._includedInSearchResultsPropertyName = "__includedInInspectorSearchResults";
+}
+
+InjectedScript.reset();
+
+InjectedScript.dispatch = function(methodName, args)
+{
+ var result = InjectedScript[methodName].apply(InjectedScript, JSON.parse(args));
+ return JSON.stringify(result);
+}
+
+InjectedScript.getStyles = function(nodeId, authorOnly)
+{
+ var node = InjectedScript._nodeForId(nodeId);
+ if (!node)
+ return false;
+ var matchedRules = InjectedScript._window().getMatchedCSSRules(node, "", authorOnly);
+ var matchedCSSRules = [];
+ for (var i = 0; matchedRules && i < matchedRules.length; ++i)
+ matchedCSSRules.push(InjectedScript._serializeRule(matchedRules[i]));
+
+ var styleAttributes = {};
+ var attributes = node.attributes;
+ for (var i = 0; attributes && i < attributes.length; ++i) {
+ if (attributes[i].style)
+ styleAttributes[attributes[i].name] = InjectedScript._serializeStyle(attributes[i].style, true);
+ }
+ var result = {};
+ result.inlineStyle = InjectedScript._serializeStyle(node.style, true);
+ result.computedStyle = InjectedScript._serializeStyle(InjectedScript._window().getComputedStyle(node));
+ result.matchedCSSRules = matchedCSSRules;
+ result.styleAttributes = styleAttributes;
+ return result;
+}
+
+InjectedScript.getComputedStyle = function(nodeId)
+{
+ var node = InjectedScript._nodeForId(nodeId);
+ if (!node)
+ return false;
+ return InjectedScript._serializeStyle(InjectedScript._window().getComputedStyle(node));
+}
+
+InjectedScript.getInlineStyle = function(nodeId)
+{
+ var node = InjectedScript._nodeForId(nodeId);
+ if (!node)
+ return false;
+ return InjectedScript._serializeStyle(node.style, true);
+}
+
+InjectedScript.applyStyleText = function(styleId, styleText, propertyName)
+{
+ var style = InjectedScript._styles[styleId];
+ if (!style)
+ return false;
+
+ var styleTextLength = styleText.length;
+
+ // Create a new element to parse the user input CSS.
+ var parseElement = document.createElement("span");
+ parseElement.setAttribute("style", styleText);
+
+ var tempStyle = parseElement.style;
+ if (tempStyle.length || !styleTextLength) {
+ // The input was parsable or the user deleted everything, so remove the
+ // original property from the real style declaration. If this represents
+ // a shorthand remove all the longhand properties.
+ if (style.getPropertyShorthand(propertyName)) {
+ var longhandProperties = InjectedScript._getLonghandProperties(style, propertyName);
+ for (var i = 0; i < longhandProperties.length; ++i)
+ style.removeProperty(longhandProperties[i]);
+ } else
+ style.removeProperty(propertyName);
+ }
+
+ // Notify caller that the property was successfully deleted.
+ if (!styleTextLength)
+ return [null, [propertyName]];
+
+ if (!tempStyle.length)
+ return false;
+
+ // Iterate of the properties on the test element's style declaration and
+ // add them to the real style declaration. We take care to move shorthands.
+ var foundShorthands = {};
+ var changedProperties = [];
+ var uniqueProperties = InjectedScript._getUniqueStyleProperties(tempStyle);
+ for (var i = 0; i < uniqueProperties.length; ++i) {
+ var name = uniqueProperties[i];
+ var shorthand = tempStyle.getPropertyShorthand(name);
+
+ if (shorthand && shorthand in foundShorthands)
+ continue;
+
+ if (shorthand) {
+ var value = InjectedScript._getShorthandValue(tempStyle, shorthand);
+ var priority = InjectedScript._getShorthandPriority(tempStyle, shorthand);
+ foundShorthands[shorthand] = true;
+ } else {
+ var value = tempStyle.getPropertyValue(name);
+ var priority = tempStyle.getPropertyPriority(name);
+ }
+
+ // Set the property on the real style declaration.
+ style.setProperty((shorthand || name), value, priority);
+ changedProperties.push(shorthand || name);
+ }
+ return [InjectedScript._serializeStyle(style, true), changedProperties];
+}
+
+InjectedScript.setStyleText = function(style, cssText)
+{
+ style.cssText = cssText;
+}
+
+InjectedScript.toggleStyleEnabled = function(styleId, propertyName, disabled)
+{
+ var style = InjectedScript._styles[styleId];
+ if (!style)
+ return false;
+
+ if (disabled) {
+ if (!style.__disabledPropertyValues || !style.__disabledPropertyPriorities) {
+ var inspectedWindow = InjectedScript._window();
+ style.__disabledProperties = new inspectedWindow.Object;
+ style.__disabledPropertyValues = new inspectedWindow.Object;
+ style.__disabledPropertyPriorities = new inspectedWindow.Object;
+ }
+
+ style.__disabledPropertyValues[propertyName] = style.getPropertyValue(propertyName);
+ style.__disabledPropertyPriorities[propertyName] = style.getPropertyPriority(propertyName);
+
+ if (style.getPropertyShorthand(propertyName)) {
+ var longhandProperties = InjectedScript._getLonghandProperties(style, propertyName);
+ for (var i = 0; i < longhandProperties.length; ++i) {
+ style.__disabledProperties[longhandProperties[i]] = true;
+ style.removeProperty(longhandProperties[i]);
+ }
+ } else {
+ style.__disabledProperties[propertyName] = true;
+ style.removeProperty(propertyName);
+ }
+ } else if (style.__disabledProperties && style.__disabledProperties[propertyName]) {
+ var value = style.__disabledPropertyValues[propertyName];
+ var priority = style.__disabledPropertyPriorities[propertyName];
+
+ style.setProperty(propertyName, value, priority);
+ delete style.__disabledProperties[propertyName];
+ delete style.__disabledPropertyValues[propertyName];
+ delete style.__disabledPropertyPriorities[propertyName];
+ }
+ return InjectedScript._serializeStyle(style, true);
+}
+
+InjectedScript.applyStyleRuleText = function(ruleId, newContent, selectedNodeId)
+{
+ var rule = InjectedScript._styleRules[ruleId];
+ if (!rule)
+ return false;
+
+ var selectedNode = InjectedScript._nodeForId(selectedNodeId);
+
+ try {
+ var stylesheet = rule.parentStyleSheet;
+ stylesheet.addRule(newContent);
+ var newRule = stylesheet.cssRules[stylesheet.cssRules.length - 1];
+ newRule.style.cssText = rule.style.cssText;
+
+ var parentRules = stylesheet.cssRules;
+ for (var i = 0; i < parentRules.length; ++i) {
+ if (parentRules[i] === rule) {
+ rule.parentStyleSheet.removeRule(i);
+ break;
+ }
+ }
+
+ return [InjectedScript._serializeRule(newRule), InjectedScript._doesSelectorAffectNode(newContent, selectedNode)];
+ } catch(e) {
+ // Report invalid syntax.
+ return false;
+ }
+}
+
+InjectedScript.addStyleSelector = function(newContent, selectedNodeId)
+{
+ var stylesheet = InjectedScript.stylesheet;
+ if (!stylesheet) {
+ var inspectedDocument = InjectedScript._window().document;
+ var head = inspectedDocument.getElementsByTagName("head")[0];
+ var styleElement = inspectedDocument.createElement("style");
+ styleElement.type = "text/css";
+ head.appendChild(styleElement);
+ stylesheet = inspectedDocument.styleSheets[inspectedDocument.styleSheets.length - 1];
+ InjectedScript.stylesheet = stylesheet;
+ }
+
+ try {
+ stylesheet.addRule(newContent);
+ } catch (e) {
+ // Invalid Syntax for a Selector
+ return false;
+ }
+
+ var selectedNode = InjectedScript._nodeForId(selectedNodeId);
+ var rule = stylesheet.cssRules[stylesheet.cssRules.length - 1];
+ rule.__isViaInspector = true;
+
+ return [ InjectedScript._serializeRule(rule), InjectedScript._doesSelectorAffectNode(newContent, selectedNode) ];
+}
+
+InjectedScript._doesSelectorAffectNode = function(selectorText, node)
+{
+ if (!node)
+ return false;
+ var nodes = node.ownerDocument.querySelectorAll(selectorText);
+ for (var i = 0; i < nodes.length; ++i) {
+ if (nodes[i] === node) {
+ return true;
+ }
+ }
+ return false;
+}
+
+InjectedScript.setStyleProperty = function(styleId, name, value)
+{
+ var style = InjectedScript._styles[styleId];
+ if (!style)
+ return false;
+
+ style.setProperty(name, value, "");
+ return true;
+}
+
+InjectedScript._serializeRule = function(rule)
+{
+ var parentStyleSheet = rule.parentStyleSheet;
+
+ var ruleValue = {};
+ ruleValue.selectorText = rule.selectorText;
+ if (parentStyleSheet) {
+ ruleValue.parentStyleSheet = {};
+ ruleValue.parentStyleSheet.href = parentStyleSheet.href;
+ }
+ ruleValue.isUserAgent = parentStyleSheet && !parentStyleSheet.ownerNode && !parentStyleSheet.href;
+ ruleValue.isUser = parentStyleSheet && parentStyleSheet.ownerNode && parentStyleSheet.ownerNode.nodeName == "#document";
+ ruleValue.isViaInspector = !!rule.__isViaInspector;
+
+ // Bind editable scripts only.
+ var doBind = !ruleValue.isUserAgent && !ruleValue.isUser;
+ ruleValue.style = InjectedScript._serializeStyle(rule.style, doBind);
+
+ if (doBind) {
+ if (!rule.id) {
+ rule.id = InjectedScript._lastStyleRuleId++;
+ InjectedScript._styleRules[rule.id] = rule;
+ }
+ ruleValue.id = rule.id;
+ }
+ return ruleValue;
+}
+
+InjectedScript._serializeStyle = function(style, doBind)
+{
+ var result = {};
+ result.width = style.width;
+ result.height = style.height;
+ result.__disabledProperties = style.__disabledProperties;
+ result.__disabledPropertyValues = style.__disabledPropertyValues;
+ result.__disabledPropertyPriorities = style.__disabledPropertyPriorities;
+ result.properties = [];
+ result.shorthandValues = {};
+ var foundShorthands = {};
+ for (var i = 0; i < style.length; ++i) {
+ var property = {};
+ var name = style[i];
+ property.name = name;
+ property.priority = style.getPropertyPriority(name);
+ property.implicit = style.isPropertyImplicit(name);
+ var shorthand = style.getPropertyShorthand(name);
+ property.shorthand = shorthand;
+ if (shorthand && !(shorthand in foundShorthands)) {
+ foundShorthands[shorthand] = true;
+ result.shorthandValues[shorthand] = InjectedScript._getShorthandValue(style, shorthand);
+ }
+ property.value = style.getPropertyValue(name);
+ result.properties.push(property);
+ }
+ result.uniqueStyleProperties = InjectedScript._getUniqueStyleProperties(style);
+
+ if (doBind) {
+ if (!style.id) {
+ style.id = InjectedScript._lastStyleId++;
+ InjectedScript._styles[style.id] = style;
+ }
+ result.id = style.id;
+ }
+ return result;
+}
+
+InjectedScript._getUniqueStyleProperties = function(style)
+{
+ var properties = [];
+ var foundProperties = {};
+
+ for (var i = 0; i < style.length; ++i) {
+ var property = style[i];
+ if (property in foundProperties)
+ continue;
+ foundProperties[property] = true;
+ properties.push(property);
+ }
+
+ return properties;
+}
+
+
+InjectedScript._getLonghandProperties = function(style, shorthandProperty)
+{
+ var properties = [];
+ var foundProperties = {};
+
+ for (var i = 0; i < style.length; ++i) {
+ var individualProperty = style[i];
+ if (individualProperty in foundProperties || style.getPropertyShorthand(individualProperty) !== shorthandProperty)
+ continue;
+ foundProperties[individualProperty] = true;
+ properties.push(individualProperty);
+ }
+
+ return properties;
+}
+
+InjectedScript._getShorthandValue = function(style, shorthandProperty)
+{
+ var value = style.getPropertyValue(shorthandProperty);
+ if (!value) {
+ // Some shorthands (like border) return a null value, so compute a shorthand value.
+ // FIXME: remove this when http://bugs.webkit.org/show_bug.cgi?id=15823 is fixed.
+
+ var foundProperties = {};
+ for (var i = 0; i < style.length; ++i) {
+ var individualProperty = style[i];
+ if (individualProperty in foundProperties || style.getPropertyShorthand(individualProperty) !== shorthandProperty)
+ continue;
+
+ var individualValue = style.getPropertyValue(individualProperty);
+ if (style.isPropertyImplicit(individualProperty) || individualValue === "initial")
+ continue;
+
+ foundProperties[individualProperty] = true;
+
+ if (!value)
+ value = "";
+ else if (value.length)
+ value += " ";
+ value += individualValue;
+ }
+ }
+ return value;
+}
+
+InjectedScript._getShorthandPriority = function(style, shorthandProperty)
+{
+ var priority = style.getPropertyPriority(shorthandProperty);
+ if (!priority) {
+ for (var i = 0; i < style.length; ++i) {
+ var individualProperty = style[i];
+ if (style.getPropertyShorthand(individualProperty) !== shorthandProperty)
+ continue;
+ priority = style.getPropertyPriority(individualProperty);
+ break;
+ }
+ }
+ return priority;
+}
+
+InjectedScript.getPrototypes = function(nodeId)
+{
+ var node = InjectedScript._nodeForId(nodeId);
+ if (!node)
+ return false;
+
+ var result = [];
+ for (var prototype = node; prototype; prototype = prototype.__proto__) {
+ var title = Object.describe(prototype);
+ if (title.match(/Prototype$/)) {
+ title = title.replace(/Prototype$/, "");
+ }
+ result.push(title);
+ }
+ return result;
+}
+
+InjectedScript.getProperties = function(objectProxy, ignoreHasOwnProperty)
+{
+ var object = InjectedScript._resolveObject(objectProxy);
+ if (!object)
+ return false;
+
+ var properties = [];
+
+ // Go over properties, prepare results.
+ for (var propertyName in object) {
+ if (!ignoreHasOwnProperty && "hasOwnProperty" in object && !object.hasOwnProperty(propertyName))
+ continue;
+
+ var property = {};
+ property.name = propertyName;
+ property.parentObjectProxy = objectProxy;
+ var isGetter = object["__lookupGetter__"] && object.__lookupGetter__(propertyName);
+ if (!property.isGetter) {
+ var childObject = object[propertyName];
+ var childObjectProxy = new InjectedScript.createProxyObject(childObject, objectProxy.objectId, true);
+ childObjectProxy.path = objectProxy.path ? objectProxy.path.slice() : [];
+ childObjectProxy.path.push(propertyName);
+ childObjectProxy.protoDepth = objectProxy.protoDepth || 0;
+ property.value = childObjectProxy;
+ } else {
+ // FIXME: this should show something like "getter" (bug 16734).
+ property.value = { description: "\u2014" }; // em dash
+ property.isGetter = true;
+ }
+ properties.push(property);
+ }
+ return properties;
+}
+
+InjectedScript.setPropertyValue = function(objectProxy, propertyName, expression)
+{
+ var object = InjectedScript._resolveObject(objectProxy);
+ if (!object)
+ return false;
+
+ var expressionLength = expression.length;
+ if (!expressionLength) {
+ delete object[propertyName];
+ return !(propertyName in object);
+ }
+
+ try {
+ // Surround the expression in parenthesis so the result of the eval is the result
+ // of the whole expression not the last potential sub-expression.
+
+ // There is a regression introduced here: eval is now happening against global object,
+ // not call frame while on a breakpoint.
+ // TODO: bring evaluation against call frame back.
+ var result = InjectedScript._window().eval("(" + expression + ")");
+ // Store the result in the property.
+ object[propertyName] = result;
+ return true;
+ } catch(e) {
+ try {
+ var result = InjectedScript._window().eval("\"" + expression.escapeCharacters("\"") + "\"");
+ object[propertyName] = result;
+ return true;
+ } catch(e) {
+ return false;
+ }
+ }
+}
+
+
+InjectedScript.getCompletions = function(expression, includeInspectorCommandLineAPI)
+{
+ var props = {};
+ try {
+ var expressionResult = InjectedScript._evaluateOn(InjectedScript._window().eval, InjectedScript._window(), expression);
+ for (var prop in expressionResult)
+ props[prop] = true;
+ if (includeInspectorCommandLineAPI)
+ for (var prop in InjectedScript._window()._inspectorCommandLineAPI)
+ if (prop.charAt(0) !== '_')
+ props[prop] = true;
+ } catch(e) {
+ }
+ return props;
+}
+
+
+InjectedScript.evaluate = function(expression)
+{
+ var result = {};
+ try {
+ var value = InjectedScript._evaluateOn(InjectedScript._window().eval, InjectedScript._window(), expression);
+ if (value === null)
+ return { value: null };
+ if (Object.type(value) === "error") {
+ result.value = Object.describe(value);
+ result.isException = true;
+ return result;
+ }
+
+ var wrapper = InspectorController.wrapObject(value);
+ if (typeof wrapper === "object" && wrapper.exception) {
+ result.value = wrapper.exception;
+ result.isException = true;
+ } else {
+ result.value = wrapper;
+ }
+ } catch (e) {
+ result.value = e.toString();
+ result.isException = true;
+ }
+ return result;
+}
+
+InjectedScript._evaluateOn = function(evalFunction, object, expression)
+{
+ InjectedScript._ensureCommandLineAPIInstalled();
+ // Surround the expression in with statements to inject our command line API so that
+ // the window object properties still take more precedent than our API functions.
+ expression = "with (window._inspectorCommandLineAPI) { with (window) { " + expression + " } }";
+ return evalFunction.call(object, expression);
+}
+
+InjectedScript.addInspectedNode = function(nodeId)
+{
+ var node = InjectedScript._nodeForId(nodeId);
+ if (!node)
+ return false;
+
+ InjectedScript._ensureCommandLineAPIInstalled();
+ var inspectedNodes = InjectedScript._window()._inspectorCommandLineAPI._inspectedNodes;
+ inspectedNodes.unshift(node);
+ if (inspectedNodes.length >= 5)
+ inspectedNodes.pop();
+ return true;
+}
+
+InjectedScript.performSearch = function(whitespaceTrimmedQuery)
+{
+ var tagNameQuery = whitespaceTrimmedQuery;
+ var attributeNameQuery = whitespaceTrimmedQuery;
+ var startTagFound = (tagNameQuery.indexOf("<") === 0);
+ var endTagFound = (tagNameQuery.lastIndexOf(">") === (tagNameQuery.length - 1));
+
+ if (startTagFound || endTagFound) {
+ var tagNameQueryLength = tagNameQuery.length;
+ tagNameQuery = tagNameQuery.substring((startTagFound ? 1 : 0), (endTagFound ? (tagNameQueryLength - 1) : tagNameQueryLength));
+ }
+
+ // Check the tagNameQuery is it is a possibly valid tag name.
+ if (!/^[a-zA-Z0-9\-_:]+$/.test(tagNameQuery))
+ tagNameQuery = null;
+
+ // Check the attributeNameQuery is it is a possibly valid tag name.
+ if (!/^[a-zA-Z0-9\-_:]+$/.test(attributeNameQuery))
+ attributeNameQuery = null;
+
+ const escapedQuery = whitespaceTrimmedQuery.escapeCharacters("'");
+ const escapedTagNameQuery = (tagNameQuery ? tagNameQuery.escapeCharacters("'") : null);
+ const escapedWhitespaceTrimmedQuery = whitespaceTrimmedQuery.escapeCharacters("'");
+ const searchResultsProperty = InjectedScript._includedInSearchResultsPropertyName;
+
+ function addNodesToResults(nodes, length, getItem)
+ {
+ if (!length)
+ return;
+
+ var nodeIds = [];
+ for (var i = 0; i < length; ++i) {
+ var node = getItem.call(nodes, i);
+ // Skip this node if it already has the property.
+ if (searchResultsProperty in node)
+ continue;
+
+ if (!InjectedScript._searchResults.length) {
+ InjectedScript._currentSearchResultIndex = 0;
+ }
+
+ node[searchResultsProperty] = true;
+ InjectedScript._searchResults.push(node);
+ var nodeId = InspectorController.pushNodePathToFrontend(node, false);
+ nodeIds.push(nodeId);
+ }
+ InspectorController.addNodesToSearchResult(nodeIds.join(","));
+ }
+
+ function matchExactItems(doc)
+ {
+ matchExactId.call(this, doc);
+ matchExactClassNames.call(this, doc);
+ matchExactTagNames.call(this, doc);
+ matchExactAttributeNames.call(this, doc);
+ }
+
+ function matchExactId(doc)
+ {
+ const result = doc.__proto__.getElementById.call(doc, whitespaceTrimmedQuery);
+ addNodesToResults.call(this, result, (result ? 1 : 0), function() { return this });
+ }
+
+ function matchExactClassNames(doc)
+ {
+ const result = doc.__proto__.getElementsByClassName.call(doc, whitespaceTrimmedQuery);
+ addNodesToResults.call(this, result, result.length, result.item);
+ }
+
+ function matchExactTagNames(doc)
+ {
+ if (!tagNameQuery)
+ return;
+ const result = doc.__proto__.getElementsByTagName.call(doc, tagNameQuery);
+ addNodesToResults.call(this, result, result.length, result.item);
+ }
+
+ function matchExactAttributeNames(doc)
+ {
+ if (!attributeNameQuery)
+ return;
+ const result = doc.__proto__.querySelectorAll.call(doc, "[" + attributeNameQuery + "]");
+ addNodesToResults.call(this, result, result.length, result.item);
+ }
+
+ function matchPartialTagNames(doc)
+ {
+ if (!tagNameQuery)
+ return;
+ const result = doc.__proto__.evaluate.call(doc, "//*[contains(name(), '" + escapedTagNameQuery + "')]", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE);
+ addNodesToResults.call(this, result, result.snapshotLength, result.snapshotItem);
+ }
+
+ function matchStartOfTagNames(doc)
+ {
+ if (!tagNameQuery)
+ return;
+ const result = doc.__proto__.evaluate.call(doc, "//*[starts-with(name(), '" + escapedTagNameQuery + "')]", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE);
+ addNodesToResults.call(this, result, result.snapshotLength, result.snapshotItem);
+ }
+
+ function matchPartialTagNamesAndAttributeValues(doc)
+ {
+ if (!tagNameQuery) {
+ matchPartialAttributeValues.call(this, doc);
+ return;
+ }
+
+ const result = doc.__proto__.evaluate.call(doc, "//*[contains(name(), '" + escapedTagNameQuery + "') or contains(@*, '" + escapedQuery + "')]", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE);
+ addNodesToResults.call(this, result, result.snapshotLength, result.snapshotItem);
+ }
+
+ function matchPartialAttributeValues(doc)
+ {
+ const result = doc.__proto__.evaluate.call(doc, "//*[contains(@*, '" + escapedQuery + "')]", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE);
+ addNodesToResults.call(this, result, result.snapshotLength, result.snapshotItem);
+ }
+
+ function matchStyleSelector(doc)
+ {
+ const result = doc.__proto__.querySelectorAll.call(doc, whitespaceTrimmedQuery);
+ addNodesToResults.call(this, result, result.length, result.item);
+ }
+
+ function matchPlainText(doc)
+ {
+ const result = doc.__proto__.evaluate.call(doc, "//text()[contains(., '" + escapedQuery + "')] | //comment()[contains(., '" + escapedQuery + "')]", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE);
+ addNodesToResults.call(this, result, result.snapshotLength, result.snapshotItem);
+ }
+
+ function matchXPathQuery(doc)
+ {
+ const result = doc.__proto__.evaluate.call(doc, whitespaceTrimmedQuery, doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE);
+ addNodesToResults.call(this, result, result.snapshotLength, result.snapshotItem);
+ }
+
+ function finishedSearching()
+ {
+ // Remove the searchResultsProperty now that the search is finished.
+ for (var i = 0; i < InjectedScript._searchResults.length; ++i)
+ delete InjectedScript._searchResults[i][searchResultsProperty];
+ }
+
+ const mainFrameDocument = InjectedScript._window().document;
+ const searchDocuments = [mainFrameDocument];
+ var searchFunctions;
+ if (tagNameQuery && startTagFound && endTagFound)
+ searchFunctions = [matchExactTagNames, matchPlainText];
+ else if (tagNameQuery && startTagFound)
+ searchFunctions = [matchStartOfTagNames, matchPlainText];
+ else if (tagNameQuery && endTagFound) {
+ // FIXME: we should have a matchEndOfTagNames search function if endTagFound is true but not startTagFound.
+ // This requires ends-with() support in XPath, WebKit only supports starts-with() and contains().
+ searchFunctions = [matchPartialTagNames, matchPlainText];
+ } else if (whitespaceTrimmedQuery === "//*" || whitespaceTrimmedQuery === "*") {
+ // These queries will match every node. Matching everything isn't useful and can be slow for large pages,
+ // so limit the search functions list to plain text and attribute matching.
+ searchFunctions = [matchPartialAttributeValues, matchPlainText];
+ } else
+ searchFunctions = [matchExactItems, matchStyleSelector, matchPartialTagNamesAndAttributeValues, matchPlainText, matchXPathQuery];
+
+ // Find all frames, iframes and object elements to search their documents.
+ const querySelectorAllFunction = InjectedScript._window().Document.prototype.querySelectorAll;
+ const subdocumentResult = querySelectorAllFunction.call(mainFrameDocument, "iframe, frame, object");
+
+ for (var i = 0; i < subdocumentResult.length; ++i) {
+ var element = subdocumentResult.item(i);
+ if (element.contentDocument)
+ searchDocuments.push(element.contentDocument);
+ }
+
+ const panel = InjectedScript;
+ var documentIndex = 0;
+ var searchFunctionIndex = 0;
+ var chunkIntervalIdentifier = null;
+
+ // Split up the work into chunks so we don't block the UI thread while processing.
+
+ function processChunk()
+ {
+ var searchDocument = searchDocuments[documentIndex];
+ var searchFunction = searchFunctions[searchFunctionIndex];
+
+ if (++searchFunctionIndex > searchFunctions.length) {
+ searchFunction = searchFunctions[0];
+ searchFunctionIndex = 0;
+
+ if (++documentIndex > searchDocuments.length) {
+ if (panel._currentSearchChunkIntervalIdentifier === chunkIntervalIdentifier)
+ delete panel._currentSearchChunkIntervalIdentifier;
+ clearInterval(chunkIntervalIdentifier);
+ finishedSearching.call(panel);
+ return;
+ }
+
+ searchDocument = searchDocuments[documentIndex];
+ }
+
+ if (!searchDocument || !searchFunction)
+ return;
+
+ try {
+ searchFunction.call(panel, searchDocument);
+ } catch(err) {
+ // ignore any exceptions. the query might be malformed, but we allow that.
+ }
+ }
+
+ processChunk();
+
+ chunkIntervalIdentifier = setInterval(processChunk, 25);
+ InjectedScript._currentSearchChunkIntervalIdentifier = chunkIntervalIdentifier;
+ return true;
+}
+
+InjectedScript.searchCanceled = function()
+{
+ if (InjectedScript._searchResults) {
+ const searchResultsProperty = InjectedScript._includedInSearchResultsPropertyName;
+ for (var i = 0; i < this._searchResults.length; ++i) {
+ var node = this._searchResults[i];
+
+ // Remove the searchResultsProperty since there might be an unfinished search.
+ delete node[searchResultsProperty];
+ }
+ }
+
+ if (InjectedScript._currentSearchChunkIntervalIdentifier) {
+ clearInterval(InjectedScript._currentSearchChunkIntervalIdentifier);
+ delete InjectedScript._currentSearchChunkIntervalIdentifier;
+ }
+ InjectedScript._searchResults = [];
+ return true;
+}
+
+InjectedScript.openInInspectedWindow = function(url)
+{
+ InjectedScript._window().open(url);
+}
+
+InjectedScript.getCallFrames = function()
+{
+ var callFrame = InspectorController.currentCallFrame();
+ if (!callFrame)
+ return false;
+
+ var result = [];
+ var depth = 0;
+ do {
+ result.push(new InjectedScript.CallFrameProxy(depth++, callFrame));
+ callFrame = callFrame.caller;
+ } while (callFrame);
+ return result;
+}
+
+InjectedScript.evaluateInCallFrame = function(callFrameId, code)
+{
+ var callFrame = InjectedScript._callFrameForId(callFrameId);
+ if (!callFrame)
+ return false;
+ return InjectedScript._evaluateOn(callFrame.evaluate, callFrame, code);
+}
+
+InjectedScript._callFrameForId = function(id)
+{
+ var callFrame = InspectorController.currentCallFrame();
+ while (--id >= 0 && callFrame)
+ callFrame = callFrame.caller;
+ return callFrame;
+}
+
+InjectedScript._clearConsoleMessages = function()
+{
+ InspectorController.clearMessages(true);
+}
+
+InjectedScript._inspectObject = function(o)
+{
+ if (arguments.length === 0)
+ return;
+
+ var inspectedWindow = InjectedScript._window();
+ inspectedWindow.console.log(o);
+ if (Object.type(o) === "node") {
+ InspectorController.pushNodePathToFrontend(o, true);
+ } else {
+ switch (Object.describe(o)) {
+ case "Database":
+ InspectorController.selectDatabase(o);
+ break;
+ case "Storage":
+ InspectorController.selectDOMStorage(o);
+ break;
+ }
+ }
+}
+
+InjectedScript._ensureCommandLineAPIInstalled = function(inspectedWindow)
+{
+ var inspectedWindow = InjectedScript._window();
+ if (inspectedWindow._inspectorCommandLineAPI)
+ return;
+
+ inspectedWindow.eval("window._inspectorCommandLineAPI = { \
+ $: function() { return document.getElementById.apply(document, arguments) }, \
+ $$: function() { return document.querySelectorAll.apply(document, arguments) }, \
+ $x: function(xpath, context) { \
+ var nodes = []; \
+ try { \
+ var doc = context || document; \
+ var results = doc.evaluate(xpath, doc, null, XPathResult.ANY_TYPE, null); \
+ var node; \
+ while (node = results.iterateNext()) nodes.push(node); \
+ } catch (e) {} \
+ return nodes; \
+ }, \
+ dir: function() { return console.dir.apply(console, arguments) }, \
+ dirxml: function() { return console.dirxml.apply(console, arguments) }, \
+ keys: function(o) { var a = []; for (var k in o) a.push(k); return a; }, \
+ values: function(o) { var a = []; for (var k in o) a.push(o[k]); return a; }, \
+ profile: function() { return console.profile.apply(console, arguments) }, \
+ profileEnd: function() { return console.profileEnd.apply(console, arguments) }, \
+ _inspectedNodes: [], \
+ get $0() { return _inspectorCommandLineAPI._inspectedNodes[0] }, \
+ get $1() { return _inspectorCommandLineAPI._inspectedNodes[1] }, \
+ get $2() { return _inspectorCommandLineAPI._inspectedNodes[2] }, \
+ get $3() { return _inspectorCommandLineAPI._inspectedNodes[3] }, \
+ get $4() { return _inspectorCommandLineAPI._inspectedNodes[4] } \
+ };");
+
+ inspectedWindow._inspectorCommandLineAPI.clear = InspectorController.wrapCallback(InjectedScript._clearConsoleMessages);
+ inspectedWindow._inspectorCommandLineAPI.inspect = InspectorController.wrapCallback(InjectedScript._inspectObject);
+}
+
+InjectedScript._resolveObject = function(objectProxy)
+{
+ var object = InjectedScript._objectForId(objectProxy.objectId);
+ var path = objectProxy.path;
+ var protoDepth = objectProxy.protoDepth;
+
+ // Follow the property path.
+ for (var i = 0; object && path && i < path.length; ++i)
+ object = object[path[i]];
+
+ // Get to the necessary proto layer.
+ for (var i = 0; object && protoDepth && i < protoDepth; ++i)
+ object = object.__proto__;
+
+ return object;
+}
+
+InjectedScript._window = function()
+{
+ // TODO: replace with 'return window;' once this script is injected into
+ // the page's context.
+ return InspectorController.inspectedWindow();
+}
+
+InjectedScript._nodeForId = function(nodeId)
+{
+ if (!nodeId)
+ return null;
+ return InspectorController.nodeForId(nodeId);
+}
+
+InjectedScript._objectForId = function(objectId)
+{
+ // There are three types of object ids used:
+ // - numbers point to DOM Node via the InspectorDOMAgent mapping
+ // - strings point to console objects cached in InspectorController for lazy evaluation upon them
+ // - objects contain complex ids and are currently used for scoped objects
+ if (typeof objectId === "number") {
+ return InjectedScript._nodeForId(objectId);
+ } else if (typeof objectId === "string") {
+ return InspectorController.unwrapObject(objectId);
+ } else if (typeof objectId === "object") {
+ var callFrame = InjectedScript._callFrameForId(objectId.callFrame);
+ if (objectId.thisObject)
+ return callFrame.thisObject;
+ else
+ return callFrame.scopeChain[objectId.chainIndex];
+ }
+ return objectId;
+}
+
+InjectedScript.pushNodeToFrontend = function(objectProxy)
+{
+ var object = InjectedScript._resolveObject(objectProxy);
+ if (!object || Object.type(object) !== "node")
+ return false;
+ return InspectorController.pushNodePathToFrontend(object, false);
+}
+
+// Called from within InspectorController on the 'inspected page' side.
+InjectedScript.createProxyObject = function(object, objectId, abbreviate)
+{
+ var result = {};
+ result.objectId = objectId;
+ result.type = Object.type(object);
+
+ var type = typeof object;
+ if ((type === "object" && object !== null) || type === "function") {
+ for (var subPropertyName in object) {
+ result.hasChildren = true;
+ break;
+ }
+ }
+ try {
+ result.description = Object.describe(object, abbreviate);
+ } catch (e) {
+ result.exception = e.toString();
+ }
+ return result;
+}
+
+InjectedScript.CallFrameProxy = function(id, callFrame)
+{
+ this.id = id;
+ this.type = callFrame.type;
+ this.functionName = (this.type === "function" ? callFrame.functionName : "");
+ this.sourceID = callFrame.sourceID;
+ this.line = callFrame.line;
+ this.scopeChain = this._wrapScopeChain(callFrame);
+}
+
+InjectedScript.CallFrameProxy.prototype = {
+ _wrapScopeChain: function(callFrame)
+ {
+ var foundLocalScope = false;
+ var scopeChain = callFrame.scopeChain;
+ var scopeChainProxy = [];
+ for (var i = 0; i < scopeChain.length; ++i) {
+ var scopeObject = scopeChain[i];
+ var scopeObjectProxy = InjectedScript.createProxyObject(scopeObject, { callFrame: this.id, chainIndex: i });
+
+ if (Object.prototype.toString.call(scopeObject) === "[object JSActivation]") {
+ if (!foundLocalScope)
+ scopeObjectProxy.thisObject = InjectedScript.createProxyObject(callFrame.thisObject, { callFrame: this.id, thisObject: true });
+ else
+ scopeObjectProxy.isClosure = true;
+ foundLocalScope = true;
+ scopeObjectProxy.isLocal = true;
+ } else if (foundLocalScope && scopeObject instanceof InjectedScript._window().Element)
+ scopeObjectProxy.isElement = true;
+ else if (foundLocalScope && scopeObject instanceof InjectedScript._window().Document)
+ scopeObjectProxy.isDocument = true;
+ else if (!foundLocalScope)
+ scopeObjectProxy.isWithBlock = true;
+ scopeObjectProxy.properties = [];
+ try {
+ for (var propertyName in scopeObject)
+ scopeObjectProxy.properties.push(propertyName);
+ } catch (e) {
+ }
+ scopeChainProxy.push(scopeObjectProxy);
+ }
+ return scopeChainProxy;
+ }
+}
+
+Object.type = function(obj)
+{
+ if (obj === null)
+ return "null";
+
+ var type = typeof obj;
+ if (type !== "object" && type !== "function")
+ return type;
+
+ var win = InjectedScript._window();
+
+ if (obj instanceof win.Node)
+ return (obj.nodeType === undefined ? type : "node");
+ if (obj instanceof win.String)
+ return "string";
+ if (obj instanceof win.Array)
+ return "array";
+ if (obj instanceof win.Boolean)
+ return "boolean";
+ if (obj instanceof win.Number)
+ return "number";
+ if (obj instanceof win.Date)
+ return "date";
+ if (obj instanceof win.RegExp)
+ return "regexp";
+ if (obj instanceof win.Error)
+ return "error";
+ return type;
+}
+
+Object.hasProperties = function(obj)
+{
+ if (typeof obj === "undefined" || typeof obj === "null")
+ return false;
+ for (var name in obj)
+ return true;
+ return false;
+}
+
+Object.describe = function(obj, abbreviated)
+{
+ var type1 = Object.type(obj);
+ var type2 = Object.className(obj);
+
+ switch (type1) {
+ case "object":
+ case "node":
+ return type2;
+ case "array":
+ return "[" + obj.toString() + "]";
+ case "string":
+ if (obj.length > 100)
+ return "\"" + obj.substring(0, 100) + "\u2026\"";
+ return "\"" + obj + "\"";
+ case "function":
+ var objectText = String(obj);
+ if (!/^function /.test(objectText))
+ objectText = (type2 == "object") ? type1 : type2;
+ else if (abbreviated)
+ objectText = /.*/.exec(obj)[0].replace(/ +$/g, "");
+ return objectText;
+ case "regexp":
+ return String(obj).replace(/([\\\/])/g, "\\$1").replace(/\\(\/[gim]*)$/, "$1").substring(1);
+ default:
+ return String(obj);
+ }
+}
+
+Object.className = function(obj)
+{
+ return Object.prototype.toString.call(obj).replace(/^\[object (.*)\]$/i, "$1")
+}
+
+// Although Function.prototype.bind and String.prototype.escapeCharacters are defined in utilities.js they will soon become
+// unavailable in the InjectedScript context. So we define them here for the local use.
+// TODO: remove this comment once InjectedScript runs in a separate context.
+Function.prototype.bind = function(thisObject)
+{
+ var func = this;
+ var args = Array.prototype.slice.call(arguments, 1);
+ return function() { return func.apply(thisObject, args.concat(Array.prototype.slice.call(arguments, 0))) };
+}
+
+String.prototype.escapeCharacters = function(chars)
+{
+ var foundChar = false;
+ for (var i = 0; i < chars.length; ++i) {
+ if (this.indexOf(chars.charAt(i)) !== -1) {
+ foundChar = true;
+ break;
+ }
+ }
+
+ if (!foundChar)
+ return this;
+
+ var result = "";
+ for (var i = 0; i < this.length; ++i) {
+ if (chars.indexOf(this.charAt(i)) !== -1)
+ result += "\\";
+ result += this.charAt(i);
+ }
+
+ return result;
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScriptAccess.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScriptAccess.js
new file mode 100644
index 0000000..da85d03
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/InjectedScriptAccess.js
@@ -0,0 +1,78 @@
+/*
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+var InjectedScriptAccess = {};
+
+InjectedScriptAccess._installHandler = function(methodName)
+{
+ InjectedScriptAccess[methodName] = function()
+ {
+ var allArgs = Array.prototype.slice.call(arguments);
+ var callback = allArgs[allArgs.length - 1];
+ var argsString = JSON.stringify(Array.prototype.slice.call(allArgs, 0, allArgs.length - 1));
+
+ function myCallback(result, isException)
+ {
+ if (!isException)
+ callback(JSON.parse(result));
+ else
+ WebInspector.console.addMessage(new WebInspector.ConsoleTextMessage("Error dispatching: " + methodName));
+ }
+ var callId = WebInspector.Callback.wrap(myCallback);
+ InspectorController.dispatchOnInjectedScript(callId, methodName, argsString);
+ };
+}
+
+InjectedScriptAccess._installHandler("getStyles");
+InjectedScriptAccess._installHandler("getComputedStyle");
+InjectedScriptAccess._installHandler("getInlineStyle");
+InjectedScriptAccess._installHandler("applyStyleText");
+InjectedScriptAccess._installHandler("setStyleText");
+InjectedScriptAccess._installHandler("toggleStyleEnabled");
+InjectedScriptAccess._installHandler("applyStyleRuleText");
+InjectedScriptAccess._installHandler("addStyleSelector");
+InjectedScriptAccess._installHandler("setStyleProperty");
+InjectedScriptAccess._installHandler("getPrototypes");
+InjectedScriptAccess._installHandler("getProperties");
+InjectedScriptAccess._installHandler("setPropertyValue");
+InjectedScriptAccess._installHandler("getCompletions");
+InjectedScriptAccess._installHandler("evaluate");
+InjectedScriptAccess._installHandler("addInspectedNode");
+InjectedScriptAccess._installHandler("pushNodeToFrontend");
+InjectedScriptAccess._installHandler("evaluate");
+InjectedScriptAccess._installHandler("addInspectedNode");
+InjectedScriptAccess._installHandler("pushNodeToFrontend");
+InjectedScriptAccess._installHandler("performSearch");
+InjectedScriptAccess._installHandler("searchCanceled");
+InjectedScriptAccess._installHandler("openInInspectedWindow");
+InjectedScriptAccess._installHandler("evaluateInCallFrame");
+
+WebInspector.didDispatchOnInjectedScript = WebInspector.Callback.processCallback;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/KeyboardShortcut.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/KeyboardShortcut.js
new file mode 100644
index 0000000..ed28a48
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/KeyboardShortcut.js
@@ -0,0 +1,108 @@
+/*
+ * Copyright (C) 2009 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.KeyboardShortcut = function()
+{
+};
+
+/**
+ * Constants for encoding modifier key set as a bit mask.
+ * @see #_makeKeyFromCodeAndModifiers
+ */
+WebInspector.KeyboardShortcut.Modifiers = {
+ None: 0, // Constant for empty modifiers set.
+ Shift: 1,
+ Ctrl: 2,
+ Alt: 4,
+ Meta: 8 // Command key on Mac, Win key on other platforms.
+};
+
+WebInspector.KeyboardShortcut.KeyCodes = {
+ Esc: 27,
+ Space: 32,
+ PageUp: 33, // also NUM_NORTH_EAST
+ PageDown: 34, // also NUM_SOUTH_EAST
+ End: 35, // also NUM_SOUTH_WEST
+ Home: 36, // also NUM_NORTH_WEST
+ Left: 37, // also NUM_WEST
+ Up: 38, // also NUM_NORTH
+ Right: 39, // also NUM_EAST
+ Down: 40, // also NUM_SOUTH
+ F1: 112,
+ F2: 113,
+ F3: 114,
+ F4: 115,
+ F5: 116,
+ F6: 117,
+ F7: 118,
+ F8: 119,
+ F9: 120,
+ F10: 121,
+ F11: 122,
+ F12: 123,
+ Semicolon: 186, // ;
+ Comma: 188, // ,
+ Period: 190, // .
+ Slash: 191, // /
+ Apostrophe: 192, // `
+ SingleQuote: 222, // '
+};
+
+/**
+ * Creates a number encoding keyCode in the lower 8 bits and modifiers mask in the higher 8 bits.
+ * It is usefull for matching pressed keys.
+ * @param {number} keyCode Code of the key.
+ * @param {number} optModifiers Optional list of modifiers passed as additional paramerters.
+ */
+WebInspector.KeyboardShortcut.makeKey = function(keyCode, optModifiers)
+{
+ var modifiers = WebInspector.KeyboardShortcut.Modifiers.None;
+ for (var i = 1; i < arguments.length; i++)
+ modifiers |= arguments[i];
+ return WebInspector.KeyboardShortcut._makeKeyFromCodeAndModifiers(keyCode, modifiers);
+};
+
+WebInspector.KeyboardShortcut.makeKeyFromEvent = function(keyboardEvent)
+{
+ var modifiers = WebInspector.KeyboardShortcut.Modifiers.None;
+ if (keyboardEvent.shiftKey)
+ modifiers |= WebInspector.KeyboardShortcut.Modifiers.Shift;
+ if (keyboardEvent.ctrlKey)
+ modifiers |= WebInspector.KeyboardShortcut.Modifiers.Ctrl;
+ if (keyboardEvent.altKey)
+ modifiers |= WebInspector.KeyboardShortcut.Modifiers.Alt;
+ if (keyboardEvent.metaKey)
+ modifiers |= WebInspector.KeyboardShortcut.Modifiers.Meta;
+ return WebInspector.KeyboardShortcut._makeKeyFromCodeAndModifiers(keyboardEvent.keyCode, modifiers);
+};
+
+WebInspector.KeyboardShortcut._makeKeyFromCodeAndModifiers = function(keyCode, modifiers)
+{
+ return (keyCode & 255) | (modifiers << 8);
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/MetricsSidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/MetricsSidebarPane.js
new file mode 100644
index 0000000..a33653b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/MetricsSidebarPane.js
@@ -0,0 +1,215 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.MetricsSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Metrics"));
+ this._inlineStyleId = null;
+}
+
+WebInspector.MetricsSidebarPane.prototype = {
+ update: function(node)
+ {
+ var body = this.bodyElement;
+
+ body.removeChildren();
+
+ if (node)
+ this.node = node;
+ else
+ node = this.node;
+
+ if (!node || !node.ownerDocument.defaultView)
+ return;
+
+ if (node.nodeType !== Node.ELEMENT_NODE)
+ return;
+
+ var self = this;
+ var callback = function(stylePayload) {
+ if (!stylePayload)
+ return;
+ var style = WebInspector.CSSStyleDeclaration.parseStyle(stylePayload);
+ self._update(node, body, style);
+ };
+ InjectedScriptAccess.getComputedStyle(node.id, callback);
+
+ var inlineStyleCallback = function(stylePayload) {
+ if (!stylePayload)
+ return;
+ self._inlineStyleId = stylePayload.id;
+ };
+ InjectedScriptAccess.getInlineStyle(node.id, inlineStyleCallback);
+ },
+
+ _update: function(node, body, style)
+ {
+ var metricsElement = document.createElement("div");
+ metricsElement.className = "metrics";
+
+ function createBoxPartElement(style, name, side, suffix)
+ {
+ var propertyName = (name !== "position" ? name + "-" : "") + side + suffix;
+ var value = style.getPropertyValue(propertyName);
+ if (value === "" || (name !== "position" && value === "0px"))
+ value = "\u2012";
+ else if (name === "position" && value === "auto")
+ value = "\u2012";
+ value = value.replace(/px$/, "");
+
+ var element = document.createElement("div");
+ element.className = side;
+ element.textContent = value;
+ element.addEventListener("dblclick", this.startEditing.bind(this, element, name, propertyName), false);
+ return element;
+ }
+
+ // Display types for which margin is ignored.
+ var noMarginDisplayType = {
+ "table-cell": true,
+ "table-column": true,
+ "table-column-group": true,
+ "table-footer-group": true,
+ "table-header-group": true,
+ "table-row": true,
+ "table-row-group": true
+ };
+
+ // Display types for which padding is ignored.
+ var noPaddingDisplayType = {
+ "table-column": true,
+ "table-column-group": true,
+ "table-footer-group": true,
+ "table-header-group": true,
+ "table-row": true,
+ "table-row-group": true
+ };
+
+ // Position types for which top, left, bottom and right are ignored.
+ var noPositionType = {
+ "static": true
+ };
+
+ var boxes = ["content", "padding", "border", "margin", "position"];
+ var boxLabels = [WebInspector.UIString("content"), WebInspector.UIString("padding"), WebInspector.UIString("border"), WebInspector.UIString("margin"), WebInspector.UIString("position")];
+ var previousBox;
+ for (var i = 0; i < boxes.length; ++i) {
+ var name = boxes[i];
+
+ if (name === "margin" && noMarginDisplayType[style.display])
+ continue;
+ if (name === "padding" && noPaddingDisplayType[style.display])
+ continue;
+ if (name === "position" && noPositionType[style.position])
+ continue;
+
+ var boxElement = document.createElement("div");
+ boxElement.className = name;
+
+ if (name === "content") {
+ var width = style.width.replace(/px$/, "");
+ var widthElement = document.createElement("span");
+ widthElement.textContent = width;
+ widthElement.addEventListener("dblclick", this.startEditing.bind(this, widthElement, "width", "width"), false);
+
+ var height = style.height.replace(/px$/, "");
+ var heightElement = document.createElement("span");
+ heightElement.textContent = height;
+ heightElement.addEventListener("dblclick", this.startEditing.bind(this, heightElement, "height", "height"), false);
+
+ boxElement.appendChild(widthElement);
+ boxElement.appendChild(document.createTextNode(" \u00D7 "));
+ boxElement.appendChild(heightElement);
+ } else {
+ var suffix = (name === "border" ? "-width" : "");
+
+ var labelElement = document.createElement("div");
+ labelElement.className = "label";
+ labelElement.textContent = boxLabels[i];
+ boxElement.appendChild(labelElement);
+
+ boxElement.appendChild(createBoxPartElement.call(this, style, name, "top", suffix));
+ boxElement.appendChild(document.createElement("br"));
+ boxElement.appendChild(createBoxPartElement.call(this, style, name, "left", suffix));
+
+ if (previousBox)
+ boxElement.appendChild(previousBox);
+
+ boxElement.appendChild(createBoxPartElement.call(this, style, name, "right", suffix));
+ boxElement.appendChild(document.createElement("br"));
+ boxElement.appendChild(createBoxPartElement.call(this, style, name, "bottom", suffix));
+ }
+
+ previousBox = boxElement;
+ }
+
+ metricsElement.appendChild(previousBox);
+ body.appendChild(metricsElement);
+ },
+
+ startEditing: function(targetElement, box, styleProperty)
+ {
+ if (WebInspector.isBeingEdited(targetElement))
+ return;
+
+ var context = { box: box, styleProperty: styleProperty };
+
+ WebInspector.startEditing(targetElement, this.editingCommitted.bind(this), this.editingCancelled.bind(this), context);
+ },
+
+ editingCancelled: function(element, context)
+ {
+ this.update();
+ },
+
+ editingCommitted: function(element, userInput, previousContent, context)
+ {
+ if (userInput === previousContent)
+ return this.editingCancelled(element, context); // nothing changed, so cancel
+
+ if (context.box !== "position" && (!userInput || userInput === "\u2012"))
+ userInput = "0px";
+ else if (context.box === "position" && (!userInput || userInput === "\u2012"))
+ userInput = "auto";
+
+ // Append a "px" unit if the user input was just a number.
+ if (/^\d+$/.test(userInput))
+ userInput += "px";
+
+ var self = this;
+ var callback = function(success) {
+ if (!success)
+ return;
+ self.dispatchEventToListeners("metrics edited");
+ self.update();
+ };
+ InjectedScriptAccess.setStyleProperty(this._inlineStyleId, context.styleProperty, userInput, callback);
+ }
+}
+
+WebInspector.MetricsSidebarPane.prototype.__proto__ = WebInspector.SidebarPane.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Object.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Object.js
new file mode 100644
index 0000000..80202b0
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Object.js
@@ -0,0 +1,82 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Object = function() {
+}
+
+WebInspector.Object.prototype = {
+ addEventListener: function(eventType, listener, thisObject) {
+ if (!("_listeners" in this))
+ this._listeners = {};
+ if (!(eventType in this._listeners))
+ this._listeners[eventType] = [];
+ this._listeners[eventType].push({ thisObject: thisObject, listener: listener });
+ },
+
+ removeEventListener: function(eventType, listener, thisObject) {
+ if (!("_listeners" in this) || !(eventType in this._listeners))
+ return;
+ var listeners = this._listeners[eventType];
+ for (var i = 0; i < listeners.length; ++i) {
+ if (listener && listeners[i].listener === listener && listeners[i].thisObject === thisObject)
+ listeners.splice(i, 1);
+ else if (!listener && thisObject && listeners[i].thisObject === thisObject)
+ listeners.splice(i, 1);
+ }
+
+ if (!listeners.length)
+ delete this._listeners[eventType];
+ },
+
+ dispatchEventToListeners: function(eventType) {
+ if (!("_listeners" in this) || !(eventType in this._listeners))
+ return;
+
+ var stoppedPropagation = false;
+
+ function stopPropagation()
+ {
+ stoppedPropagation = true;
+ }
+
+ function preventDefault()
+ {
+ this.defaultPrevented = true;
+ }
+
+ var event = {target: this, type: eventType, defaultPrevented: false};
+ event.stopPropagation = stopPropagation.bind(event);
+ event.preventDefault = preventDefault.bind(event);
+
+ var listeners = this._listeners[eventType];
+ for (var i = 0; i < listeners.length; ++i) {
+ listeners[i].listener.call(listeners[i].thisObject, event);
+ if (stoppedPropagation)
+ break;
+ }
+
+ return event.defaultPrevented;
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectPropertiesSection.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectPropertiesSection.js
new file mode 100644
index 0000000..8bb4e35
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectPropertiesSection.js
@@ -0,0 +1,258 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ObjectPropertiesSection = function(object, title, subtitle, emptyPlaceholder, ignoreHasOwnProperty, extraProperties, treeElementConstructor)
+{
+ this.emptyPlaceholder = (emptyPlaceholder || WebInspector.UIString("No Properties"));
+ this.object = object;
+ this.ignoreHasOwnProperty = ignoreHasOwnProperty;
+ this.extraProperties = extraProperties;
+ this.treeElementConstructor = treeElementConstructor || WebInspector.ObjectPropertyTreeElement;
+ this.editable = true;
+
+ WebInspector.PropertiesSection.call(this, title, subtitle);
+}
+
+WebInspector.ObjectPropertiesSection.prototype = {
+ onpopulate: function()
+ {
+ this.update();
+ },
+
+ update: function()
+ {
+ var self = this;
+ var callback = function(properties) {
+ if (!properties)
+ return;
+ self.updateProperties(properties);
+ };
+ InjectedScriptAccess.getProperties(this.object, this.ignoreHasOwnProperty, callback);
+ },
+
+ updateProperties: function(properties, rootTreeElementConstructor, rootPropertyComparer)
+ {
+ if (!rootTreeElementConstructor)
+ rootTreeElementConstructor = this.treeElementConstructor;
+
+ if (!rootPropertyComparer)
+ rootPropertyComparer = WebInspector.ObjectPropertiesSection.CompareProperties;
+
+ if (this.extraProperties)
+ for (var i = 0; i < this.extraProperties.length; ++i)
+ properties.push(this.extraProperties[i]);
+
+ properties.sort(rootPropertyComparer);
+
+ this.propertiesTreeOutline.removeChildren();
+
+ for (var i = 0; i < properties.length; ++i)
+ this.propertiesTreeOutline.appendChild(new rootTreeElementConstructor(properties[i]));
+
+ if (!this.propertiesTreeOutline.children.length) {
+ var title = "<div class=\"info\">" + this.emptyPlaceholder + "</div>";
+ var infoElement = new TreeElement(title, null, false);
+ this.propertiesTreeOutline.appendChild(infoElement);
+ }
+ }
+}
+
+WebInspector.ObjectPropertiesSection.prototype.__proto__ = WebInspector.PropertiesSection.prototype;
+
+WebInspector.ObjectPropertiesSection.CompareProperties = function(propertyA, propertyB)
+{
+ var a = propertyA.name;
+ var b = propertyB.name;
+
+ // if used elsewhere make sure to
+ // - convert a and b to strings (not needed here, properties are all strings)
+ // - check if a == b (not needed here, no two properties can be the same)
+
+ var diff = 0;
+ var chunk = /^\d+|^\D+/;
+ var chunka, chunkb, anum, bnum;
+ while (diff === 0) {
+ if (!a && b)
+ return -1;
+ if (!b && a)
+ return 1;
+ chunka = a.match(chunk)[0];
+ chunkb = b.match(chunk)[0];
+ anum = !isNaN(chunka);
+ bnum = !isNaN(chunkb);
+ if (anum && !bnum)
+ return -1;
+ if (bnum && !anum)
+ return 1;
+ if (anum && bnum) {
+ diff = chunka - chunkb;
+ if (diff === 0 && chunka.length !== chunkb.length) {
+ if (!+chunka && !+chunkb) // chunks are strings of all 0s (special case)
+ return chunka.length - chunkb.length;
+ else
+ return chunkb.length - chunka.length;
+ }
+ } else if (chunka !== chunkb)
+ return (chunka < chunkb) ? -1 : 1;
+ a = a.substring(chunka.length);
+ b = b.substring(chunkb.length);
+ }
+ return diff;
+}
+
+WebInspector.ObjectPropertyTreeElement = function(property)
+{
+ this.property = property;
+
+ // Pass an empty title, the title gets made later in onattach.
+ TreeElement.call(this, "", null, false);
+}
+
+WebInspector.ObjectPropertyTreeElement.prototype = {
+ onpopulate: function()
+ {
+ if (this.children.length && !this.shouldRefreshChildren)
+ return;
+
+ var callback = function(properties) {
+ this.removeChildren();
+ if (!properties)
+ return;
+
+ properties.sort(WebInspector.ObjectPropertiesSection.CompareProperties);
+ for (var i = 0; i < properties.length; ++i) {
+ this.appendChild(new this.treeOutline.section.treeElementConstructor(properties[i]));
+ }
+ };
+ InjectedScriptAccess.getProperties(this.property.value, false, callback.bind(this));
+ },
+
+ ondblclick: function(element, event)
+ {
+ this.startEditing();
+ },
+
+ onattach: function()
+ {
+ this.update();
+ },
+
+ update: function()
+ {
+ this.nameElement = document.createElement("span");
+ this.nameElement.className = "name";
+ this.nameElement.textContent = this.property.name;
+
+ var separatorElement = document.createElement("span");
+ separatorElement.className = "separator";
+ separatorElement.textContent = ": ";
+
+ this.valueElement = document.createElement("span");
+ this.valueElement.className = "value";
+ this.valueElement.textContent = this.property.value.description;
+ if (this.property.isGetter)
+ this.valueElement.addStyleClass("dimmed");
+
+ this.listItemElement.removeChildren();
+
+ this.listItemElement.appendChild(this.nameElement);
+ this.listItemElement.appendChild(separatorElement);
+ this.listItemElement.appendChild(this.valueElement);
+ this.hasChildren = this.property.value.hasChildren;
+ },
+
+ updateSiblings: function()
+ {
+ if (this.parent.root)
+ this.treeOutline.section.update();
+ else
+ this.parent.shouldRefreshChildren = true;
+ },
+
+ startEditing: function()
+ {
+ if (WebInspector.isBeingEdited(this.valueElement) || !this.treeOutline.section.editable)
+ return;
+
+ var context = { expanded: this.expanded };
+
+ // Lie about our children to prevent expanding on double click and to collapse subproperties.
+ this.hasChildren = false;
+
+ this.listItemElement.addStyleClass("editing-sub-part");
+
+ WebInspector.startEditing(this.valueElement, this.editingCommitted.bind(this), this.editingCancelled.bind(this), context);
+ },
+
+ editingEnded: function(context)
+ {
+ this.listItemElement.scrollLeft = 0;
+ this.listItemElement.removeStyleClass("editing-sub-part");
+ if (context.expanded)
+ this.expand();
+ },
+
+ editingCancelled: function(element, context)
+ {
+ this.update();
+ this.editingEnded(context);
+ },
+
+ editingCommitted: function(element, userInput, previousContent, context)
+ {
+ if (userInput === previousContent)
+ return this.editingCancelled(element, context); // nothing changed, so cancel
+
+ this.applyExpression(userInput, true);
+
+ this.editingEnded(context);
+ },
+
+ applyExpression: function(expression, updateInterface)
+ {
+ expression = expression.trimWhitespace();
+ var expressionLength = expression.length;
+ var self = this;
+ var callback = function(success) {
+ if (!updateInterface)
+ return;
+
+ if (!success)
+ self.update();
+
+ if (!expressionLength) {
+ // The property was deleted, so remove this tree element.
+ self.parent.removeChild(this);
+ } else {
+ // Call updateSiblings since their value might be based on the value that just changed.
+ self.updateSiblings();
+ }
+ };
+ InjectedScriptAccess.setPropertyValue(this.property.parentObjectProxy, this.property.name, expression.trimWhitespace(), callback);
+ }
+}
+
+WebInspector.ObjectPropertyTreeElement.prototype.__proto__ = TreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectProxy.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectProxy.js
new file mode 100644
index 0000000..03d16ab
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ObjectProxy.js
@@ -0,0 +1,44 @@
+/*
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ObjectProxy = function(objectId, path, protoDepth, description, hasChildren)
+{
+ this.objectId = objectId;
+ this.path = path || [];
+ this.protoDepth = protoDepth || 0;
+ this.description = description;
+ this.hasChildren = hasChildren;
+}
+
+WebInspector.ObjectPropertyProxy = function(name, value)
+{
+ this.name = name;
+ this.value = value;
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Panel.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Panel.js
new file mode 100644
index 0000000..5046f6b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Panel.js
@@ -0,0 +1,273 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Panel = function()
+{
+ WebInspector.View.call(this);
+
+ this.element.addStyleClass("panel");
+}
+
+WebInspector.Panel.prototype = {
+ get toolbarItem()
+ {
+ if (this._toolbarItem)
+ return this._toolbarItem;
+
+ // Sample toolbar item as markup:
+ // <button class="toolbar-item resources toggleable">
+ // <div class="toolbar-icon"></div>
+ // <div class="toolbar-label">Resources</div>
+ // </button>
+
+ this._toolbarItem = document.createElement("button");
+ this._toolbarItem.className = "toolbar-item toggleable";
+ this._toolbarItem.panel = this;
+
+ if ("toolbarItemClass" in this)
+ this._toolbarItem.addStyleClass(this.toolbarItemClass);
+
+ var iconElement = document.createElement("div");
+ iconElement.className = "toolbar-icon";
+ this._toolbarItem.appendChild(iconElement);
+
+ if ("toolbarItemLabel" in this) {
+ var labelElement = document.createElement("div");
+ labelElement.className = "toolbar-label";
+ labelElement.textContent = this.toolbarItemLabel;
+ this._toolbarItem.appendChild(labelElement);
+ }
+
+ return this._toolbarItem;
+ },
+
+ show: function()
+ {
+ WebInspector.View.prototype.show.call(this);
+
+ var statusBarItems = this.statusBarItems;
+ if (statusBarItems) {
+ this._statusBarItemContainer = document.createElement("div");
+ for (var i = 0; i < statusBarItems.length; ++i)
+ this._statusBarItemContainer.appendChild(statusBarItems[i]);
+ document.getElementById("main-status-bar").appendChild(this._statusBarItemContainer);
+ }
+
+ if ("_toolbarItem" in this)
+ this._toolbarItem.addStyleClass("toggled-on");
+
+ WebInspector.currentFocusElement = document.getElementById("main-panels");
+ },
+
+ hide: function()
+ {
+ WebInspector.View.prototype.hide.call(this);
+
+ if (this._statusBarItemContainer && this._statusBarItemContainer.parentNode)
+ this._statusBarItemContainer.parentNode.removeChild(this._statusBarItemContainer);
+ delete this._statusBarItemContainer;
+ if ("_toolbarItem" in this)
+ this._toolbarItem.removeStyleClass("toggled-on");
+ },
+
+ attach: function()
+ {
+ if (!this.element.parentNode)
+ document.getElementById("main-panels").appendChild(this.element);
+ },
+
+ searchCanceled: function(startingNewSearch)
+ {
+ if (this._searchResults) {
+ for (var i = 0; i < this._searchResults.length; ++i) {
+ var view = this._searchResults[i];
+ if (view.searchCanceled)
+ view.searchCanceled();
+ delete view.currentQuery;
+ }
+ }
+
+ WebInspector.updateSearchMatchesCount(0, this);
+
+ if (this._currentSearchChunkIntervalIdentifier) {
+ clearInterval(this._currentSearchChunkIntervalIdentifier);
+ delete this._currentSearchChunkIntervalIdentifier;
+ }
+
+ this._totalSearchMatches = 0;
+ this._currentSearchResultIndex = 0;
+ this._searchResults = [];
+ },
+
+ performSearch: function(query)
+ {
+ // Call searchCanceled since it will reset everything we need before doing a new search.
+ this.searchCanceled(true);
+
+ var searchableViews = this.searchableViews;
+ if (!searchableViews || !searchableViews.length)
+ return;
+
+ var parentElement = this.viewsContainerElement;
+ var visibleView = this.visibleView;
+ var sortFuction = this.searchResultsSortFunction;
+
+ var matchesCountUpdateTimeout = null;
+
+ function updateMatchesCount()
+ {
+ WebInspector.updateSearchMatchesCount(this._totalSearchMatches, this);
+ matchesCountUpdateTimeout = null;
+ }
+
+ function updateMatchesCountSoon()
+ {
+ if (matchesCountUpdateTimeout)
+ return;
+ // Update the matches count every half-second so it doesn't feel twitchy.
+ matchesCountUpdateTimeout = setTimeout(updateMatchesCount.bind(this), 500);
+ }
+
+ function finishedCallback(view, searchMatches)
+ {
+ if (!searchMatches)
+ return;
+
+ this._totalSearchMatches += searchMatches;
+ this._searchResults.push(view);
+
+ if (sortFuction)
+ this._searchResults.sort(sortFuction);
+
+ if (this.searchMatchFound)
+ this.searchMatchFound(view, searchMatches);
+
+ updateMatchesCountSoon.call(this);
+
+ if (view === visibleView)
+ view.jumpToFirstSearchResult();
+ }
+
+ var i = 0;
+ var panel = this;
+ var boundFinishedCallback = finishedCallback.bind(this);
+ var chunkIntervalIdentifier = null;
+
+ // Split up the work into chunks so we don't block the
+ // UI thread while processing.
+
+ function processChunk()
+ {
+ var view = searchableViews[i];
+
+ if (++i >= searchableViews.length) {
+ if (panel._currentSearchChunkIntervalIdentifier === chunkIntervalIdentifier)
+ delete panel._currentSearchChunkIntervalIdentifier;
+ clearInterval(chunkIntervalIdentifier);
+ }
+
+ if (!view)
+ return;
+
+ if (view.element.parentNode !== parentElement && view.element.parentNode && parentElement)
+ view.detach();
+
+ view.currentQuery = query;
+ view.performSearch(query, boundFinishedCallback);
+ }
+
+ processChunk();
+
+ chunkIntervalIdentifier = setInterval(processChunk, 25);
+ this._currentSearchChunkIntervalIdentifier = chunkIntervalIdentifier;
+ },
+
+ jumpToNextSearchResult: function()
+ {
+ if (!this.showView || !this._searchResults || !this._searchResults.length)
+ return;
+
+ var showFirstResult = false;
+
+ this._currentSearchResultIndex = this._searchResults.indexOf(this.visibleView);
+ if (this._currentSearchResultIndex === -1) {
+ this._currentSearchResultIndex = 0;
+ showFirstResult = true;
+ }
+
+ var currentView = this._searchResults[this._currentSearchResultIndex];
+
+ if (currentView.showingLastSearchResult()) {
+ if (++this._currentSearchResultIndex >= this._searchResults.length)
+ this._currentSearchResultIndex = 0;
+ currentView = this._searchResults[this._currentSearchResultIndex];
+ showFirstResult = true;
+ }
+
+ if (currentView !== this.visibleView)
+ this.showView(currentView);
+
+ if (showFirstResult)
+ currentView.jumpToFirstSearchResult();
+ else
+ currentView.jumpToNextSearchResult();
+ },
+
+ jumpToPreviousSearchResult: function()
+ {
+ if (!this.showView || !this._searchResults || !this._searchResults.length)
+ return;
+
+ var showLastResult = false;
+
+ this._currentSearchResultIndex = this._searchResults.indexOf(this.visibleView);
+ if (this._currentSearchResultIndex === -1) {
+ this._currentSearchResultIndex = 0;
+ showLastResult = true;
+ }
+
+ var currentView = this._searchResults[this._currentSearchResultIndex];
+
+ if (currentView.showingFirstSearchResult()) {
+ if (--this._currentSearchResultIndex < 0)
+ this._currentSearchResultIndex = (this._searchResults.length - 1);
+ currentView = this._searchResults[this._currentSearchResultIndex];
+ showLastResult = true;
+ }
+
+ if (currentView !== this.visibleView)
+ this.showView(currentView);
+
+ if (showLastResult)
+ currentView.jumpToLastSearchResult();
+ else
+ currentView.jumpToPreviousSearchResult();
+ }
+}
+
+WebInspector.Panel.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PanelEnablerView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PanelEnablerView.js
new file mode 100644
index 0000000..fab6d76
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PanelEnablerView.js
@@ -0,0 +1,97 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.PanelEnablerView = function(identifier, headingText, disclaimerText, buttonTitle)
+{
+ WebInspector.View.call(this);
+
+ this.element.addStyleClass("panel-enabler-view");
+ this.element.addStyleClass(identifier);
+
+ this.contentElement = document.createElement("div");
+ this.contentElement.className = "panel-enabler-view-content";
+ this.element.appendChild(this.contentElement);
+
+ this.imageElement = document.createElement("img");
+ this.contentElement.appendChild(this.imageElement);
+
+ this.choicesForm = document.createElement("form");
+ this.contentElement.appendChild(this.choicesForm);
+
+ this.headerElement = document.createElement("h1");
+ this.headerElement.textContent = headingText;
+ this.choicesForm.appendChild(this.headerElement);
+
+ var self = this;
+ function enableOption(text, checked) {
+ var label = document.createElement("label");
+ var option = document.createElement("input");
+ option.type = "radio";
+ option.name = "enable-option";
+ if (checked)
+ option.checked = true;
+ label.appendChild(option);
+ label.appendChild(document.createTextNode(text));
+ self.choicesForm.appendChild(label);
+ return option;
+ };
+
+ this.enabledForSession = enableOption(WebInspector.UIString("Only enable for this session"), true);
+ this.enabledAlways = enableOption(WebInspector.UIString("Always enable"));
+
+ this.disclaimerElement = document.createElement("div");
+ this.disclaimerElement.className = "panel-enabler-disclaimer";
+ this.disclaimerElement.textContent = disclaimerText;
+ this.choicesForm.appendChild(this.disclaimerElement);
+
+ this.enableButton = document.createElement("button");
+ this.enableButton.setAttribute("type", "button");
+ this.enableButton.textContent = buttonTitle;
+ this.enableButton.addEventListener("click", this._enableButtonCicked.bind(this), false);
+ this.choicesForm.appendChild(this.enableButton);
+
+ window.addEventListener("resize", this._windowResized.bind(this), true);
+}
+
+WebInspector.PanelEnablerView.prototype = {
+ _enableButtonCicked: function()
+ {
+ this.dispatchEventToListeners("enable clicked");
+ },
+
+ _windowResized: function()
+ {
+ this.imageElement.removeStyleClass("hidden");
+
+ if (this.element.offsetWidth < (this.choicesForm.offsetWidth + this.imageElement.offsetWidth))
+ this.imageElement.addStyleClass("hidden");
+ },
+
+ get alwaysEnabled() {
+ return this.enabledAlways.checked;
+ }
+}
+
+WebInspector.PanelEnablerView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Placard.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Placard.js
new file mode 100644
index 0000000..69a168e
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Placard.js
@@ -0,0 +1,106 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Placard = function(title, subtitle)
+{
+ this.element = document.createElement("div");
+ this.element.className = "placard";
+ this.element.placard = this;
+
+ this.titleElement = document.createElement("div");
+ this.titleElement.className = "title";
+
+ this.subtitleElement = document.createElement("div");
+ this.subtitleElement.className = "subtitle";
+
+ this.element.appendChild(this.subtitleElement);
+ this.element.appendChild(this.titleElement);
+
+ this.title = title;
+ this.subtitle = subtitle;
+ this.selected = false;
+}
+
+WebInspector.Placard.prototype = {
+ get title()
+ {
+ return this._title;
+ },
+
+ set title(x)
+ {
+ if (this._title === x)
+ return;
+ this._title = x;
+ this.titleElement.textContent = x;
+ },
+
+ get subtitle()
+ {
+ return this._subtitle;
+ },
+
+ set subtitle(x)
+ {
+ if (this._subtitle === x)
+ return;
+ this._subtitle = x;
+ this.subtitleElement.innerHTML = x;
+ },
+
+ get selected()
+ {
+ return this._selected;
+ },
+
+ set selected(x)
+ {
+ if (x)
+ this.select();
+ else
+ this.deselect();
+ },
+
+ select: function()
+ {
+ if (this._selected)
+ return;
+ this._selected = true;
+ this.element.addStyleClass("selected");
+ },
+
+ deselect: function()
+ {
+ if (!this._selected)
+ return;
+ this._selected = false;
+ this.element.removeStyleClass("selected");
+ },
+
+ toggleSelected: function()
+ {
+ this.selected = !this.selected;
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Popup.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Popup.js
new file mode 100644
index 0000000..9c8ef24
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Popup.js
@@ -0,0 +1,168 @@
+/*
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+/**
+ * This class provides a popup that can be shown relative to an anchor element
+ * or at an arbitrary absolute position.
+ * Points are Objects: {x: xValue, y: yValue}.
+ * Rectangles are Objects: {x: xValue, y: yValue, width: widthValue, height: heightValue}.
+ *
+ * element is an optional unparented visible element (style.display != "none" AND style.visibility != "hidden").
+ * If the element is absent/undefined, it must have been set with the element(x) setter before the show() method invocation.
+ */
+WebInspector.Popup = function(element)
+{
+ if (element)
+ this.element = element;
+ this._keyHandler = this._keyEventHandler.bind(this);
+ this._mouseDownHandler = this._mouseDownEventHandler.bind(this);
+ this._visible = false;
+ this._autoHide = true;
+}
+
+WebInspector.Popup.prototype = {
+ show: function()
+ {
+ if (this.visible)
+ return;
+ var ownerDocument = this._contentElement.ownerDocument;
+ if (!ownerDocument)
+ return;
+
+ this._glasspaneElement = ownerDocument.createElement("div");
+ this._glasspaneElement.className = "popup-glasspane";
+ ownerDocument.body.appendChild(this._glasspaneElement);
+
+ this._contentElement.positionAt(0, 0);
+ this._contentElement.removeStyleClass("hidden");
+ ownerDocument.body.appendChild(this._contentElement);
+
+ this.positionElement();
+ this._visible = true;
+ ownerDocument.addEventListener("keydown", this._keyHandler, false);
+ ownerDocument.addEventListener("mousedown", this._mouseDownHandler, false);
+ },
+
+ hide: function()
+ {
+ if (this.visible) {
+ this._visible = false;
+ this._contentElement.ownerDocument.removeEventListener("keydown", this._keyHandler, false);
+ this._contentElement.ownerDocument.removeEventListener("mousedown", this._mouseDownHandler, false);
+ this._glasspaneElement.parentElement.removeChild(this._glasspaneElement);
+ this._contentElement.parentElement.removeChild(this._contentElement);
+ }
+ },
+
+ get visible()
+ {
+ return this._visible;
+ },
+
+ set element(x)
+ {
+ this._checkNotVisible();
+ this._contentElement = x;
+ this._contentElement.addStyleClass("hidden");
+ },
+
+ get element()
+ {
+ return this._contentElement;
+ },
+
+ positionElement: function()
+ {
+ var element = this._contentElement;
+ var anchorElement = this._anchorElement;
+
+ var targetDocument = element.ownerDocument;
+ var targetDocumentBody = targetDocument.body;
+ var targetDocumentElement = targetDocument.documentElement;
+ var clippingBox = {x: 0, y: 0, width: targetDocumentElement.clientWidth, height: targetDocumentElement.clientHeight};
+ var parentElement = element.offsetParent || element.parentElement;
+
+ var anchorPosition = {x: anchorElement.totalOffsetLeft, y: anchorElement.totalOffsetTop};
+
+ // FIXME(apavlov@chromium.org): Translate anchorPosition to the element.ownerDocument frame when https://bugs.webkit.org/show_bug.cgi?id=28913 is fixed.
+ var anchorBox = {x: anchorPosition.x, y: anchorPosition.y, width: anchorElement.offsetWidth, height: anchorElement.offsetHeight};
+ var elementBox = {x: element.totalOffsetLeft, y: element.totalOffsetTop, width: element.offsetWidth, height: element.offsetHeight};
+ var newElementPosition = {x: 0, y: 0};
+
+ if (anchorBox.y - elementBox.height >= clippingBox.y)
+ newElementPosition.y = anchorBox.y - elementBox.height;
+ else
+ newElementPosition.y = Math.min(anchorBox.y + anchorBox.height, Math.max(clippingBox.y, clippingBox.y + clippingBox.height - elementBox.height));
+
+ if (anchorBox.x + elementBox.height <= clippingBox.x + clippingBox.height)
+ newElementPosition.x = anchorBox.x;
+ else
+ newElementPosition.x = Math.max(clippingBox.x, clippingBox.x + clippingBox.height - elementBox.height);
+ element.positionAt(newElementPosition.x, newElementPosition.y);
+ },
+
+ set anchor(x)
+ {
+ this._checkNotVisible();
+ this._anchorElement = x;
+ },
+
+ get anchor()
+ {
+ return this._anchorElement;
+ },
+
+ set autoHide(x)
+ {
+ this._autoHide = x;
+ },
+
+ _checkNotVisible: function()
+ {
+ if (this.visible)
+ throw new Error("The popup must not be visible.");
+ },
+
+ _keyEventHandler: function(event)
+ {
+ // Escape hides the popup.
+ if (event.keyIdentifier == "U+001B") {
+ this.hide();
+ event.preventDefault();
+ event.handled = true;
+ }
+ },
+
+ _mouseDownEventHandler: function(event)
+ {
+ if (this._autoHide && event.originalTarget === this._glasspaneElement)
+ this.hide();
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileDataGridTree.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileDataGridTree.js
new file mode 100644
index 0000000..356f57d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileDataGridTree.js
@@ -0,0 +1,398 @@
+/*
+ * Copyright (C) 2009 280 North Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ProfileDataGridNode = function(profileView, profileNode, owningTree, hasChildren)
+{
+ this.profileView = profileView;
+ this.profileNode = profileNode;
+
+ WebInspector.DataGridNode.call(this, null, hasChildren);
+
+ this.addEventListener("populate", this._populate, this);
+
+ this.tree = owningTree;
+
+ this.childrenByCallUID = {};
+ this.lastComparator = null;
+
+ this.callUID = profileNode.callUID;
+ this.selfTime = profileNode.selfTime;
+ this.totalTime = profileNode.totalTime;
+ this.functionName = profileNode.functionName;
+ this.numberOfCalls = profileNode.numberOfCalls;
+ this.url = profileNode.url;
+}
+
+WebInspector.ProfileDataGridNode.prototype = {
+ get data()
+ {
+ function formatMilliseconds(time)
+ {
+ return Number.secondsToString(time / 1000, WebInspector.UIString.bind(WebInspector), !Preferences.samplingCPUProfiler);
+ }
+
+ var data = {};
+
+ data["function"] = this.functionName;
+ data["calls"] = this.numberOfCalls;
+
+ if (this.profileView.showSelfTimeAsPercent)
+ data["self"] = WebInspector.UIString("%.2f%%", this.selfPercent);
+ else
+ data["self"] = formatMilliseconds(this.selfTime);
+
+ if (this.profileView.showTotalTimeAsPercent)
+ data["total"] = WebInspector.UIString("%.2f%%", this.totalPercent);
+ else
+ data["total"] = formatMilliseconds(this.totalTime);
+
+ if (this.profileView.showAverageTimeAsPercent)
+ data["average"] = WebInspector.UIString("%.2f%%", this.averagePercent);
+ else
+ data["average"] = formatMilliseconds(this.averageTime);
+
+ return data;
+ },
+
+ createCell: function(columnIdentifier)
+ {
+ var cell = WebInspector.DataGridNode.prototype.createCell.call(this, columnIdentifier);
+
+ if (columnIdentifier === "self" && this._searchMatchedSelfColumn)
+ cell.addStyleClass("highlight");
+ else if (columnIdentifier === "total" && this._searchMatchedTotalColumn)
+ cell.addStyleClass("highlight");
+ else if (columnIdentifier === "average" && this._searchMatchedAverageColumn)
+ cell.addStyleClass("highlight");
+ else if (columnIdentifier === "calls" && this._searchMatchedCallsColumn)
+ cell.addStyleClass("highlight");
+
+ if (columnIdentifier !== "function")
+ return cell;
+
+ if (this.profileNode._searchMatchedFunctionColumn)
+ cell.addStyleClass("highlight");
+
+ if (this.profileNode.url) {
+ var fileName = WebInspector.displayNameForURL(this.profileNode.url);
+
+ var urlElement = document.createElement("a");
+ urlElement.className = "profile-node-file webkit-html-resource-link";
+ urlElement.href = this.profileNode.url;
+ urlElement.lineNumber = this.profileNode.lineNumber;
+
+ if (this.profileNode.lineNumber > 0)
+ urlElement.textContent = fileName + ":" + this.profileNode.lineNumber;
+ else
+ urlElement.textContent = fileName;
+
+ cell.insertBefore(urlElement, cell.firstChild);
+ }
+
+ return cell;
+ },
+
+ select: function(supressSelectedEvent)
+ {
+ WebInspector.DataGridNode.prototype.select.call(this, supressSelectedEvent);
+ this.profileView._dataGridNodeSelected(this);
+ },
+
+ deselect: function(supressDeselectedEvent)
+ {
+ WebInspector.DataGridNode.prototype.deselect.call(this, supressDeselectedEvent);
+ this.profileView._dataGridNodeDeselected(this);
+ },
+
+ expand: function()
+ {
+ if (!this.parent) {
+ var currentComparator = this.parent.lastComparator;
+
+ if (!currentComparator || (currentComparator === this.lastComparator))
+ return;
+
+ this.sort(currentComparator);
+ }
+
+ WebInspector.DataGridNode.prototype.expand.call(this);
+ },
+
+ sort: function(/*Function*/ comparator, /*Boolean*/ force)
+ {
+ var gridNodeGroups = [[this]];
+
+ for (var gridNodeGroupIndex = 0; gridNodeGroupIndex < gridNodeGroups.length; ++gridNodeGroupIndex) {
+ var gridNodes = gridNodeGroups[gridNodeGroupIndex];
+ var count = gridNodes.length;
+
+ for (var index = 0; index < count; ++index) {
+ var gridNode = gridNodes[index];
+
+ // If the grid node is collapsed, then don't sort children (save operation for later).
+ // If the grid node has the same sorting as previously, then there is no point in sorting it again.
+ if (!force && !gridNode.expanded || gridNode.lastComparator === comparator)
+ continue;
+
+ gridNode.lastComparator = comparator;
+
+ var children = gridNode.children;
+ var childCount = children.length;
+
+ if (childCount) {
+ children.sort(comparator);
+
+ for (var childIndex = 0; childIndex < childCount; ++childIndex)
+ children[childIndex]._recalculateSiblings(childIndex);
+
+ gridNodeGroups.push(children);
+ }
+ }
+ }
+ },
+
+ insertChild: function(/*ProfileDataGridNode*/ profileDataGridNode, index)
+ {
+ WebInspector.DataGridNode.prototype.insertChild.call(this, profileDataGridNode, index);
+
+ this.childrenByCallUID[profileDataGridNode.callUID] = profileDataGridNode;
+ },
+
+ removeChild: function(/*ProfileDataGridNode*/ profileDataGridNode)
+ {
+ WebInspector.DataGridNode.prototype.removeChild.call(this, profileDataGridNode);
+
+ delete this.childrenByCallUID[profileDataGridNode.callUID];
+ },
+
+ removeChildren: function(/*ProfileDataGridNode*/ profileDataGridNode)
+ {
+ WebInspector.DataGridNode.prototype.removeChildren.call(this);
+
+ this.childrenByCallUID = {};
+ },
+
+ findChild: function(/*Node*/ node)
+ {
+ if (!node)
+ return null;
+ return this.childrenByCallUID[node.callUID];
+ },
+
+ get averageTime()
+ {
+ return this.selfTime / Math.max(1, this.numberOfCalls);
+ },
+
+ get averagePercent()
+ {
+ return this.averageTime / this.tree.totalTime * 100.0;
+ },
+
+ get selfPercent()
+ {
+ return this.selfTime / this.tree.totalTime * 100.0;
+ },
+
+ get totalPercent()
+ {
+ return this.totalTime / this.tree.totalTime * 100.0;
+ },
+
+ // When focusing and collapsing we modify lots of nodes in the tree.
+ // This allows us to restore them all to their original state when we revert.
+ _save: function()
+ {
+ if (this._savedChildren)
+ return;
+
+ this._savedSelfTime = this.selfTime;
+ this._savedTotalTime = this.totalTime;
+ this._savedNumberOfCalls = this.numberOfCalls;
+
+ this._savedChildren = this.children.slice();
+ },
+
+ // When focusing and collapsing we modify lots of nodes in the tree.
+ // This allows us to restore them all to their original state when we revert.
+ _restore: function()
+ {
+ if (!this._savedChildren)
+ return;
+
+ this.selfTime = this._savedSelfTime;
+ this.totalTime = this._savedTotalTime;
+ this.numberOfCalls = this._savedNumberOfCalls;
+
+ this.removeChildren();
+
+ var children = this._savedChildren;
+ var count = children.length;
+
+ for (var index = 0; index < count; ++index) {
+ children[index]._restore();
+ this.appendChild(children[index]);
+ }
+ },
+
+ _merge: function(child, shouldAbsorb)
+ {
+ this.selfTime += child.selfTime;
+
+ if (!shouldAbsorb) {
+ this.totalTime += child.totalTime;
+ this.numberOfCalls += child.numberOfCalls;
+ }
+
+ var children = this.children.slice();
+
+ this.removeChildren();
+
+ var count = children.length;
+
+ for (var index = 0; index < count; ++index) {
+ if (!shouldAbsorb || children[index] !== child)
+ this.appendChild(children[index]);
+ }
+
+ children = child.children.slice();
+ count = children.length;
+
+ for (var index = 0; index < count; ++index) {
+ var orphanedChild = children[index],
+ existingChild = this.childrenByCallUID[orphanedChild.callUID];
+
+ if (existingChild)
+ existingChild._merge(orphanedChild, false);
+ else
+ this.appendChild(orphanedChild);
+ }
+ }
+}
+
+WebInspector.ProfileDataGridNode.prototype.__proto__ = WebInspector.DataGridNode.prototype;
+
+WebInspector.ProfileDataGridTree = function(profileView, profileNode)
+{
+ this.tree = this;
+ this.children = [];
+
+ this.profileView = profileView;
+
+ this.totalTime = profileNode.totalTime;
+ this.lastComparator = null;
+
+ this.childrenByCallUID = {};
+}
+
+WebInspector.ProfileDataGridTree.prototype = {
+ get expanded()
+ {
+ return true;
+ },
+
+ appendChild: function(child)
+ {
+ this.insertChild(child, this.children.length);
+ },
+
+ insertChild: function(child, index)
+ {
+ this.children.splice(index, 0, child);
+ this.childrenByCallUID[child.callUID] = child;
+ },
+
+ removeChildren: function()
+ {
+ this.children = [];
+ this.childrenByCallUID = {};
+ },
+
+ findChild: WebInspector.ProfileDataGridNode.prototype.findChild,
+ sort: WebInspector.ProfileDataGridNode.prototype.sort,
+
+ _save: function()
+ {
+ if (this._savedChildren)
+ return;
+
+ this._savedTotalTime = this.totalTime;
+ this._savedChildren = this.children.slice();
+ },
+
+ restore: function()
+ {
+ if (!this._savedChildren)
+ return;
+
+ this.children = this._savedChildren;
+ this.totalTime = this._savedTotalTime;
+
+ var children = this.children;
+ var count = children.length;
+
+ for (var index = 0; index < count; ++index)
+ children[index]._restore();
+
+ this._savedChildren = null;
+ }
+}
+
+WebInspector.ProfileDataGridTree.propertyComparators = [{}, {}];
+
+WebInspector.ProfileDataGridTree.propertyComparator = function(/*String*/ property, /*Boolean*/ isAscending)
+{
+ var comparator = this.propertyComparators[(isAscending ? 1 : 0)][property];
+
+ if (!comparator) {
+ if (isAscending) {
+ comparator = function(lhs, rhs)
+ {
+ if (lhs[property] < rhs[property])
+ return -1;
+
+ if (lhs[property] > rhs[property])
+ return 1;
+
+ return 0;
+ }
+ } else {
+ comparator = function(lhs, rhs)
+ {
+ if (lhs[property] > rhs[property])
+ return -1;
+
+ if (lhs[property] < rhs[property])
+ return 1;
+
+ return 0;
+ }
+ }
+
+ this.propertyComparators[(isAscending ? 1 : 0)][property] = comparator;
+ }
+
+ return comparator;
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileView.js
new file mode 100644
index 0000000..2b8c6ce
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfileView.js
@@ -0,0 +1,539 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ProfileView = function(profile)
+{
+ WebInspector.View.call(this);
+
+ this.element.addStyleClass("profile-view");
+
+ this.showSelfTimeAsPercent = true;
+ this.showTotalTimeAsPercent = true;
+ this.showAverageTimeAsPercent = true;
+
+ var columns = { "self": { title: WebInspector.UIString("Self"), width: "72px", sort: "descending", sortable: true },
+ "total": { title: WebInspector.UIString("Total"), width: "72px", sortable: true },
+ "average": { title: WebInspector.UIString("Average"), width: "72px", sortable: true },
+ "calls": { title: WebInspector.UIString("Calls"), width: "54px", sortable: true },
+ "function": { title: WebInspector.UIString("Function"), disclosure: true, sortable: true } };
+
+ if (Preferences.samplingCPUProfiler) {
+ delete columns.average;
+ delete columns.calls;
+ }
+
+ this.dataGrid = new WebInspector.DataGrid(columns);
+ this.dataGrid.addEventListener("sorting changed", this._sortData, this);
+ this.dataGrid.element.addEventListener("mousedown", this._mouseDownInDataGrid.bind(this), true);
+ this.element.appendChild(this.dataGrid.element);
+
+ this.viewSelectElement = document.createElement("select");
+ this.viewSelectElement.className = "status-bar-item";
+ this.viewSelectElement.addEventListener("change", this._changeView.bind(this), false);
+ this.view = "Heavy";
+
+ var heavyViewOption = document.createElement("option");
+ heavyViewOption.label = WebInspector.UIString("Heavy (Bottom Up)");
+ var treeViewOption = document.createElement("option");
+ treeViewOption.label = WebInspector.UIString("Tree (Top Down)");
+ this.viewSelectElement.appendChild(heavyViewOption);
+ this.viewSelectElement.appendChild(treeViewOption);
+
+ this.percentButton = new WebInspector.StatusBarButton("", "percent-time-status-bar-item");
+ this.percentButton.addEventListener("click", this._percentClicked.bind(this), false);
+
+ this.focusButton = new WebInspector.StatusBarButton(WebInspector.UIString("Focus selected function."), "focus-profile-node-status-bar-item");
+ this.focusButton.disabled = true;
+ this.focusButton.addEventListener("click", this._focusClicked.bind(this), false);
+
+ this.excludeButton = new WebInspector.StatusBarButton(WebInspector.UIString("Exclude selected function."), "exclude-profile-node-status-bar-item");
+ this.excludeButton.disabled = true;
+ this.excludeButton.addEventListener("click", this._excludeClicked.bind(this), false);
+
+ this.resetButton = new WebInspector.StatusBarButton(WebInspector.UIString("Restore all functions."), "reset-profile-status-bar-item");
+ this.resetButton.visible = false;
+ this.resetButton.addEventListener("click", this._resetClicked.bind(this), false);
+
+ this.profile = profile;
+
+ this.profileDataGridTree = this.bottomUpProfileDataGridTree;
+ this.profileDataGridTree.sort(WebInspector.ProfileDataGridTree.propertyComparator("selfTime", false));
+
+ this.refresh();
+
+ this._updatePercentButton();
+}
+
+WebInspector.ProfileView.prototype = {
+ get statusBarItems()
+ {
+ return [this.viewSelectElement, this.percentButton.element, this.focusButton.element, this.excludeButton.element, this.resetButton.element];
+ },
+
+ get profile()
+ {
+ return this._profile;
+ },
+
+ set profile(profile)
+ {
+ this._profile = profile;
+ },
+
+ get bottomUpProfileDataGridTree()
+ {
+ if (!this._bottomUpProfileDataGridTree)
+ this._bottomUpProfileDataGridTree = new WebInspector.BottomUpProfileDataGridTree(this, this.profile.head);
+ return this._bottomUpProfileDataGridTree;
+ },
+
+ get topDownProfileDataGridTree()
+ {
+ if (!this._topDownProfileDataGridTree)
+ this._topDownProfileDataGridTree = new WebInspector.TopDownProfileDataGridTree(this, this.profile.head);
+ return this._topDownProfileDataGridTree;
+ },
+
+ get currentTree()
+ {
+ return this._currentTree;
+ },
+
+ set currentTree(tree)
+ {
+ this._currentTree = tree;
+ this.refresh();
+ },
+
+ get topDownTree()
+ {
+ if (!this._topDownTree) {
+ this._topDownTree = WebInspector.TopDownTreeFactory.create(this.profile.head);
+ this._sortProfile(this._topDownTree);
+ }
+
+ return this._topDownTree;
+ },
+
+ get bottomUpTree()
+ {
+ if (!this._bottomUpTree) {
+ this._bottomUpTree = WebInspector.BottomUpTreeFactory.create(this.profile.head);
+ this._sortProfile(this._bottomUpTree);
+ }
+
+ return this._bottomUpTree;
+ },
+
+ show: function(parentElement)
+ {
+ WebInspector.View.prototype.show.call(this, parentElement);
+ this.dataGrid.updateWidths();
+ },
+
+ hide: function()
+ {
+ WebInspector.View.prototype.hide.call(this);
+ this._currentSearchResultIndex = -1;
+ },
+
+ resize: function()
+ {
+ if (this.dataGrid)
+ this.dataGrid.updateWidths();
+ },
+
+ refresh: function()
+ {
+ var selectedProfileNode = this.dataGrid.selectedNode ? this.dataGrid.selectedNode.profileNode : null;
+
+ this.dataGrid.removeChildren();
+
+ var children = this.profileDataGridTree.children;
+ var count = children.length;
+
+ for (var index = 0; index < count; ++index)
+ this.dataGrid.appendChild(children[index]);
+
+ if (selectedProfileNode)
+ selectedProfileNode.selected = true;
+ },
+
+ refreshVisibleData: function()
+ {
+ var child = this.dataGrid.children[0];
+ while (child) {
+ child.refresh();
+ child = child.traverseNextNode(false, null, true);
+ }
+ },
+
+ refreshShowAsPercents: function()
+ {
+ this._updatePercentButton();
+ this.refreshVisibleData();
+ },
+
+ searchCanceled: function()
+ {
+ if (this._searchResults) {
+ for (var i = 0; i < this._searchResults.length; ++i) {
+ var profileNode = this._searchResults[i].profileNode;
+
+ delete profileNode._searchMatchedSelfColumn;
+ delete profileNode._searchMatchedTotalColumn;
+ delete profileNode._searchMatchedCallsColumn;
+ delete profileNode._searchMatchedFunctionColumn;
+
+ profileNode.refresh();
+ }
+ }
+
+ delete this._searchFinishedCallback;
+ this._currentSearchResultIndex = -1;
+ this._searchResults = [];
+ },
+
+ performSearch: function(query, finishedCallback)
+ {
+ // Call searchCanceled since it will reset everything we need before doing a new search.
+ this.searchCanceled();
+
+ query = query.trimWhitespace();
+
+ if (!query.length)
+ return;
+
+ this._searchFinishedCallback = finishedCallback;
+
+ var greaterThan = (query.indexOf(">") === 0);
+ var lessThan = (query.indexOf("<") === 0);
+ var equalTo = (query.indexOf("=") === 0 || ((greaterThan || lessThan) && query.indexOf("=") === 1));
+ var percentUnits = (query.lastIndexOf("%") === (query.length - 1));
+ var millisecondsUnits = (query.length > 2 && query.lastIndexOf("ms") === (query.length - 2));
+ var secondsUnits = (!millisecondsUnits && query.lastIndexOf("s") === (query.length - 1));
+
+ var queryNumber = parseFloat(query);
+ if (greaterThan || lessThan || equalTo) {
+ if (equalTo && (greaterThan || lessThan))
+ queryNumber = parseFloat(query.substring(2));
+ else
+ queryNumber = parseFloat(query.substring(1));
+ }
+
+ var queryNumberMilliseconds = (secondsUnits ? (queryNumber * 1000) : queryNumber);
+
+ // Make equalTo implicitly true if it wasn't specified there is no other operator.
+ if (!isNaN(queryNumber) && !(greaterThan || lessThan))
+ equalTo = true;
+
+ function matchesQuery(/*ProfileDataGridNode*/ profileDataGridNode)
+ {
+ delete profileDataGridNode._searchMatchedSelfColumn;
+ delete profileDataGridNode._searchMatchedTotalColumn;
+ delete profileDataGridNode._searchMatchedAverageColumn;
+ delete profileDataGridNode._searchMatchedCallsColumn;
+ delete profileDataGridNode._searchMatchedFunctionColumn;
+
+ if (percentUnits) {
+ if (lessThan) {
+ if (profileDataGridNode.selfPercent < queryNumber)
+ profileDataGridNode._searchMatchedSelfColumn = true;
+ if (profileDataGridNode.totalPercent < queryNumber)
+ profileDataGridNode._searchMatchedTotalColumn = true;
+ if (profileDataGridNode.averagePercent < queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedAverageColumn = true;
+ } else if (greaterThan) {
+ if (profileDataGridNode.selfPercent > queryNumber)
+ profileDataGridNode._searchMatchedSelfColumn = true;
+ if (profileDataGridNode.totalPercent > queryNumber)
+ profileDataGridNode._searchMatchedTotalColumn = true;
+ if (profileDataGridNode.averagePercent < queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedAverageColumn = true;
+ }
+
+ if (equalTo) {
+ if (profileDataGridNode.selfPercent == queryNumber)
+ profileDataGridNode._searchMatchedSelfColumn = true;
+ if (profileDataGridNode.totalPercent == queryNumber)
+ profileDataGridNode._searchMatchedTotalColumn = true;
+ if (profileDataGridNode.averagePercent < queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedAverageColumn = true;
+ }
+ } else if (millisecondsUnits || secondsUnits) {
+ if (lessThan) {
+ if (profileDataGridNode.selfTime < queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedSelfColumn = true;
+ if (profileDataGridNode.totalTime < queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedTotalColumn = true;
+ if (profileDataGridNode.averageTime < queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedAverageColumn = true;
+ } else if (greaterThan) {
+ if (profileDataGridNode.selfTime > queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedSelfColumn = true;
+ if (profileDataGridNode.totalTime > queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedTotalColumn = true;
+ if (profileDataGridNode.averageTime > queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedAverageColumn = true;
+ }
+
+ if (equalTo) {
+ if (profileDataGridNode.selfTime == queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedSelfColumn = true;
+ if (profileDataGridNode.totalTime == queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedTotalColumn = true;
+ if (profileDataGridNode.averageTime == queryNumberMilliseconds)
+ profileDataGridNode._searchMatchedAverageColumn = true;
+ }
+ } else {
+ if (equalTo && profileDataGridNode.numberOfCalls == queryNumber)
+ profileDataGridNode._searchMatchedCallsColumn = true;
+ if (greaterThan && profileDataGridNode.numberOfCalls > queryNumber)
+ profileDataGridNode._searchMatchedCallsColumn = true;
+ if (lessThan && profileDataGridNode.numberOfCalls < queryNumber)
+ profileDataGridNode._searchMatchedCallsColumn = true;
+ }
+
+ if (profileDataGridNode.functionName.hasSubstring(query, true) || profileDataGridNode.url.hasSubstring(query, true))
+ profileDataGridNode._searchMatchedFunctionColumn = true;
+
+ if (profileDataGridNode._searchMatchedSelfColumn ||
+ profileDataGridNode._searchMatchedTotalColumn ||
+ profileDataGridNode._searchMatchedAverageColumn ||
+ profileDataGridNode._searchMatchedCallsColumn ||
+ profileDataGridNode._searchMatchedFunctionColumn)
+ {
+ profileDataGridNode.refresh();
+ return true;
+ }
+
+ return false;
+ }
+
+ var current = this.profileDataGridTree.children[0];
+
+ while (current) {
+ if (matchesQuery(current)) {
+ this._searchResults.push({ profileNode: current });
+ }
+
+ current = current.traverseNextNode(false, null, false);
+ }
+
+ finishedCallback(this, this._searchResults.length);
+ },
+
+ jumpToFirstSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ this._currentSearchResultIndex = 0;
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ jumpToLastSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ this._currentSearchResultIndex = (this._searchResults.length - 1);
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ jumpToNextSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ if (++this._currentSearchResultIndex >= this._searchResults.length)
+ this._currentSearchResultIndex = 0;
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ jumpToPreviousSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ if (--this._currentSearchResultIndex < 0)
+ this._currentSearchResultIndex = (this._searchResults.length - 1);
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ showingFirstSearchResult: function()
+ {
+ return (this._currentSearchResultIndex === 0);
+ },
+
+ showingLastSearchResult: function()
+ {
+ return (this._searchResults && this._currentSearchResultIndex === (this._searchResults.length - 1));
+ },
+
+ _jumpToSearchResult: function(index)
+ {
+ var searchResult = this._searchResults[index];
+ if (!searchResult)
+ return;
+
+ var profileNode = searchResult.profileNode;
+ profileNode.reveal();
+ profileNode.select();
+ },
+
+ _changeView: function(event)
+ {
+ if (!event || !this.profile)
+ return;
+
+ if (event.target.selectedIndex == 1 && this.view == "Heavy") {
+ this.profileDataGridTree = this.topDownProfileDataGridTree;
+ this._sortProfile();
+ this.view = "Tree";
+ } else if (event.target.selectedIndex == 0 && this.view == "Tree") {
+ this.profileDataGridTree = this.bottomUpProfileDataGridTree;
+ this._sortProfile();
+ this.view = "Heavy";
+ }
+
+ if (!this.currentQuery || !this._searchFinishedCallback || !this._searchResults)
+ return;
+
+ // The current search needs to be performed again. First negate out previous match
+ // count by calling the search finished callback with a negative number of matches.
+ // Then perform the search again the with same query and callback.
+ this._searchFinishedCallback(this, -this._searchResults.length);
+ this.performSearch(this.currentQuery, this._searchFinishedCallback);
+ },
+
+ _percentClicked: function(event)
+ {
+ var currentState = this.showSelfTimeAsPercent && this.showTotalTimeAsPercent && this.showAverageTimeAsPercent;
+ this.showSelfTimeAsPercent = !currentState;
+ this.showTotalTimeAsPercent = !currentState;
+ this.showAverageTimeAsPercent = !currentState;
+ this.refreshShowAsPercents();
+ },
+
+ _updatePercentButton: function()
+ {
+ if (this.showSelfTimeAsPercent && this.showTotalTimeAsPercent && this.showAverageTimeAsPercent) {
+ this.percentButton.title = WebInspector.UIString("Show absolute total and self times.");
+ this.percentButton.toggled = true;
+ } else {
+ this.percentButton.title = WebInspector.UIString("Show total and self times as percentages.");
+ this.percentButton.toggled = false;
+ }
+ },
+
+ _focusClicked: function(event)
+ {
+ if (!this.dataGrid.selectedNode)
+ return;
+
+ this.resetButton.visible = true;
+ this.profileDataGridTree.focus(this.dataGrid.selectedNode);
+ this.refresh();
+ this.refreshVisibleData();
+ },
+
+ _excludeClicked: function(event)
+ {
+ var selectedNode = this.dataGrid.selectedNode
+
+ if (!selectedNode)
+ return;
+
+ selectedNode.deselect();
+
+ this.resetButton.visible = true;
+ this.profileDataGridTree.exclude(selectedNode);
+ this.refresh();
+ this.refreshVisibleData();
+ },
+
+ _resetClicked: function(event)
+ {
+ this.resetButton.visible = false;
+ this.profileDataGridTree.restore();
+ this.refresh();
+ this.refreshVisibleData();
+ },
+
+ _dataGridNodeSelected: function(node)
+ {
+ this.focusButton.disabled = false;
+ this.excludeButton.disabled = false;
+ },
+
+ _dataGridNodeDeselected: function(node)
+ {
+ this.focusButton.disabled = true;
+ this.excludeButton.disabled = true;
+ },
+
+ _sortData: function(event)
+ {
+ this._sortProfile(this.profile);
+ },
+
+ _sortProfile: function()
+ {
+ var sortAscending = this.dataGrid.sortOrder === "ascending";
+ var sortColumnIdentifier = this.dataGrid.sortColumnIdentifier;
+ var sortProperty = {
+ "average": "averageTime",
+ "self": "selfTime",
+ "total": "totalTime",
+ "calls": "numberOfCalls",
+ "function": "functionName"
+ }[sortColumnIdentifier];
+
+ this.profileDataGridTree.sort(WebInspector.ProfileDataGridTree.propertyComparator(sortProperty, sortAscending));
+
+ this.refresh();
+ },
+
+ _mouseDownInDataGrid: function(event)
+ {
+ if (event.detail < 2)
+ return;
+
+ var cell = event.target.enclosingNodeOrSelfWithNodeName("td");
+ if (!cell || (!cell.hasStyleClass("total-column") && !cell.hasStyleClass("self-column") && !cell.hasStyleClass("average-column")))
+ return;
+
+ if (cell.hasStyleClass("total-column"))
+ this.showTotalTimeAsPercent = !this.showTotalTimeAsPercent;
+ else if (cell.hasStyleClass("self-column"))
+ this.showSelfTimeAsPercent = !this.showSelfTimeAsPercent;
+ else if (cell.hasStyleClass("average-column"))
+ this.showAverageTimeAsPercent = !this.showAverageTimeAsPercent;
+
+ this.refreshShowAsPercents();
+
+ event.preventDefault();
+ event.stopPropagation();
+ }
+}
+
+WebInspector.ProfileView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfilesPanel.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfilesPanel.js
new file mode 100644
index 0000000..c010033
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ProfilesPanel.js
@@ -0,0 +1,534 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+const UserInitiatedProfileName = "org.webkit.profiles.user-initiated";
+
+WebInspector.ProfilesPanel = function()
+{
+ WebInspector.Panel.call(this);
+
+ this.element.addStyleClass("profiles");
+
+ var panelEnablerHeading = WebInspector.UIString("You need to enable profiling before you can use the Profiles panel.");
+ var panelEnablerDisclaimer = WebInspector.UIString("Enabling profiling will make scripts run slower.");
+ var panelEnablerButton = WebInspector.UIString("Enable Profiling");
+ this.panelEnablerView = new WebInspector.PanelEnablerView("profiles", panelEnablerHeading, panelEnablerDisclaimer, panelEnablerButton);
+ this.panelEnablerView.addEventListener("enable clicked", this._enableProfiling, this);
+
+ this.element.appendChild(this.panelEnablerView.element);
+
+ this.sidebarElement = document.createElement("div");
+ this.sidebarElement.id = "profiles-sidebar";
+ this.sidebarElement.className = "sidebar";
+ this.element.appendChild(this.sidebarElement);
+
+ this.sidebarResizeElement = document.createElement("div");
+ this.sidebarResizeElement.className = "sidebar-resizer-vertical";
+ this.sidebarResizeElement.addEventListener("mousedown", this._startSidebarDragging.bind(this), false);
+ this.element.appendChild(this.sidebarResizeElement);
+
+ this.sidebarTreeElement = document.createElement("ol");
+ this.sidebarTreeElement.className = "sidebar-tree";
+ this.sidebarElement.appendChild(this.sidebarTreeElement);
+
+ this.sidebarTree = new TreeOutline(this.sidebarTreeElement);
+
+ this.profilesListTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("CPU PROFILES"), null, true);
+ this.sidebarTree.appendChild(this.profilesListTreeElement);
+ this.profilesListTreeElement.expand();
+
+ this.snapshotsListTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("HEAP SNAPSHOTS"), null, true);
+ if (Preferences.heapProfilerPresent) {
+ this.sidebarTree.appendChild(this.snapshotsListTreeElement);
+ this.snapshotsListTreeElement.expand();
+ }
+
+ this.profileViews = document.createElement("div");
+ this.profileViews.id = "profile-views";
+ this.element.appendChild(this.profileViews);
+
+ this.enableToggleButton = new WebInspector.StatusBarButton("", "enable-toggle-status-bar-item");
+ this.enableToggleButton.addEventListener("click", this._toggleProfiling.bind(this), false);
+
+ this.recordButton = new WebInspector.StatusBarButton(WebInspector.UIString("Start profiling."), "record-profile-status-bar-item");
+ this.recordButton.addEventListener("click", this._recordClicked.bind(this), false);
+
+ this.recording = false;
+
+ this.snapshotButton = new WebInspector.StatusBarButton(WebInspector.UIString("Take heap snapshot."), "heap-snapshot-status-bar-item");
+ this.snapshotButton.visible = Preferences.heapProfilerPresent;
+ this.snapshotButton.addEventListener("click", this._snapshotClicked.bind(this), false);
+
+ this.profileViewStatusBarItemsContainer = document.createElement("div");
+ this.profileViewStatusBarItemsContainer.id = "profile-view-status-bar-items";
+
+ this.reset();
+}
+
+WebInspector.ProfilesPanel.prototype = {
+ toolbarItemClass: "profiles",
+
+ get toolbarItemLabel()
+ {
+ return WebInspector.UIString("Profiles");
+ },
+
+ get statusBarItems()
+ {
+ return [this.enableToggleButton.element, this.recordButton.element, this.snapshotButton.element, this.profileViewStatusBarItemsContainer];
+ },
+
+ show: function()
+ {
+ WebInspector.Panel.prototype.show.call(this);
+ this._updateSidebarWidth();
+ if (this._shouldPopulateProfiles)
+ this._populateProfiles();
+ },
+
+ populateInterface: function()
+ {
+ if (this.visible)
+ this._populateProfiles();
+ else
+ this._shouldPopulateProfiles = true;
+ },
+
+ profilerWasEnabled: function()
+ {
+ this.reset();
+ this.populateInterface();
+ },
+
+ profilerWasDisabled: function()
+ {
+ this.reset();
+ },
+
+ reset: function()
+ {
+ if (this._profiles) {
+ var profiledLength = this._profiles.length;
+ for (var i = 0; i < profiledLength; ++i) {
+ var profile = this._profiles[i];
+ delete profile._profileView;
+ }
+ }
+
+ delete this.currentQuery;
+ this.searchCanceled();
+
+ this._profiles = [];
+ this._profilesIdMap = {};
+ this._profileGroups = {};
+ this._profileGroupsForLinks = {}
+
+ this.sidebarTreeElement.removeStyleClass("some-expandable");
+
+ this.profilesListTreeElement.removeChildren();
+ this.snapshotsListTreeElement.removeChildren();
+ this.profileViews.removeChildren();
+
+ this.profileViewStatusBarItemsContainer.removeChildren();
+
+ this._updateInterface();
+ },
+
+ handleKeyEvent: function(event)
+ {
+ this.sidebarTree.handleKeyEvent(event);
+ },
+
+ addProfile: function(profile)
+ {
+ this._profiles.push(profile);
+ this._profilesIdMap[profile.uid] = profile;
+
+ var sidebarParent = this.profilesListTreeElement;
+ var small = false;
+ var alternateTitle;
+
+ if (profile.title.indexOf(UserInitiatedProfileName) !== 0) {
+ if (!(profile.title in this._profileGroups))
+ this._profileGroups[profile.title] = [];
+
+ var group = this._profileGroups[profile.title];
+ group.push(profile);
+
+ if (group.length === 2) {
+ // Make a group TreeElement now that there are 2 profiles.
+ group._profilesTreeElement = new WebInspector.ProfileGroupSidebarTreeElement(profile.title);
+
+ // Insert at the same index for the first profile of the group.
+ var index = this.sidebarTree.children.indexOf(group[0]._profilesTreeElement);
+ this.sidebarTree.insertChild(group._profilesTreeElement, index);
+
+ // Move the first profile to the group.
+ var selected = group[0]._profilesTreeElement.selected;
+ this.sidebarTree.removeChild(group[0]._profilesTreeElement);
+ group._profilesTreeElement.appendChild(group[0]._profilesTreeElement);
+ if (selected) {
+ group[0]._profilesTreeElement.select();
+ group[0]._profilesTreeElement.reveal();
+ }
+
+ group[0]._profilesTreeElement.small = true;
+ group[0]._profilesTreeElement.mainTitle = WebInspector.UIString("Run %d", 1);
+
+ this.sidebarTreeElement.addStyleClass("some-expandable");
+ }
+
+ if (group.length >= 2) {
+ sidebarParent = group._profilesTreeElement;
+ alternateTitle = WebInspector.UIString("Run %d", group.length);
+ small = true;
+ }
+ }
+
+ var profileTreeElement = new WebInspector.ProfileSidebarTreeElement(profile);
+ profileTreeElement.small = small;
+ if (alternateTitle)
+ profileTreeElement.mainTitle = alternateTitle;
+ profile._profilesTreeElement = profileTreeElement;
+
+ sidebarParent.appendChild(profileTreeElement);
+ },
+
+ showProfile: function(profile)
+ {
+ if (!profile)
+ return;
+
+ if (this.visibleView)
+ this.visibleView.hide();
+
+ var view = this.profileViewForProfile(profile);
+
+ view.show(this.profileViews);
+
+ profile._profilesTreeElement.select(true);
+ profile._profilesTreeElement.reveal();
+
+ this.visibleView = view;
+
+ this.profileViewStatusBarItemsContainer.removeChildren();
+
+ var statusBarItems = view.statusBarItems;
+ for (var i = 0; i < statusBarItems.length; ++i)
+ this.profileViewStatusBarItemsContainer.appendChild(statusBarItems[i]);
+ },
+
+ showView: function(view)
+ {
+ this.showProfile(view.profile);
+ },
+
+ profileViewForProfile: function(profile)
+ {
+ if (!profile)
+ return null;
+ if (!profile._profileView)
+ profile._profileView = new WebInspector.ProfileView(profile);
+ return profile._profileView;
+ },
+
+ showProfileById: function(uid)
+ {
+ this.showProfile(this._profilesIdMap[uid]);
+ },
+
+ closeVisibleView: function()
+ {
+ if (this.visibleView)
+ this.visibleView.hide();
+ delete this.visibleView;
+ },
+
+ displayTitleForProfileLink: function(title)
+ {
+ title = unescape(title);
+ if (title.indexOf(UserInitiatedProfileName) === 0) {
+ title = WebInspector.UIString("Profile %d", title.substring(UserInitiatedProfileName.length + 1));
+ } else {
+ if (!(title in this._profileGroupsForLinks))
+ this._profileGroupsForLinks[title] = 0;
+
+ groupNumber = ++this._profileGroupsForLinks[title];
+
+ if (groupNumber > 2)
+ // The title is used in the console message announcing that a profile has started so it gets
+ // incremented twice as often as it's displayed
+ title += " " + WebInspector.UIString("Run %d", groupNumber / 2);
+ }
+
+ return title;
+ },
+
+ get searchableViews()
+ {
+ var views = [];
+
+ const visibleView = this.visibleView;
+ if (visibleView && visibleView.performSearch)
+ views.push(visibleView);
+
+ var profilesLength = this._profiles.length;
+ for (var i = 0; i < profilesLength; ++i) {
+ var view = this.profileViewForProfile(this._profiles[i]);
+ if (!view.performSearch || view === visibleView)
+ continue;
+ views.push(view);
+ }
+
+ return views;
+ },
+
+ searchMatchFound: function(view, matches)
+ {
+ view.profile._profilesTreeElement.searchMatches = matches;
+ },
+
+ searchCanceled: function(startingNewSearch)
+ {
+ WebInspector.Panel.prototype.searchCanceled.call(this, startingNewSearch);
+
+ if (!this._profiles)
+ return;
+
+ for (var i = 0; i < this._profiles.length; ++i) {
+ var profile = this._profiles[i];
+ profile._profilesTreeElement.searchMatches = 0;
+ }
+ },
+
+ setRecordingProfile: function(isProfiling)
+ {
+ this.recording = isProfiling;
+
+ if (isProfiling) {
+ this.recordButton.toggled = true;
+ this.recordButton.title = WebInspector.UIString("Stop profiling.");
+ } else {
+ this.recordButton.toggled = false;
+ this.recordButton.title = WebInspector.UIString("Start profiling.");
+ }
+ },
+
+ resize: function()
+ {
+ var visibleView = this.visibleView;
+ if (visibleView && "resize" in visibleView)
+ visibleView.resize();
+ },
+
+ _updateInterface: function()
+ {
+ if (InspectorController.profilerEnabled()) {
+ this.enableToggleButton.title = WebInspector.UIString("Profiling enabled. Click to disable.");
+ this.enableToggleButton.toggled = true;
+ this.recordButton.visible = true;
+ if (Preferences.heapProfilerPresent)
+ this.snapshotButton.visible = true;
+ this.profileViewStatusBarItemsContainer.removeStyleClass("hidden");
+ this.panelEnablerView.visible = false;
+ } else {
+ this.enableToggleButton.title = WebInspector.UIString("Profiling disabled. Click to enable.");
+ this.enableToggleButton.toggled = false;
+ this.recordButton.visible = false;
+ this.snapshotButton.visible = false;
+ this.profileViewStatusBarItemsContainer.addStyleClass("hidden");
+ this.panelEnablerView.visible = true;
+ }
+ },
+
+ _recordClicked: function()
+ {
+ this.recording = !this.recording;
+
+ if (this.recording)
+ InspectorController.startProfiling();
+ else
+ InspectorController.stopProfiling();
+ },
+
+ _snapshotClicked: function()
+ {
+ InspectorController.takeHeapSnapshot();
+ },
+
+ _enableProfiling: function()
+ {
+ if (InspectorController.profilerEnabled())
+ return;
+ this._toggleProfiling(this.panelEnablerView.alwaysEnabled);
+ },
+
+ _toggleProfiling: function(optionalAlways)
+ {
+ if (InspectorController.profilerEnabled())
+ InspectorController.disableProfiler(true);
+ else
+ InspectorController.enableProfiler(!!optionalAlways);
+ },
+
+ _populateProfiles: function()
+ {
+ if (this.sidebarTree.children.length)
+ return;
+
+ var profiles = InspectorController.profiles();
+ var profilesLength = profiles.length;
+ for (var i = 0; i < profilesLength; ++i) {
+ var profile = profiles[i];
+ this.addProfile(profile);
+ }
+
+ if (this.sidebarTree.children[0])
+ this.sidebarTree.children[0].select();
+
+ delete this._shouldPopulateProfiles;
+ },
+
+ _startSidebarDragging: function(event)
+ {
+ WebInspector.elementDragStart(this.sidebarResizeElement, this._sidebarDragging.bind(this), this._endSidebarDragging.bind(this), event, "col-resize");
+ },
+
+ _sidebarDragging: function(event)
+ {
+ this._updateSidebarWidth(event.pageX);
+
+ event.preventDefault();
+ },
+
+ _endSidebarDragging: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+ },
+
+ _updateSidebarWidth: function(width)
+ {
+ if (this.sidebarElement.offsetWidth <= 0) {
+ // The stylesheet hasn't loaded yet or the window is closed,
+ // so we can't calculate what is need. Return early.
+ return;
+ }
+
+ if (!("_currentSidebarWidth" in this))
+ this._currentSidebarWidth = this.sidebarElement.offsetWidth;
+
+ if (typeof width === "undefined")
+ width = this._currentSidebarWidth;
+
+ width = Number.constrain(width, Preferences.minSidebarWidth, window.innerWidth / 2);
+
+ this._currentSidebarWidth = width;
+
+ this.sidebarElement.style.width = width + "px";
+ this.profileViews.style.left = width + "px";
+ this.profileViewStatusBarItemsContainer.style.left = width + "px";
+ this.sidebarResizeElement.style.left = (width - 3) + "px";
+
+ var visibleView = this.visibleView;
+ if (visibleView && "resize" in visibleView)
+ visibleView.resize();
+ }
+}
+
+WebInspector.ProfilesPanel.prototype.__proto__ = WebInspector.Panel.prototype;
+
+WebInspector.ProfileSidebarTreeElement = function(profile)
+{
+ this.profile = profile;
+
+ if (this.profile.title.indexOf(UserInitiatedProfileName) === 0)
+ this._profileNumber = this.profile.title.substring(UserInitiatedProfileName.length + 1);
+
+ WebInspector.SidebarTreeElement.call(this, "profile-sidebar-tree-item", "", "", profile, false);
+
+ this.refreshTitles();
+}
+
+WebInspector.ProfileSidebarTreeElement.prototype = {
+ onselect: function()
+ {
+ WebInspector.panels.profiles.showProfile(this.profile);
+ },
+
+ get mainTitle()
+ {
+ if (this._mainTitle)
+ return this._mainTitle;
+ if (this.profile.title.indexOf(UserInitiatedProfileName) === 0)
+ return WebInspector.UIString("Profile %d", this._profileNumber);
+ return this.profile.title;
+ },
+
+ set mainTitle(x)
+ {
+ this._mainTitle = x;
+ this.refreshTitles();
+ },
+
+ get subtitle()
+ {
+ // There is no subtitle.
+ },
+
+ set subtitle(x)
+ {
+ // Can't change subtitle.
+ },
+
+ set searchMatches(matches)
+ {
+ if (!matches) {
+ if (!this.bubbleElement)
+ return;
+ this.bubbleElement.removeStyleClass("search-matches");
+ this.bubbleText = "";
+ return;
+ }
+
+ this.bubbleText = matches;
+ this.bubbleElement.addStyleClass("search-matches");
+ }
+}
+
+WebInspector.ProfileSidebarTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
+
+WebInspector.ProfileGroupSidebarTreeElement = function(title, subtitle)
+{
+ WebInspector.SidebarTreeElement.call(this, "profile-group-sidebar-tree-item", title, subtitle, null, true);
+}
+
+WebInspector.ProfileGroupSidebarTreeElement.prototype = {
+ onselect: function()
+ {
+ WebInspector.panels.profiles.showProfile(this.children[this.children.length - 1].profile);
+ }
+}
+
+WebInspector.ProfileGroupSidebarTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSection.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSection.js
new file mode 100644
index 0000000..a4b2fba
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSection.js
@@ -0,0 +1,145 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.PropertiesSection = function(title, subtitle)
+{
+ this.element = document.createElement("div");
+ this.element.className = "section";
+
+ this.headerElement = document.createElement("div");
+ this.headerElement.className = "header";
+
+ this.titleElement = document.createElement("div");
+ this.titleElement.className = "title";
+
+ this.subtitleElement = document.createElement("div");
+ this.subtitleElement.className = "subtitle";
+
+ this.headerElement.appendChild(this.subtitleElement);
+ this.headerElement.appendChild(this.titleElement);
+
+ this.headerElement.addEventListener("click", this.toggleExpanded.bind(this), false);
+
+ this.propertiesElement = document.createElement("ol");
+ this.propertiesElement.className = "properties";
+ this.propertiesTreeOutline = new TreeOutline(this.propertiesElement);
+ this.propertiesTreeOutline.section = this;
+
+ this.element.appendChild(this.headerElement);
+ this.element.appendChild(this.propertiesElement);
+
+ this.title = title;
+ this.subtitle = subtitle;
+ this._expanded = false;
+}
+
+WebInspector.PropertiesSection.prototype = {
+ get title()
+ {
+ return this._title;
+ },
+
+ set title(x)
+ {
+ if (this._title === x)
+ return;
+ this._title = x;
+
+ if (x instanceof Node) {
+ this.titleElement.removeChildren();
+ this.titleElement.appendChild(x);
+ } else
+ this.titleElement.textContent = x;
+ },
+
+ get subtitle()
+ {
+ return this._subtitle;
+ },
+
+ set subtitle(x)
+ {
+ if (this._subtitle === x)
+ return;
+ this._subtitle = x;
+ this.subtitleElement.innerHTML = x;
+ },
+
+ get expanded()
+ {
+ return this._expanded;
+ },
+
+ set expanded(x)
+ {
+ if (x)
+ this.expand();
+ else
+ this.collapse();
+ },
+
+ get populated()
+ {
+ return this._populated;
+ },
+
+ set populated(x)
+ {
+ this._populated = x;
+ if (!x && this.onpopulate && this._expanded) {
+ this.onpopulate(this);
+ this._populated = true;
+ }
+ },
+
+ expand: function()
+ {
+ if (this._expanded)
+ return;
+ this._expanded = true;
+ this.element.addStyleClass("expanded");
+
+ if (!this._populated && this.onpopulate) {
+ this.onpopulate(this);
+ this._populated = true;
+ }
+ },
+
+ collapse: function()
+ {
+ if (!this._expanded)
+ return;
+ this._expanded = false;
+ this.element.removeStyleClass("expanded");
+ },
+
+ toggleExpanded: function()
+ {
+ this.expanded = !this.expanded;
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSidebarPane.js
new file mode 100644
index 0000000..ec08210
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/PropertiesSidebarPane.js
@@ -0,0 +1,64 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.PropertiesSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Properties"));
+}
+
+WebInspector.PropertiesSidebarPane.prototype = {
+ update: function(node)
+ {
+ var body = this.bodyElement;
+
+ body.removeChildren();
+
+ this.sections = [];
+
+ if (!node)
+ return;
+
+ var self = this;
+ var callback = function(prototypes) {
+ var body = self.bodyElement;
+ body.removeChildren();
+ self.sections = [];
+
+ // Get array of prototype user-friendly names.
+ for (var i = 0; i < prototypes.length; ++i) {
+ var prototype = new WebInspector.ObjectProxy(node.id, [], i);
+ var section = new WebInspector.ObjectPropertiesSection(prototype, prototypes[i], WebInspector.UIString("Prototype"));
+ self.sections.push(section);
+ body.appendChild(section.element);
+ }
+ };
+ InjectedScriptAccess.getPrototypes(node.id, callback);
+ }
+}
+
+WebInspector.PropertiesSidebarPane.prototype.__proto__ = WebInspector.SidebarPane.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Resource.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Resource.js
new file mode 100644
index 0000000..4dac093
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Resource.js
@@ -0,0 +1,622 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Resource = function(requestHeaders, url, domain, path, lastPathComponent, identifier, mainResource, cached, requestMethod, requestFormData)
+{
+ this.identifier = identifier;
+
+ this.startTime = -1;
+ this.endTime = -1;
+ this.mainResource = mainResource;
+ this.requestHeaders = requestHeaders;
+ this.url = url;
+ this.domain = domain;
+ this.path = path;
+ this.lastPathComponent = lastPathComponent;
+ this.cached = cached;
+ this.requestMethod = requestMethod || "";
+ this.requestFormData = requestFormData || "";
+
+ this.category = WebInspector.resourceCategories.other;
+}
+
+// Keep these in sync with WebCore::InspectorResource::Type
+WebInspector.Resource.Type = {
+ Document: 0,
+ Stylesheet: 1,
+ Image: 2,
+ Font: 3,
+ Script: 4,
+ XHR: 5,
+ Other: 6,
+
+ isTextType: function(type)
+ {
+ return (type === this.Document) || (type === this.Stylesheet) || (type === this.Script) || (type === this.XHR);
+ },
+
+ toString: function(type)
+ {
+ switch (type) {
+ case this.Document:
+ return WebInspector.UIString("document");
+ case this.Stylesheet:
+ return WebInspector.UIString("stylesheet");
+ case this.Image:
+ return WebInspector.UIString("image");
+ case this.Font:
+ return WebInspector.UIString("font");
+ case this.Script:
+ return WebInspector.UIString("script");
+ case this.XHR:
+ return WebInspector.UIString("XHR");
+ case this.Other:
+ default:
+ return WebInspector.UIString("other");
+ }
+ }
+}
+
+WebInspector.Resource.prototype = {
+ get url()
+ {
+ return this._url;
+ },
+
+ set url(x)
+ {
+ if (this._url === x)
+ return;
+
+ var oldURL = this._url;
+ this._url = x;
+
+ // FIXME: We should make the WebInspector object listen for the "url changed" event.
+ // Then resourceURLChanged can be removed.
+ WebInspector.resourceURLChanged(this, oldURL);
+
+ this.dispatchEventToListeners("url changed");
+ },
+
+ get domain()
+ {
+ return this._domain;
+ },
+
+ set domain(x)
+ {
+ if (this._domain === x)
+ return;
+ this._domain = x;
+ },
+
+ get lastPathComponent()
+ {
+ return this._lastPathComponent;
+ },
+
+ set lastPathComponent(x)
+ {
+ if (this._lastPathComponent === x)
+ return;
+ this._lastPathComponent = x;
+ this._lastPathComponentLowerCase = x ? x.toLowerCase() : null;
+ },
+
+ get displayName()
+ {
+ var title = this.lastPathComponent;
+ if (!title)
+ title = this.displayDomain;
+ if (!title && this.url)
+ title = this.url.trimURL(WebInspector.mainResource ? WebInspector.mainResource.domain : "");
+ if (title === "/")
+ title = this.url;
+ return title;
+ },
+
+ get displayDomain()
+ {
+ // WebInspector.Database calls this, so don't access more than this.domain.
+ if (this.domain && (!WebInspector.mainResource || (WebInspector.mainResource && this.domain !== WebInspector.mainResource.domain)))
+ return this.domain;
+ return "";
+ },
+
+ get startTime()
+ {
+ return this._startTime || -1;
+ },
+
+ set startTime(x)
+ {
+ if (this._startTime === x)
+ return;
+
+ this._startTime = x;
+
+ if (WebInspector.panels.resources)
+ WebInspector.panels.resources.refreshResource(this);
+ },
+
+ get responseReceivedTime()
+ {
+ return this._responseReceivedTime || -1;
+ },
+
+ set responseReceivedTime(x)
+ {
+ if (this._responseReceivedTime === x)
+ return;
+
+ this._responseReceivedTime = x;
+
+ if (WebInspector.panels.resources)
+ WebInspector.panels.resources.refreshResource(this);
+ },
+
+ get endTime()
+ {
+ return this._endTime || -1;
+ },
+
+ set endTime(x)
+ {
+ if (this._endTime === x)
+ return;
+
+ this._endTime = x;
+
+ if (WebInspector.panels.resources)
+ WebInspector.panels.resources.refreshResource(this);
+ },
+
+ get duration()
+ {
+ if (this._endTime === -1 || this._startTime === -1)
+ return -1;
+ return this._endTime - this._startTime;
+ },
+
+ get latency()
+ {
+ if (this._responseReceivedTime === -1 || this._startTime === -1)
+ return -1;
+ return this._responseReceivedTime - this._startTime;
+ },
+
+ get contentLength()
+ {
+ return this._contentLength || 0;
+ },
+
+ set contentLength(x)
+ {
+ if (this._contentLength === x)
+ return;
+
+ this._contentLength = x;
+
+ if (WebInspector.panels.resources)
+ WebInspector.panels.resources.refreshResource(this);
+ },
+
+ get expectedContentLength()
+ {
+ return this._expectedContentLength || 0;
+ },
+
+ set expectedContentLength(x)
+ {
+ if (this._expectedContentLength === x)
+ return;
+ this._expectedContentLength = x;
+ },
+
+ get finished()
+ {
+ return this._finished;
+ },
+
+ set finished(x)
+ {
+ if (this._finished === x)
+ return;
+
+ this._finished = x;
+
+ if (x) {
+ this._checkTips();
+ this._checkWarnings();
+ this.dispatchEventToListeners("finished");
+ }
+ },
+
+ get failed()
+ {
+ return this._failed;
+ },
+
+ set failed(x)
+ {
+ this._failed = x;
+ },
+
+ get category()
+ {
+ return this._category;
+ },
+
+ set category(x)
+ {
+ if (this._category === x)
+ return;
+
+ var oldCategory = this._category;
+ if (oldCategory)
+ oldCategory.removeResource(this);
+
+ this._category = x;
+
+ if (this._category)
+ this._category.addResource(this);
+
+ if (WebInspector.panels.resources) {
+ WebInspector.panels.resources.refreshResource(this);
+ WebInspector.panels.resources.recreateViewForResourceIfNeeded(this);
+ }
+ },
+
+ get mimeType()
+ {
+ return this._mimeType;
+ },
+
+ set mimeType(x)
+ {
+ if (this._mimeType === x)
+ return;
+
+ this._mimeType = x;
+ },
+
+ get type()
+ {
+ return this._type;
+ },
+
+ set type(x)
+ {
+ if (this._type === x)
+ return;
+
+ this._type = x;
+
+ switch (x) {
+ case WebInspector.Resource.Type.Document:
+ this.category = WebInspector.resourceCategories.documents;
+ break;
+ case WebInspector.Resource.Type.Stylesheet:
+ this.category = WebInspector.resourceCategories.stylesheets;
+ break;
+ case WebInspector.Resource.Type.Script:
+ this.category = WebInspector.resourceCategories.scripts;
+ break;
+ case WebInspector.Resource.Type.Image:
+ this.category = WebInspector.resourceCategories.images;
+ break;
+ case WebInspector.Resource.Type.Font:
+ this.category = WebInspector.resourceCategories.fonts;
+ break;
+ case WebInspector.Resource.Type.XHR:
+ this.category = WebInspector.resourceCategories.xhr;
+ break;
+ case WebInspector.Resource.Type.Other:
+ default:
+ this.category = WebInspector.resourceCategories.other;
+ break;
+ }
+ },
+
+ get requestHeaders()
+ {
+ if (this._requestHeaders === undefined)
+ this._requestHeaders = {};
+ return this._requestHeaders;
+ },
+
+ set requestHeaders(x)
+ {
+ if (this._requestHeaders === x)
+ return;
+
+ this._requestHeaders = x;
+ delete this._sortedRequestHeaders;
+
+ this.dispatchEventToListeners("requestHeaders changed");
+ },
+
+ get sortedRequestHeaders()
+ {
+ if (this._sortedRequestHeaders !== undefined)
+ return this._sortedRequestHeaders;
+
+ this._sortedRequestHeaders = [];
+ for (var key in this.requestHeaders)
+ this._sortedRequestHeaders.push({header: key, value: this.requestHeaders[key]});
+ this._sortedRequestHeaders.sort(function(a,b) { return a.header.localeCompare(b.header) });
+
+ return this._sortedRequestHeaders;
+ },
+
+ get responseHeaders()
+ {
+ if (this._responseHeaders === undefined)
+ this._responseHeaders = {};
+ return this._responseHeaders;
+ },
+
+ set responseHeaders(x)
+ {
+ if (this._responseHeaders === x)
+ return;
+
+ this._responseHeaders = x;
+ delete this._sortedResponseHeaders;
+
+ this.dispatchEventToListeners("responseHeaders changed");
+ },
+
+ get sortedResponseHeaders()
+ {
+ if (this._sortedResponseHeaders !== undefined)
+ return this._sortedResponseHeaders;
+
+ this._sortedResponseHeaders = [];
+ for (var key in this.responseHeaders)
+ this._sortedResponseHeaders.push({header: key, value: this.responseHeaders[key]});
+ this._sortedResponseHeaders.sort(function(a,b) { return a.header.localeCompare(b.header) });
+
+ return this._sortedResponseHeaders;
+ },
+
+ get scripts()
+ {
+ if (!("_scripts" in this))
+ this._scripts = [];
+ return this._scripts;
+ },
+
+ addScript: function(script)
+ {
+ if (!script)
+ return;
+ this.scripts.unshift(script);
+ script.resource = this;
+ },
+
+ removeAllScripts: function()
+ {
+ if (!this._scripts)
+ return;
+
+ for (var i = 0; i < this._scripts.length; ++i) {
+ if (this._scripts[i].resource === this)
+ delete this._scripts[i].resource;
+ }
+
+ delete this._scripts;
+ },
+
+ removeScript: function(script)
+ {
+ if (!script)
+ return;
+
+ if (script.resource === this)
+ delete script.resource;
+
+ if (!this._scripts)
+ return;
+
+ this._scripts.remove(script);
+ },
+
+ get errors()
+ {
+ return this._errors || 0;
+ },
+
+ set errors(x)
+ {
+ this._errors = x;
+ },
+
+ get warnings()
+ {
+ return this._warnings || 0;
+ },
+
+ set warnings(x)
+ {
+ this._warnings = x;
+ },
+
+ get tips()
+ {
+ if (!("_tips" in this))
+ this._tips = {};
+ return this._tips;
+ },
+
+ _addTip: function(tip)
+ {
+ if (tip.id in this.tips)
+ return;
+
+ this.tips[tip.id] = tip;
+
+ // FIXME: Re-enable this code once we have a scope bar in the Console.
+ // Otherwise, we flood the Console with too many tips.
+ /*
+ var msg = new WebInspector.ConsoleMessage(WebInspector.ConsoleMessage.MessageSource.Other,
+ WebInspector.ConsoleMessage.MessageType.Log, WebInspector.ConsoleMessage.MessageLevel.Tip,
+ -1, this.url, null, 1, tip.message);
+ WebInspector.console.addMessage(msg);
+ */
+ },
+
+ _checkTips: function()
+ {
+ for (var tip in WebInspector.Tips)
+ this._checkTip(WebInspector.Tips[tip]);
+ },
+
+ _checkTip: function(tip)
+ {
+ var addTip = false;
+ switch (tip.id) {
+ case WebInspector.Tips.ResourceNotCompressed.id:
+ addTip = this._shouldCompress();
+ break;
+ }
+
+ if (addTip)
+ this._addTip(tip);
+ },
+
+ _shouldCompress: function()
+ {
+ return WebInspector.Resource.Type.isTextType(this.type)
+ && this.domain
+ && !("Content-Encoding" in this.responseHeaders)
+ && this.contentLength !== undefined
+ && this.contentLength >= 512;
+ },
+
+ _mimeTypeIsConsistentWithType: function()
+ {
+ if (typeof this.type === "undefined"
+ || this.type === WebInspector.Resource.Type.Other
+ || this.type === WebInspector.Resource.Type.XHR)
+ return true;
+
+ if (this.mimeType in WebInspector.MIMETypes)
+ return this.type in WebInspector.MIMETypes[this.mimeType];
+
+ return true;
+ },
+
+ _checkWarnings: function()
+ {
+ for (var warning in WebInspector.Warnings)
+ this._checkWarning(WebInspector.Warnings[warning]);
+ },
+
+ _checkWarning: function(warning)
+ {
+ var msg;
+ switch (warning.id) {
+ case WebInspector.Warnings.IncorrectMIMEType.id:
+ if (!this._mimeTypeIsConsistentWithType())
+ msg = new WebInspector.ConsoleMessage(WebInspector.ConsoleMessage.MessageSource.Other,
+ WebInspector.ConsoleMessage.MessageType.Log,
+ WebInspector.ConsoleMessage.MessageLevel.Warning, -1, this.url, null, 1,
+ String.sprintf(WebInspector.Warnings.IncorrectMIMEType.message,
+ WebInspector.Resource.Type.toString(this.type), this.mimeType));
+ break;
+ }
+
+ if (msg)
+ WebInspector.console.addMessage(msg);
+ }
+}
+
+WebInspector.Resource.prototype.__proto__ = WebInspector.Object.prototype;
+
+WebInspector.Resource.CompareByStartTime = function(a, b)
+{
+ if (a.startTime < b.startTime)
+ return -1;
+ if (a.startTime > b.startTime)
+ return 1;
+ return 0;
+}
+
+WebInspector.Resource.CompareByResponseReceivedTime = function(a, b)
+{
+ if (a.responseReceivedTime === -1 && b.responseReceivedTime !== -1)
+ return 1;
+ if (a.responseReceivedTime !== -1 && b.responseReceivedTime === -1)
+ return -1;
+ if (a.responseReceivedTime < b.responseReceivedTime)
+ return -1;
+ if (a.responseReceivedTime > b.responseReceivedTime)
+ return 1;
+ return 0;
+}
+
+WebInspector.Resource.CompareByEndTime = function(a, b)
+{
+ if (a.endTime === -1 && b.endTime !== -1)
+ return 1;
+ if (a.endTime !== -1 && b.endTime === -1)
+ return -1;
+ if (a.endTime < b.endTime)
+ return -1;
+ if (a.endTime > b.endTime)
+ return 1;
+ return 0;
+}
+
+WebInspector.Resource.CompareByDuration = function(a, b)
+{
+ if (a.duration < b.duration)
+ return -1;
+ if (a.duration > b.duration)
+ return 1;
+ return 0;
+}
+
+WebInspector.Resource.CompareByLatency = function(a, b)
+{
+ if (a.latency < b.latency)
+ return -1;
+ if (a.latency > b.latency)
+ return 1;
+ return 0;
+}
+
+WebInspector.Resource.CompareBySize = function(a, b)
+{
+ if (a.contentLength < b.contentLength)
+ return -1;
+ if (a.contentLength > b.contentLength)
+ return 1;
+ return 0;
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceCategory.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceCategory.js
new file mode 100644
index 0000000..fc508d0
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceCategory.js
@@ -0,0 +1,68 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ResourceCategory = function(title, name)
+{
+ this.name = name;
+ this.title = title;
+ this.resources = [];
+}
+
+WebInspector.ResourceCategory.prototype = {
+ toString: function()
+ {
+ return this.title;
+ },
+
+ addResource: function(resource)
+ {
+ var a = resource;
+ var resourcesLength = this.resources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var b = this.resources[i];
+ if (a._lastPathComponentLowerCase && b._lastPathComponentLowerCase)
+ if (a._lastPathComponentLowerCase < b._lastPathComponentLowerCase)
+ break;
+ else if (a.name && b.name)
+ if (a.name < b.name)
+ break;
+ }
+
+ this.resources.splice(i, 0, resource);
+ },
+
+ removeResource: function(resource)
+ {
+ this.resources.remove(resource, true);
+ },
+
+ removeAllResources: function(resource)
+ {
+ this.resources = [];
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceView.js
new file mode 100644
index 0000000..d745920
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourceView.js
@@ -0,0 +1,263 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) IBM Corp. 2009 All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ResourceView = function(resource)
+{
+ WebInspector.View.call(this);
+
+ this.element.addStyleClass("resource-view");
+
+ this.resource = resource;
+
+ this.headersElement = document.createElement("div");
+ this.headersElement.className = "resource-view-headers";
+ this.element.appendChild(this.headersElement);
+
+ this.contentElement = document.createElement("div");
+ this.contentElement.className = "resource-view-content";
+ this.element.appendChild(this.contentElement);
+
+ this.headersListElement = document.createElement("ol");
+ this.headersListElement.className = "outline-disclosure";
+ this.headersElement.appendChild(this.headersListElement);
+
+ this.headersTreeOutline = new TreeOutline(this.headersListElement);
+ this.headersTreeOutline.expandTreeElementsWhenArrowing = true;
+
+ this.urlTreeElement = new TreeElement("", null, false);
+ this.urlTreeElement.selectable = false;
+ this.headersTreeOutline.appendChild(this.urlTreeElement);
+
+ this.requestHeadersTreeElement = new TreeElement("", null, true);
+ this.requestHeadersTreeElement.expanded = false;
+ this.requestHeadersTreeElement.selectable = false;
+ this.headersTreeOutline.appendChild(this.requestHeadersTreeElement);
+
+ this._decodeHover = WebInspector.UIString("Double-Click to toggle between URL encoded and decoded formats");
+ this._decodeRequestParameters = true;
+
+ this.queryStringTreeElement = new TreeElement("", null, true);
+ this.queryStringTreeElement.expanded = false;
+ this.queryStringTreeElement.selectable = false;
+ this.queryStringTreeElement.hidden = true;
+ this.headersTreeOutline.appendChild(this.queryStringTreeElement);
+
+ this.formDataTreeElement = new TreeElement("", null, true);
+ this.formDataTreeElement.expanded = false;
+ this.formDataTreeElement.selectable = false;
+ this.formDataTreeElement.hidden = true;
+ this.headersTreeOutline.appendChild(this.formDataTreeElement);
+
+ this.requestPayloadTreeElement = new TreeElement(WebInspector.UIString("Request Payload"), null, true);
+ this.requestPayloadTreeElement.expanded = false;
+ this.requestPayloadTreeElement.selectable = false;
+ this.requestPayloadTreeElement.hidden = true;
+ this.headersTreeOutline.appendChild(this.requestPayloadTreeElement);
+
+ this.responseHeadersTreeElement = new TreeElement("", null, true);
+ this.responseHeadersTreeElement.expanded = false;
+ this.responseHeadersTreeElement.selectable = false;
+ this.headersTreeOutline.appendChild(this.responseHeadersTreeElement);
+
+ this.headersVisible = true;
+
+ resource.addEventListener("url changed", this._refreshURL, this);
+ resource.addEventListener("requestHeaders changed", this._refreshRequestHeaders, this);
+ resource.addEventListener("responseHeaders changed", this._refreshResponseHeaders, this);
+
+ this._refreshURL();
+ this._refreshRequestHeaders();
+ this._refreshResponseHeaders();
+}
+
+WebInspector.ResourceView.prototype = {
+ get headersVisible()
+ {
+ return this._headersVisible;
+ },
+
+ set headersVisible(x)
+ {
+ if (x === this._headersVisible)
+ return;
+
+ this._headersVisible = x;
+
+ if (x)
+ this.element.addStyleClass("headers-visible");
+ else
+ this.element.removeStyleClass("headers-visible");
+ },
+
+ attach: function()
+ {
+ if (!this.element.parentNode) {
+ var parentElement = (document.getElementById("resource-views") || document.getElementById("script-resource-views"));
+ if (parentElement)
+ parentElement.appendChild(this.element);
+ }
+ },
+
+ _refreshURL: function()
+ {
+ var url = this.resource.url;
+ this.urlTreeElement.title = this.resource.requestMethod + " " + url.escapeHTML();
+ this._refreshQueryString();
+ },
+
+ _refreshQueryString: function()
+ {
+ var url = this.resource.url;
+ var hasQueryString = url.indexOf("?") >= 0;
+
+ if (!hasQueryString) {
+ this.queryStringTreeElement.hidden = true;
+ return;
+ }
+
+ this.queryStringTreeElement.hidden = false;
+ var parmString = url.split("?", 2)[1];
+ this._refreshParms(WebInspector.UIString("Query String Parameters"), parmString, this.queryStringTreeElement);
+ },
+
+ _refreshFormData: function()
+ {
+ this.formDataTreeElement.hidden = true;
+ this.requestPayloadTreeElement.hidden = true;
+
+ var isFormData = this.resource.requestFormData;
+ if (!isFormData)
+ return;
+
+ var isFormEncoded = false;
+ var requestContentType = this._getHeaderValue(this.resource.requestHeaders, "Content-Type");
+ if (requestContentType == "application/x-www-form-urlencoded")
+ isFormEncoded = true;
+
+ if (isFormEncoded) {
+ this.formDataTreeElement.hidden = false;
+ this._refreshParms(WebInspector.UIString("Form Data"), this.resource.requestFormData, this.formDataTreeElement);
+ } else {
+ this.requestPayloadTreeElement.hidden = false;
+ this._refreshRequestPayload(this.resource.requestFormData);
+ }
+ },
+
+ _refreshRequestPayload: function(formData)
+ {
+ this.requestPayloadTreeElement.removeChildren();
+
+ var title = "<div class=\"header-name\">&nbsp;</div>";
+ title += "<div class=\"raw-form-data header-value\">" + formData.escapeHTML() + "</div>";
+ var parmTreeElement = new TreeElement(title, null, false);
+ this.requestPayloadTreeElement.appendChild(parmTreeElement);
+ },
+
+ _refreshParms: function(title, parmString, parmsTreeElement)
+ {
+ var parms = parmString.split("&");
+ for (var i = 0; i < parms.length; ++i) {
+ var parm = parms[i];
+ parm = parm.split("=", 2);
+ if (parm.length == 1)
+ parm.push("");
+ parms[i] = parm;
+ }
+
+ parmsTreeElement.removeChildren();
+
+ parmsTreeElement.title = title + "<span class=\"header-count\">" + WebInspector.UIString(" (%d)", parms.length) + "</span>";
+
+ for (var i = 0; i < parms.length; ++i) {
+ var key = parms[i][0];
+ var val = parms[i][1];
+
+ if (val.indexOf("%") >= 0)
+ if (this._decodeRequestParameters)
+ val = decodeURIComponent(val).replace(/\+/g, " ");
+
+ var title = "<div class=\"header-name\">" + key.escapeHTML() + ":</div>";
+ title += "<div class=\"header-value\">" + val.escapeHTML() + "</div>";
+
+ var parmTreeElement = new TreeElement(title, null, false);
+ parmTreeElement.selectable = false;
+ parmTreeElement.tooltip = this._decodeHover;
+ parmTreeElement.ondblclick = this._toggleURLdecoding.bind(this);
+ parmsTreeElement.appendChild(parmTreeElement);
+ }
+ },
+
+ _toggleURLdecoding: function(treeElement, event)
+ {
+ this._decodeRequestParameters = !this._decodeRequestParameters;
+ this._refreshQueryString();
+ this._refreshFormData();
+ },
+
+ _getHeaderValue: function(headers, key)
+ {
+ var lowerKey = key.toLowerCase();
+ for (var testKey in headers) {
+ if (testKey.toLowerCase() === lowerKey)
+ return headers[testKey];
+ }
+ },
+
+ _refreshRequestHeaders: function()
+ {
+ this._refreshHeaders(WebInspector.UIString("Request Headers"), this.resource.sortedRequestHeaders, this.requestHeadersTreeElement);
+ this._refreshFormData();
+ },
+
+ _refreshResponseHeaders: function()
+ {
+ this._refreshHeaders(WebInspector.UIString("Response Headers"), this.resource.sortedResponseHeaders, this.responseHeadersTreeElement);
+ },
+
+ _refreshHeaders: function(title, headers, headersTreeElement)
+ {
+ headersTreeElement.removeChildren();
+
+ var length = headers.length;
+ headersTreeElement.title = title.escapeHTML() + "<span class=\"header-count\">" + WebInspector.UIString(" (%d)", length) + "</span>";
+ headersTreeElement.hidden = !length;
+
+ var length = headers.length;
+ for (var i = 0; i < length; ++i) {
+ var title = "<div class=\"header-name\">" + headers[i].header.escapeHTML() + ":</div>";
+ title += "<div class=\"header-value\">" + headers[i].value.escapeHTML() + "</div>"
+
+ var headerTreeElement = new TreeElement(title, null, false);
+ headerTreeElement.selectable = false;
+ headersTreeElement.appendChild(headerTreeElement);
+ }
+ }
+}
+
+WebInspector.ResourceView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourcesPanel.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourcesPanel.js
new file mode 100644
index 0000000..680f66c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ResourcesPanel.js
@@ -0,0 +1,1462 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2008, 2009 Anthony Ricaud <rik@webkit.org>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ResourcesPanel = function()
+{
+ WebInspector.Panel.call(this);
+
+ this.element.addStyleClass("resources");
+
+ this.filterBarElement = document.createElement("div");
+ this.filterBarElement.id = "resources-filter";
+ this.element.appendChild(this.filterBarElement);
+
+ this.viewsContainerElement = document.createElement("div");
+ this.viewsContainerElement.id = "resource-views";
+ this.element.appendChild(this.viewsContainerElement);
+
+ this.containerElement = document.createElement("div");
+ this.containerElement.id = "resources-container";
+ this.containerElement.addEventListener("scroll", this._updateDividersLabelBarPosition.bind(this), false);
+ this.element.appendChild(this.containerElement);
+
+ this.sidebarElement = document.createElement("div");
+ this.sidebarElement.id = "resources-sidebar";
+ this.sidebarElement.className = "sidebar";
+ this.containerElement.appendChild(this.sidebarElement);
+
+ this.sidebarResizeElement = document.createElement("div");
+ this.sidebarResizeElement.className = "sidebar-resizer-vertical";
+ this.sidebarResizeElement.addEventListener("mousedown", this._startSidebarDragging.bind(this), false);
+ this.element.appendChild(this.sidebarResizeElement);
+
+ this.containerContentElement = document.createElement("div");
+ this.containerContentElement.id = "resources-container-content";
+ this.containerElement.appendChild(this.containerContentElement);
+
+ this.summaryBar = new WebInspector.SummaryBar(this.categories);
+ this.summaryBar.element.id = "resources-summary";
+ this.containerContentElement.appendChild(this.summaryBar.element);
+
+ this.resourcesGraphsElement = document.createElement("div");
+ this.resourcesGraphsElement.id = "resources-graphs";
+ this.containerContentElement.appendChild(this.resourcesGraphsElement);
+
+ this.dividersElement = document.createElement("div");
+ this.dividersElement.id = "resources-dividers";
+ this.containerContentElement.appendChild(this.dividersElement);
+
+ this.dividersLabelBarElement = document.createElement("div");
+ this.dividersLabelBarElement.id = "resources-dividers-label-bar";
+ this.containerContentElement.appendChild(this.dividersLabelBarElement);
+
+ this.sidebarTreeElement = document.createElement("ol");
+ this.sidebarTreeElement.className = "sidebar-tree";
+ this.sidebarElement.appendChild(this.sidebarTreeElement);
+
+ this.sidebarTree = new TreeOutline(this.sidebarTreeElement);
+
+ var timeGraphItem = new WebInspector.SidebarTreeElement("resources-time-graph-sidebar-item", WebInspector.UIString("Time"));
+ timeGraphItem.onselect = this._graphSelected.bind(this);
+
+ var transferTimeCalculator = new WebInspector.ResourceTransferTimeCalculator();
+ var transferDurationCalculator = new WebInspector.ResourceTransferDurationCalculator();
+
+ timeGraphItem.sortingOptions = [
+ { name: WebInspector.UIString("Sort by Start Time"), sortingFunction: WebInspector.ResourceSidebarTreeElement.CompareByAscendingStartTime, calculator: transferTimeCalculator },
+ { name: WebInspector.UIString("Sort by Response Time"), sortingFunction: WebInspector.ResourceSidebarTreeElement.CompareByAscendingResponseReceivedTime, calculator: transferTimeCalculator },
+ { name: WebInspector.UIString("Sort by End Time"), sortingFunction: WebInspector.ResourceSidebarTreeElement.CompareByAscendingEndTime, calculator: transferTimeCalculator },
+ { name: WebInspector.UIString("Sort by Duration"), sortingFunction: WebInspector.ResourceSidebarTreeElement.CompareByDescendingDuration, calculator: transferDurationCalculator },
+ { name: WebInspector.UIString("Sort by Latency"), sortingFunction: WebInspector.ResourceSidebarTreeElement.CompareByDescendingLatency, calculator: transferDurationCalculator },
+ ];
+
+ timeGraphItem.selectedSortingOptionIndex = 1;
+
+ var sizeGraphItem = new WebInspector.SidebarTreeElement("resources-size-graph-sidebar-item", WebInspector.UIString("Size"));
+ sizeGraphItem.onselect = this._graphSelected.bind(this);
+
+ var transferSizeCalculator = new WebInspector.ResourceTransferSizeCalculator();
+ sizeGraphItem.sortingOptions = [
+ { name: WebInspector.UIString("Sort by Size"), sortingFunction: WebInspector.ResourceSidebarTreeElement.CompareByDescendingSize, calculator: transferSizeCalculator },
+ ];
+
+ sizeGraphItem.selectedSortingOptionIndex = 0;
+
+ this.graphsTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("GRAPHS"), {}, true);
+ this.sidebarTree.appendChild(this.graphsTreeElement);
+
+ this.graphsTreeElement.appendChild(timeGraphItem);
+ this.graphsTreeElement.appendChild(sizeGraphItem);
+ this.graphsTreeElement.expand();
+
+ this.resourcesTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("RESOURCES"), {}, true);
+ this.sidebarTree.appendChild(this.resourcesTreeElement);
+
+ this.resourcesTreeElement.expand();
+
+ var panelEnablerHeading = WebInspector.UIString("You need to enable resource tracking to use this panel.");
+ var panelEnablerDisclaimer = WebInspector.UIString("Enabling resource tracking will reload the page and make page loading slower.");
+ var panelEnablerButton = WebInspector.UIString("Enable resource tracking");
+
+ this.panelEnablerView = new WebInspector.PanelEnablerView("resources", panelEnablerHeading, panelEnablerDisclaimer, panelEnablerButton);
+ this.panelEnablerView.addEventListener("enable clicked", this._enableResourceTracking, this);
+
+ this.element.appendChild(this.panelEnablerView.element);
+
+ this.enableToggleButton = new WebInspector.StatusBarButton("", "enable-toggle-status-bar-item");
+ this.enableToggleButton.addEventListener("click", this._toggleResourceTracking.bind(this), false);
+
+ this.largerResourcesButton = new WebInspector.StatusBarButton(WebInspector.UIString("Use small resource rows."), "resources-larger-resources-status-bar-item");
+ this.largerResourcesButton.toggled = true;
+ this.largerResourcesButton.addEventListener("click", this._toggleLargerResources.bind(this), false);
+
+ this.sortingSelectElement = document.createElement("select");
+ this.sortingSelectElement.className = "status-bar-item";
+ this.sortingSelectElement.addEventListener("change", this._changeSortingFunction.bind(this), false);
+
+ var createFilterElement = function (category) {
+ var categoryElement = document.createElement("li");
+ categoryElement.category = category;
+ categoryElement.addStyleClass(category);
+ var label = WebInspector.UIString("All");
+ if (WebInspector.resourceCategories[category])
+ label = WebInspector.resourceCategories[category].title;
+ categoryElement.appendChild(document.createTextNode(label));
+ categoryElement.addEventListener("click", this._updateFilter.bind(this), false);
+ this.filterBarElement.appendChild(categoryElement);
+ return categoryElement;
+ };
+
+ var allElement = createFilterElement.call(this, "all");
+ this.filter(allElement.category);
+ for (var category in this.categories)
+ createFilterElement.call(this, category);
+
+ this.reset();
+
+ timeGraphItem.select();
+}
+
+WebInspector.ResourcesPanel.prototype = {
+ toolbarItemClass: "resources",
+
+ get categories()
+ {
+ if (!this._categories) {
+ this._categories = {documents: {color: {r: 47, g: 102, b: 236}}, stylesheets: {color: {r: 157, g: 231, b: 119}}, images: {color: {r: 164, g: 60, b: 255}}, scripts: {color: {r: 255, g: 121, b: 0}}, xhr: {color: {r: 231, g: 231, b: 10}}, fonts: {color: {r: 255, g: 82, b: 62}}, other: {color: {r: 186, g: 186, b: 186}}};
+ for (var category in this._categories) {
+ this._categories[category].title = WebInspector.resourceCategories[category].title;
+ }
+ }
+ return this._categories;
+ },
+
+ filter: function (category) {
+ if (this._filterCategory && this._filterCategory === category)
+ return;
+
+ if (this._filterCategory) {
+ var filterElement = this.filterBarElement.getElementsByClassName(this._filterCategory)[0];
+ filterElement.removeStyleClass("selected");
+ var oldClass = "filter-" + this._filterCategory;
+ this.resourcesTreeElement.childrenListElement.removeStyleClass(oldClass);
+ this.resourcesGraphsElement.removeStyleClass(oldClass);
+ }
+ this._filterCategory = category;
+ var filterElement = this.filterBarElement.getElementsByClassName(this._filterCategory)[0];
+ filterElement.addStyleClass("selected");
+ var newClass = "filter-" + this._filterCategory;
+ this.resourcesTreeElement.childrenListElement.addStyleClass(newClass);
+ this.resourcesGraphsElement.addStyleClass(newClass);
+ },
+
+ _updateFilter: function (e) {
+ this.filter(e.target.category);
+ },
+
+ get toolbarItemLabel()
+ {
+ return WebInspector.UIString("Resources");
+ },
+
+ get statusBarItems()
+ {
+ return [this.enableToggleButton.element, this.largerResourcesButton.element, this.sortingSelectElement];
+ },
+
+ show: function()
+ {
+ WebInspector.Panel.prototype.show.call(this);
+
+ this._updateDividersLabelBarPosition();
+ this._updateSidebarWidth();
+ this.refreshIfNeeded();
+
+ var visibleView = this.visibleView;
+ if (visibleView) {
+ visibleView.headersVisible = true;
+ visibleView.show(this.viewsContainerElement);
+ }
+
+ // Hide any views that are visible that are not this panel's current visible view.
+ // This can happen when a ResourceView is visible in the Scripts panel then switched
+ // to the this panel.
+ var resourcesLength = this._resources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var resource = this._resources[i];
+ var view = resource._resourcesView;
+ if (!view || view === visibleView)
+ continue;
+ view.visible = false;
+ }
+ },
+
+ resize: function()
+ {
+ this._updateGraphDividersIfNeeded();
+
+ var visibleView = this.visibleView;
+ if (visibleView && "resize" in visibleView)
+ visibleView.resize();
+ },
+
+ get searchableViews()
+ {
+ var views = [];
+
+ const visibleView = this.visibleView;
+ if (visibleView && visibleView.performSearch)
+ views.push(visibleView);
+
+ var resourcesLength = this._resources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var resource = this._resources[i];
+ if (!resource._resourcesTreeElement)
+ continue;
+ var resourceView = this.resourceViewForResource(resource);
+ if (!resourceView.performSearch || resourceView === visibleView)
+ continue;
+ views.push(resourceView);
+ }
+
+ return views;
+ },
+
+ get searchResultsSortFunction()
+ {
+ const resourceTreeElementSortFunction = this.sortingFunction;
+
+ function sortFuction(a, b)
+ {
+ return resourceTreeElementSortFunction(a.resource._resourcesTreeElement, b.resource._resourcesTreeElement);
+ }
+
+ return sortFuction;
+ },
+
+ searchMatchFound: function(view, matches)
+ {
+ view.resource._resourcesTreeElement.searchMatches = matches;
+ },
+
+ searchCanceled: function(startingNewSearch)
+ {
+ WebInspector.Panel.prototype.searchCanceled.call(this, startingNewSearch);
+
+ if (startingNewSearch || !this._resources)
+ return;
+
+ for (var i = 0; i < this._resources.length; ++i) {
+ var resource = this._resources[i];
+ if (resource._resourcesTreeElement)
+ resource._resourcesTreeElement.updateErrorsAndWarnings();
+ }
+ },
+
+ performSearch: function(query)
+ {
+ for (var i = 0; i < this._resources.length; ++i) {
+ var resource = this._resources[i];
+ if (resource._resourcesTreeElement)
+ resource._resourcesTreeElement.resetBubble();
+ }
+
+ WebInspector.Panel.prototype.performSearch.call(this, query);
+ },
+
+ get visibleView()
+ {
+ if (this.visibleResource)
+ return this.visibleResource._resourcesView;
+ return null;
+ },
+
+ get calculator()
+ {
+ return this._calculator;
+ },
+
+ set calculator(x)
+ {
+ if (!x || this._calculator === x)
+ return;
+
+ this._calculator = x;
+ this._calculator.reset();
+
+ this._staleResources = this._resources;
+ this.refresh();
+ },
+
+ get sortingFunction()
+ {
+ return this._sortingFunction;
+ },
+
+ set sortingFunction(x)
+ {
+ this._sortingFunction = x;
+ this._sortResourcesIfNeeded();
+ },
+
+ get needsRefresh()
+ {
+ return this._needsRefresh;
+ },
+
+ set needsRefresh(x)
+ {
+ if (this._needsRefresh === x)
+ return;
+
+ this._needsRefresh = x;
+
+ if (x) {
+ if (this.visible && !("_refreshTimeout" in this))
+ this._refreshTimeout = setTimeout(this.refresh.bind(this), 500);
+ } else {
+ if ("_refreshTimeout" in this) {
+ clearTimeout(this._refreshTimeout);
+ delete this._refreshTimeout;
+ }
+ }
+ },
+
+ refreshIfNeeded: function()
+ {
+ if (this.needsRefresh)
+ this.refresh();
+ },
+
+ refresh: function()
+ {
+ this.needsRefresh = false;
+
+ var staleResourcesLength = this._staleResources.length;
+ var boundariesChanged = false;
+
+ for (var i = 0; i < staleResourcesLength; ++i) {
+ var resource = this._staleResources[i];
+ if (!resource._resourcesTreeElement) {
+ // Create the resource tree element and graph.
+ resource._resourcesTreeElement = new WebInspector.ResourceSidebarTreeElement(resource);
+ resource._resourcesTreeElement._resourceGraph = new WebInspector.ResourceGraph(resource);
+
+ this.resourcesTreeElement.appendChild(resource._resourcesTreeElement);
+ this.resourcesGraphsElement.appendChild(resource._resourcesTreeElement._resourceGraph.graphElement);
+ }
+
+ resource._resourcesTreeElement.refresh();
+
+ if (this.calculator.updateBoundaries(resource))
+ boundariesChanged = true;
+ }
+
+ if (boundariesChanged) {
+ // The boundaries changed, so all resource graphs are stale.
+ this._staleResources = this._resources;
+ staleResourcesLength = this._staleResources.length;
+ }
+
+ for (var i = 0; i < staleResourcesLength; ++i)
+ this._staleResources[i]._resourcesTreeElement._resourceGraph.refresh(this.calculator);
+
+ this._staleResources = [];
+
+ this._updateGraphDividersIfNeeded();
+ this._sortResourcesIfNeeded();
+ this._updateSummaryGraph();
+ },
+
+ resourceTrackingWasEnabled: function()
+ {
+ this.reset();
+ },
+
+ resourceTrackingWasDisabled: function()
+ {
+ this.reset();
+ },
+
+ reset: function()
+ {
+ this.closeVisibleResource();
+
+ this.containerElement.scrollTop = 0;
+
+ delete this.currentQuery;
+ this.searchCanceled();
+
+ if (this._calculator)
+ this._calculator.reset();
+
+ if (this._resources) {
+ var resourcesLength = this._resources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var resource = this._resources[i];
+
+ resource.warnings = 0;
+ resource.errors = 0;
+
+ delete resource._resourcesTreeElement;
+ delete resource._resourcesView;
+ }
+ }
+
+ this._resources = [];
+ this._staleResources = [];
+
+ this.resourcesTreeElement.removeChildren();
+ this.viewsContainerElement.removeChildren();
+ this.resourcesGraphsElement.removeChildren();
+ this.summaryBar.reset();
+
+ this._updateGraphDividersIfNeeded(true);
+
+ if (InspectorController.resourceTrackingEnabled()) {
+ this.enableToggleButton.title = WebInspector.UIString("Resource tracking enabled. Click to disable.");
+ this.enableToggleButton.toggled = true;
+ this.largerResourcesButton.visible = true;
+ this.sortingSelectElement.removeStyleClass("hidden");
+ this.panelEnablerView.visible = false;
+ } else {
+ this.enableToggleButton.title = WebInspector.UIString("Resource tracking disabled. Click to enable.");
+ this.enableToggleButton.toggled = false;
+ this.largerResourcesButton.visible = false;
+ this.sortingSelectElement.addStyleClass("hidden");
+ this.panelEnablerView.visible = true;
+ }
+ },
+
+ addResource: function(resource)
+ {
+ this._resources.push(resource);
+ this.refreshResource(resource);
+ },
+
+ removeResource: function(resource)
+ {
+ if (this.visibleView === resource._resourcesView)
+ this.closeVisibleResource();
+
+ this._resources.remove(resource, true);
+
+ if (resource._resourcesTreeElement) {
+ this.resourcesTreeElement.removeChild(resource._resourcesTreeElement);
+ this.resourcesGraphsElement.removeChild(resource._resourcesTreeElement._resourceGraph.graphElement);
+ }
+
+ resource.warnings = 0;
+ resource.errors = 0;
+
+ delete resource._resourcesTreeElement;
+ delete resource._resourcesView;
+
+ this._adjustScrollPosition();
+ },
+
+ addMessageToResource: function(resource, msg)
+ {
+ if (!resource)
+ return;
+
+ switch (msg.level) {
+ case WebInspector.ConsoleMessage.MessageLevel.Warning:
+ resource.warnings += msg.repeatDelta;
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Error:
+ resource.errors += msg.repeatDelta;
+ break;
+ }
+
+ if (!this.currentQuery && resource._resourcesTreeElement)
+ resource._resourcesTreeElement.updateErrorsAndWarnings();
+
+ var view = this.resourceViewForResource(resource);
+ if (view.addMessage)
+ view.addMessage(msg);
+ },
+
+ clearMessages: function()
+ {
+ var resourcesLength = this._resources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var resource = this._resources[i];
+ resource.warnings = 0;
+ resource.errors = 0;
+
+ if (!this.currentQuery && resource._resourcesTreeElement)
+ resource._resourcesTreeElement.updateErrorsAndWarnings();
+
+ var view = resource._resourcesView;
+ if (!view || !view.clearMessages)
+ continue;
+ view.clearMessages();
+ }
+ },
+
+ refreshResource: function(resource)
+ {
+ this._staleResources.push(resource);
+ this.needsRefresh = true;
+ },
+
+ recreateViewForResourceIfNeeded: function(resource)
+ {
+ if (!resource || !resource._resourcesView)
+ return;
+
+ var newView = this._createResourceView(resource);
+ if (newView.prototype === resource._resourcesView.prototype)
+ return;
+
+ resource.warnings = 0;
+ resource.errors = 0;
+
+ if (!this.currentQuery && resource._resourcesTreeElement)
+ resource._resourcesTreeElement.updateErrorsAndWarnings();
+
+ var oldView = resource._resourcesView;
+
+ resource._resourcesView.detach();
+ delete resource._resourcesView;
+
+ resource._resourcesView = newView;
+
+ newView.headersVisible = oldView.headersVisible;
+
+ if (oldView.visible && oldView.element.parentNode)
+ newView.show(oldView.element.parentNode);
+ },
+
+ showResource: function(resource, line)
+ {
+ if (!resource)
+ return;
+
+ this.containerElement.addStyleClass("viewing-resource");
+
+ if (this.visibleResource && this.visibleResource._resourcesView)
+ this.visibleResource._resourcesView.hide();
+
+ var view = this.resourceViewForResource(resource);
+ view.headersVisible = true;
+ view.show(this.viewsContainerElement);
+
+ if (line) {
+ if (view.revealLine)
+ view.revealLine(line);
+ if (view.highlightLine)
+ view.highlightLine(line);
+ }
+
+ if (resource._resourcesTreeElement) {
+ resource._resourcesTreeElement.reveal();
+ resource._resourcesTreeElement.select(true);
+ }
+
+ this.visibleResource = resource;
+
+ this._updateSidebarWidth();
+ },
+
+ showView: function(view)
+ {
+ if (!view)
+ return;
+ this.showResource(view.resource);
+ },
+
+ closeVisibleResource: function()
+ {
+ this.containerElement.removeStyleClass("viewing-resource");
+ this._updateDividersLabelBarPosition();
+
+ if (this.visibleResource && this.visibleResource._resourcesView)
+ this.visibleResource._resourcesView.hide();
+ delete this.visibleResource;
+
+ if (this._lastSelectedGraphTreeElement)
+ this._lastSelectedGraphTreeElement.select(true);
+
+ this._updateSidebarWidth();
+ },
+
+ resourceViewForResource: function(resource)
+ {
+ if (!resource)
+ return null;
+ if (!resource._resourcesView)
+ resource._resourcesView = this._createResourceView(resource);
+ return resource._resourcesView;
+ },
+
+ sourceFrameForResource: function(resource)
+ {
+ var view = this.resourceViewForResource(resource);
+ if (!view)
+ return null;
+
+ if (!view.setupSourceFrameIfNeeded)
+ return null;
+
+ // Setting up the source frame requires that we be attached.
+ if (!this.element.parentNode)
+ this.attach();
+
+ view.setupSourceFrameIfNeeded();
+ return view.sourceFrame;
+ },
+
+ handleKeyEvent: function(event)
+ {
+ this.sidebarTree.handleKeyEvent(event);
+ },
+
+ _sortResourcesIfNeeded: function()
+ {
+ var sortedElements = [].concat(this.resourcesTreeElement.children);
+ sortedElements.sort(this.sortingFunction);
+
+ var sortedElementsLength = sortedElements.length;
+ for (var i = 0; i < sortedElementsLength; ++i) {
+ var treeElement = sortedElements[i];
+ if (treeElement === this.resourcesTreeElement.children[i])
+ continue;
+
+ var wasSelected = treeElement.selected;
+ this.resourcesTreeElement.removeChild(treeElement);
+ this.resourcesTreeElement.insertChild(treeElement, i);
+ if (wasSelected)
+ treeElement.select(true);
+
+ var graphElement = treeElement._resourceGraph.graphElement;
+ this.resourcesGraphsElement.insertBefore(graphElement, this.resourcesGraphsElement.children[i]);
+ }
+ },
+
+ _updateGraphDividersIfNeeded: function(force)
+ {
+ if (!this.visible) {
+ this.needsRefresh = true;
+ return;
+ }
+
+ if (document.body.offsetWidth <= 0) {
+ // The stylesheet hasn't loaded yet or the window is closed,
+ // so we can't calculate what is need. Return early.
+ return;
+ }
+
+ var dividerCount = Math.round(this.dividersElement.offsetWidth / 64);
+ var slice = this.calculator.boundarySpan / dividerCount;
+ if (!force && this._currentDividerSlice === slice)
+ return;
+
+ this._currentDividerSlice = slice;
+
+ this.dividersElement.removeChildren();
+ this.dividersLabelBarElement.removeChildren();
+
+ for (var i = 1; i <= dividerCount; ++i) {
+ var divider = document.createElement("div");
+ divider.className = "resources-divider";
+ if (i === dividerCount)
+ divider.addStyleClass("last");
+ divider.style.left = ((i / dividerCount) * 100) + "%";
+
+ this.dividersElement.appendChild(divider.cloneNode());
+
+ var label = document.createElement("div");
+ label.className = "resources-divider-label";
+ if (!isNaN(slice))
+ label.textContent = this.calculator.formatValue(slice * i);
+ divider.appendChild(label);
+
+ this.dividersLabelBarElement.appendChild(divider);
+ }
+ },
+
+ _updateSummaryGraph: function()
+ {
+ this.summaryBar.update(this._resources);
+ },
+
+ _updateDividersLabelBarPosition: function()
+ {
+ var scrollTop = this.containerElement.scrollTop;
+ var dividersTop = (scrollTop < this.summaryBar.element.offsetHeight ? this.summaryBar.element.offsetHeight : scrollTop);
+ this.dividersElement.style.top = scrollTop + "px";
+ this.dividersLabelBarElement.style.top = dividersTop + "px";
+ },
+
+ _graphSelected: function(treeElement)
+ {
+ if (this._lastSelectedGraphTreeElement)
+ this._lastSelectedGraphTreeElement.selectedSortingOptionIndex = this.sortingSelectElement.selectedIndex;
+
+ this._lastSelectedGraphTreeElement = treeElement;
+
+ this.sortingSelectElement.removeChildren();
+ for (var i = 0; i < treeElement.sortingOptions.length; ++i) {
+ var sortingOption = treeElement.sortingOptions[i];
+ var option = document.createElement("option");
+ option.label = sortingOption.name;
+ option.sortingFunction = sortingOption.sortingFunction;
+ option.calculator = sortingOption.calculator;
+ this.sortingSelectElement.appendChild(option);
+ }
+
+ this.sortingSelectElement.selectedIndex = treeElement.selectedSortingOptionIndex;
+ this._changeSortingFunction();
+
+ this.closeVisibleResource();
+ this.containerElement.scrollTop = 0;
+ },
+
+ _toggleLargerResources: function()
+ {
+ if (!this.resourcesTreeElement._childrenListNode)
+ return;
+
+ this.resourcesTreeElement.smallChildren = !this.resourcesTreeElement.smallChildren;
+
+ if (this.resourcesTreeElement.smallChildren) {
+ this.resourcesGraphsElement.addStyleClass("small");
+ this.largerResourcesButton.title = WebInspector.UIString("Use large resource rows.");
+ this.largerResourcesButton.toggled = false;
+ this._adjustScrollPosition();
+ } else {
+ this.resourcesGraphsElement.removeStyleClass("small");
+ this.largerResourcesButton.title = WebInspector.UIString("Use small resource rows.");
+ this.largerResourcesButton.toggled = true;
+ }
+ },
+
+ _adjustScrollPosition: function()
+ {
+ // Prevent the container from being scrolled off the end.
+ if ((this.containerElement.scrollTop + this.containerElement.offsetHeight) > this.sidebarElement.offsetHeight)
+ this.containerElement.scrollTop = (this.sidebarElement.offsetHeight - this.containerElement.offsetHeight);
+ },
+
+ _changeSortingFunction: function()
+ {
+ var selectedOption = this.sortingSelectElement[this.sortingSelectElement.selectedIndex];
+ this.sortingFunction = selectedOption.sortingFunction;
+ this.calculator = this.summaryBar.calculator = selectedOption.calculator;
+ },
+
+ _createResourceView: function(resource)
+ {
+ switch (resource.category) {
+ case WebInspector.resourceCategories.documents:
+ case WebInspector.resourceCategories.stylesheets:
+ case WebInspector.resourceCategories.scripts:
+ case WebInspector.resourceCategories.xhr:
+ return new WebInspector.SourceView(resource);
+ case WebInspector.resourceCategories.images:
+ return new WebInspector.ImageView(resource);
+ case WebInspector.resourceCategories.fonts:
+ return new WebInspector.FontView(resource);
+ default:
+ return new WebInspector.ResourceView(resource);
+ }
+ },
+
+ _startSidebarDragging: function(event)
+ {
+ WebInspector.elementDragStart(this.sidebarResizeElement, this._sidebarDragging.bind(this), this._endSidebarDragging.bind(this), event, "col-resize");
+ },
+
+ _sidebarDragging: function(event)
+ {
+ this._updateSidebarWidth(event.pageX);
+
+ event.preventDefault();
+ },
+
+ _endSidebarDragging: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+ },
+
+ _updateSidebarWidth: function(width)
+ {
+ if (this.sidebarElement.offsetWidth <= 0) {
+ // The stylesheet hasn't loaded yet or the window is closed,
+ // so we can't calculate what is need. Return early.
+ return;
+ }
+
+ if (!("_currentSidebarWidth" in this))
+ this._currentSidebarWidth = this.sidebarElement.offsetWidth;
+
+ if (typeof width === "undefined")
+ width = this._currentSidebarWidth;
+
+ width = Number.constrain(width, Preferences.minSidebarWidth, window.innerWidth / 2);
+
+ this._currentSidebarWidth = width;
+
+ if (this.visibleResource) {
+ this.containerElement.style.width = width + "px";
+ this.sidebarElement.style.removeProperty("width");
+ } else {
+ this.sidebarElement.style.width = width + "px";
+ this.containerElement.style.removeProperty("width");
+ }
+
+ this.containerContentElement.style.left = width + "px";
+ this.viewsContainerElement.style.left = width + "px";
+ this.sidebarResizeElement.style.left = (width - 3) + "px";
+
+ this._updateGraphDividersIfNeeded();
+
+ var visibleView = this.visibleView;
+ if (visibleView && "resize" in visibleView)
+ visibleView.resize();
+ },
+
+ _enableResourceTracking: function()
+ {
+ if (InspectorController.resourceTrackingEnabled())
+ return;
+ this._toggleResourceTracking(this.panelEnablerView.alwaysEnabled);
+ },
+
+ _toggleResourceTracking: function(optionalAlways)
+ {
+ if (InspectorController.resourceTrackingEnabled()) {
+ this.largerResourcesButton.visible = false;
+ this.sortingSelectElement.visible = false;
+ InspectorController.disableResourceTracking(true);
+ } else {
+ this.largerResourcesButton.visible = true;
+ this.sortingSelectElement.visible = true;
+ InspectorController.enableResourceTracking(!!optionalAlways);
+ }
+ }
+}
+
+WebInspector.ResourcesPanel.prototype.__proto__ = WebInspector.Panel.prototype;
+
+WebInspector.ResourceCalculator = function()
+{
+}
+
+WebInspector.ResourceCalculator.prototype = {
+ computeSummaryValues: function(resources)
+ {
+ var total = 0;
+ var categoryValues = {};
+
+ var resourcesLength = resources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var resource = resources[i];
+ var value = this._value(resource);
+ if (typeof value === "undefined")
+ continue;
+ if (!(resource.category.name in categoryValues))
+ categoryValues[resource.category.name] = 0;
+ categoryValues[resource.category.name] += value;
+ total += value;
+ }
+
+ return {categoryValues: categoryValues, total: total};
+ },
+
+ computeBarGraphPercentages: function(resource)
+ {
+ return {start: 0, middle: 0, end: (this._value(resource) / this.boundarySpan) * 100};
+ },
+
+ computeBarGraphLabels: function(resource)
+ {
+ const label = this.formatValue(this._value(resource));
+ var tooltip = label;
+ if (resource.cached)
+ tooltip = WebInspector.UIString("%s (from cache)", tooltip);
+ return {left: label, right: label, tooltip: tooltip};
+ },
+
+ get boundarySpan()
+ {
+ return this.maximumBoundary - this.minimumBoundary;
+ },
+
+ updateBoundaries: function(resource)
+ {
+ this.minimumBoundary = 0;
+
+ var value = this._value(resource);
+ if (typeof this.maximumBoundary === "undefined" || value > this.maximumBoundary) {
+ this.maximumBoundary = value;
+ return true;
+ }
+
+ return false;
+ },
+
+ reset: function()
+ {
+ delete this.minimumBoundary;
+ delete this.maximumBoundary;
+ },
+
+ _value: function(resource)
+ {
+ return 0;
+ },
+
+ formatValue: function(value)
+ {
+ return value.toString();
+ }
+}
+
+WebInspector.ResourceTimeCalculator = function(startAtZero)
+{
+ WebInspector.ResourceCalculator.call(this);
+ this.startAtZero = startAtZero;
+}
+
+WebInspector.ResourceTimeCalculator.prototype = {
+ computeSummaryValues: function(resources)
+ {
+ var resourcesByCategory = {};
+ var resourcesLength = resources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var resource = resources[i];
+ if (!(resource.category.name in resourcesByCategory))
+ resourcesByCategory[resource.category.name] = [];
+ resourcesByCategory[resource.category.name].push(resource);
+ }
+
+ var earliestStart;
+ var latestEnd;
+ var categoryValues = {};
+ for (var category in resourcesByCategory) {
+ resourcesByCategory[category].sort(WebInspector.Resource.CompareByTime);
+ categoryValues[category] = 0;
+
+ var segment = {start: -1, end: -1};
+
+ var categoryResources = resourcesByCategory[category];
+ var resourcesLength = categoryResources.length;
+ for (var i = 0; i < resourcesLength; ++i) {
+ var resource = categoryResources[i];
+ if (resource.startTime === -1 || resource.endTime === -1)
+ continue;
+
+ if (typeof earliestStart === "undefined")
+ earliestStart = resource.startTime;
+ else
+ earliestStart = Math.min(earliestStart, resource.startTime);
+
+ if (typeof latestEnd === "undefined")
+ latestEnd = resource.endTime;
+ else
+ latestEnd = Math.max(latestEnd, resource.endTime);
+
+ if (resource.startTime <= segment.end) {
+ segment.end = Math.max(segment.end, resource.endTime);
+ continue;
+ }
+
+ categoryValues[category] += segment.end - segment.start;
+
+ segment.start = resource.startTime;
+ segment.end = resource.endTime;
+ }
+
+ // Add the last segment
+ categoryValues[category] += segment.end - segment.start;
+ }
+
+ return {categoryValues: categoryValues, total: latestEnd - earliestStart};
+ },
+
+ computeBarGraphPercentages: function(resource)
+ {
+ if (resource.startTime !== -1)
+ var start = ((resource.startTime - this.minimumBoundary) / this.boundarySpan) * 100;
+ else
+ var start = 0;
+
+ if (resource.responseReceivedTime !== -1)
+ var middle = ((resource.responseReceivedTime - this.minimumBoundary) / this.boundarySpan) * 100;
+ else
+ var middle = (this.startAtZero ? start : 100);
+
+ if (resource.endTime !== -1)
+ var end = ((resource.endTime - this.minimumBoundary) / this.boundarySpan) * 100;
+ else
+ var end = (this.startAtZero ? middle : 100);
+
+ if (this.startAtZero) {
+ end -= start;
+ middle -= start;
+ start = 0;
+ }
+
+ return {start: start, middle: middle, end: end};
+ },
+
+ computeBarGraphLabels: function(resource)
+ {
+ var leftLabel = "";
+ if (resource.latency > 0)
+ leftLabel = this.formatValue(resource.latency);
+
+ var rightLabel = "";
+ if (resource.responseReceivedTime !== -1 && resource.endTime !== -1)
+ rightLabel = this.formatValue(resource.endTime - resource.responseReceivedTime);
+
+ if (leftLabel && rightLabel) {
+ var total = this.formatValue(resource.duration);
+ var tooltip = WebInspector.UIString("%s latency, %s download (%s total)", leftLabel, rightLabel, total);
+ } else if (leftLabel)
+ var tooltip = WebInspector.UIString("%s latency", leftLabel);
+ else if (rightLabel)
+ var tooltip = WebInspector.UIString("%s download", rightLabel);
+
+ if (resource.cached)
+ tooltip = WebInspector.UIString("%s (from cache)", tooltip);
+
+ return {left: leftLabel, right: rightLabel, tooltip: tooltip};
+ },
+
+ updateBoundaries: function(resource)
+ {
+ var didChange = false;
+
+ var lowerBound;
+ if (this.startAtZero)
+ lowerBound = 0;
+ else
+ lowerBound = this._lowerBound(resource);
+
+ if (lowerBound !== -1 && (typeof this.minimumBoundary === "undefined" || lowerBound < this.minimumBoundary)) {
+ this.minimumBoundary = lowerBound;
+ didChange = true;
+ }
+
+ var upperBound = this._upperBound(resource);
+ if (upperBound !== -1 && (typeof this.maximumBoundary === "undefined" || upperBound > this.maximumBoundary)) {
+ this.maximumBoundary = upperBound;
+ didChange = true;
+ }
+
+ return didChange;
+ },
+
+ formatValue: function(value)
+ {
+ return Number.secondsToString(value, WebInspector.UIString.bind(WebInspector));
+ },
+
+ _lowerBound: function(resource)
+ {
+ return 0;
+ },
+
+ _upperBound: function(resource)
+ {
+ return 0;
+ },
+}
+
+WebInspector.ResourceTimeCalculator.prototype.__proto__ = WebInspector.ResourceCalculator.prototype;
+
+WebInspector.ResourceTransferTimeCalculator = function()
+{
+ WebInspector.ResourceTimeCalculator.call(this, false);
+}
+
+WebInspector.ResourceTransferTimeCalculator.prototype = {
+ formatValue: function(value)
+ {
+ return Number.secondsToString(value, WebInspector.UIString.bind(WebInspector));
+ },
+
+ _lowerBound: function(resource)
+ {
+ return resource.startTime;
+ },
+
+ _upperBound: function(resource)
+ {
+ return resource.endTime;
+ }
+}
+
+WebInspector.ResourceTransferTimeCalculator.prototype.__proto__ = WebInspector.ResourceTimeCalculator.prototype;
+
+WebInspector.ResourceTransferDurationCalculator = function()
+{
+ WebInspector.ResourceTimeCalculator.call(this, true);
+}
+
+WebInspector.ResourceTransferDurationCalculator.prototype = {
+ formatValue: function(value)
+ {
+ return Number.secondsToString(value, WebInspector.UIString.bind(WebInspector));
+ },
+
+ _upperBound: function(resource)
+ {
+ return resource.duration;
+ }
+}
+
+WebInspector.ResourceTransferDurationCalculator.prototype.__proto__ = WebInspector.ResourceTimeCalculator.prototype;
+
+WebInspector.ResourceTransferSizeCalculator = function()
+{
+ WebInspector.ResourceCalculator.call(this);
+}
+
+WebInspector.ResourceTransferSizeCalculator.prototype = {
+ _value: function(resource)
+ {
+ return resource.contentLength;
+ },
+
+ formatValue: function(value)
+ {
+ return Number.bytesToString(value, WebInspector.UIString.bind(WebInspector));
+ }
+}
+
+WebInspector.ResourceTransferSizeCalculator.prototype.__proto__ = WebInspector.ResourceCalculator.prototype;
+
+WebInspector.ResourceSidebarTreeElement = function(resource)
+{
+ this.resource = resource;
+
+ this.createIconElement();
+
+ WebInspector.SidebarTreeElement.call(this, "resource-sidebar-tree-item", "", "", resource);
+
+ this.refreshTitles();
+}
+
+WebInspector.ResourceSidebarTreeElement.prototype = {
+ onattach: function()
+ {
+ WebInspector.SidebarTreeElement.prototype.onattach.call(this);
+
+ var link = document.createElement("a");
+ link.href = this.resource.url;
+ link.className = "invisible";
+ while (this._listItemNode.firstChild)
+ link.appendChild(this._listItemNode.firstChild);
+ this._listItemNode.appendChild(link);
+ this._listItemNode.addStyleClass("resources-category-" + this.resource.category.name);
+ },
+
+ onselect: function()
+ {
+ WebInspector.panels.resources.showResource(this.resource);
+ },
+
+ ondblclick: function(treeElement, event)
+ {
+ InjectedScriptAccess.openInInspectedWindow(this.resource.url);
+ },
+
+ get mainTitle()
+ {
+ return this.resource.displayName;
+ },
+
+ set mainTitle(x)
+ {
+ // Do nothing.
+ },
+
+ get subtitle()
+ {
+ var subtitle = this.resource.displayDomain;
+
+ if (this.resource.path && this.resource.lastPathComponent) {
+ var lastPathComponentIndex = this.resource.path.lastIndexOf("/" + this.resource.lastPathComponent);
+ if (lastPathComponentIndex != -1)
+ subtitle += this.resource.path.substring(0, lastPathComponentIndex);
+ }
+
+ return subtitle;
+ },
+
+ set subtitle(x)
+ {
+ // Do nothing.
+ },
+
+ get selectable()
+ {
+ return WebInspector.panels.resources._filterCategory == "all" || WebInspector.panels.resources._filterCategory == this.resource.category.name;
+ },
+
+ createIconElement: function()
+ {
+ var previousIconElement = this.iconElement;
+
+ if (this.resource.category === WebInspector.resourceCategories.images) {
+ var previewImage = document.createElement("img");
+ previewImage.className = "image-resource-icon-preview";
+ previewImage.src = this.resource.url;
+
+ this.iconElement = document.createElement("div");
+ this.iconElement.className = "icon";
+ this.iconElement.appendChild(previewImage);
+ } else {
+ this.iconElement = document.createElement("img");
+ this.iconElement.className = "icon";
+ }
+
+ if (previousIconElement)
+ previousIconElement.parentNode.replaceChild(this.iconElement, previousIconElement);
+ },
+
+ refresh: function()
+ {
+ this.refreshTitles();
+
+ if (!this._listItemNode.hasStyleClass("resources-category-" + this.resource.category.name)) {
+ this._listItemNode.removeMatchingStyleClasses("resources-category-\\w+");
+ this._listItemNode.addStyleClass("resources-category-" + this.resource.category.name);
+
+ this.createIconElement();
+ }
+ },
+
+ resetBubble: function()
+ {
+ this.bubbleText = "";
+ this.bubbleElement.removeStyleClass("search-matches");
+ this.bubbleElement.removeStyleClass("warning");
+ this.bubbleElement.removeStyleClass("error");
+ },
+
+ set searchMatches(matches)
+ {
+ this.resetBubble();
+
+ if (!matches)
+ return;
+
+ this.bubbleText = matches;
+ this.bubbleElement.addStyleClass("search-matches");
+ },
+
+ updateErrorsAndWarnings: function()
+ {
+ this.resetBubble();
+
+ if (this.resource.warnings || this.resource.errors)
+ this.bubbleText = (this.resource.warnings + this.resource.errors);
+
+ if (this.resource.warnings)
+ this.bubbleElement.addStyleClass("warning");
+
+ if (this.resource.errors)
+ this.bubbleElement.addStyleClass("error");
+ }
+}
+
+WebInspector.ResourceSidebarTreeElement.CompareByAscendingStartTime = function(a, b)
+{
+ return WebInspector.Resource.CompareByStartTime(a.resource, b.resource)
+ || WebInspector.Resource.CompareByEndTime(a.resource, b.resource)
+ || WebInspector.Resource.CompareByResponseReceivedTime(a.resource, b.resource);
+}
+
+WebInspector.ResourceSidebarTreeElement.CompareByAscendingResponseReceivedTime = function(a, b)
+{
+ return WebInspector.Resource.CompareByResponseReceivedTime(a.resource, b.resource)
+ || WebInspector.Resource.CompareByStartTime(a.resource, b.resource)
+ || WebInspector.Resource.CompareByEndTime(a.resource, b.resource);
+}
+
+WebInspector.ResourceSidebarTreeElement.CompareByAscendingEndTime = function(a, b)
+{
+ return WebInspector.Resource.CompareByEndTime(a.resource, b.resource)
+ || WebInspector.Resource.CompareByStartTime(a.resource, b.resource)
+ || WebInspector.Resource.CompareByResponseReceivedTime(a.resource, b.resource);
+}
+
+WebInspector.ResourceSidebarTreeElement.CompareByDescendingDuration = function(a, b)
+{
+ return -1 * WebInspector.Resource.CompareByDuration(a.resource, b.resource);
+}
+
+WebInspector.ResourceSidebarTreeElement.CompareByDescendingLatency = function(a, b)
+{
+ return -1 * WebInspector.Resource.CompareByLatency(a.resource, b.resource);
+}
+
+WebInspector.ResourceSidebarTreeElement.CompareByDescendingSize = function(a, b)
+{
+ return -1 * WebInspector.Resource.CompareBySize(a.resource, b.resource);
+}
+
+WebInspector.ResourceSidebarTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
+
+WebInspector.ResourceGraph = function(resource)
+{
+ this.resource = resource;
+
+ this._graphElement = document.createElement("div");
+ this._graphElement.className = "resources-graph-side";
+ this._graphElement.addEventListener("mouseover", this.refreshLabelPositions.bind(this), false);
+
+ if (resource.cached)
+ this._graphElement.addStyleClass("resource-cached");
+
+ this._barAreaElement = document.createElement("div");
+ this._barAreaElement.className = "resources-graph-bar-area hidden";
+ this._graphElement.appendChild(this._barAreaElement);
+
+ this._barLeftElement = document.createElement("div");
+ this._barLeftElement.className = "resources-graph-bar waiting";
+ this._barAreaElement.appendChild(this._barLeftElement);
+
+ this._barRightElement = document.createElement("div");
+ this._barRightElement.className = "resources-graph-bar";
+ this._barAreaElement.appendChild(this._barRightElement);
+
+ this._labelLeftElement = document.createElement("div");
+ this._labelLeftElement.className = "resources-graph-label waiting";
+ this._barAreaElement.appendChild(this._labelLeftElement);
+
+ this._labelRightElement = document.createElement("div");
+ this._labelRightElement.className = "resources-graph-label";
+ this._barAreaElement.appendChild(this._labelRightElement);
+
+ this._graphElement.addStyleClass("resources-category-" + resource.category.name);
+}
+
+WebInspector.ResourceGraph.prototype = {
+ get graphElement()
+ {
+ return this._graphElement;
+ },
+
+ refreshLabelPositions: function()
+ {
+ this._labelLeftElement.style.removeProperty("left");
+ this._labelLeftElement.style.removeProperty("right");
+ this._labelLeftElement.removeStyleClass("before");
+ this._labelLeftElement.removeStyleClass("hidden");
+
+ this._labelRightElement.style.removeProperty("left");
+ this._labelRightElement.style.removeProperty("right");
+ this._labelRightElement.removeStyleClass("after");
+ this._labelRightElement.removeStyleClass("hidden");
+
+ const labelPadding = 10;
+ const rightBarWidth = (this._barRightElement.offsetWidth - labelPadding);
+ const leftBarWidth = ((this._barLeftElement.offsetWidth - this._barRightElement.offsetWidth) - labelPadding);
+
+ var labelBefore = (this._labelLeftElement.offsetWidth > leftBarWidth);
+ var labelAfter = (this._labelRightElement.offsetWidth > rightBarWidth);
+
+ if (labelBefore) {
+ if ((this._graphElement.offsetWidth * (this._percentages.start / 100)) < (this._labelLeftElement.offsetWidth + 10))
+ this._labelLeftElement.addStyleClass("hidden");
+ this._labelLeftElement.style.setProperty("right", (100 - this._percentages.start) + "%");
+ this._labelLeftElement.addStyleClass("before");
+ } else {
+ this._labelLeftElement.style.setProperty("left", this._percentages.start + "%");
+ this._labelLeftElement.style.setProperty("right", (100 - this._percentages.middle) + "%");
+ }
+
+ if (labelAfter) {
+ if ((this._graphElement.offsetWidth * ((100 - this._percentages.end) / 100)) < (this._labelRightElement.offsetWidth + 10))
+ this._labelRightElement.addStyleClass("hidden");
+ this._labelRightElement.style.setProperty("left", this._percentages.end + "%");
+ this._labelRightElement.addStyleClass("after");
+ } else {
+ this._labelRightElement.style.setProperty("left", this._percentages.middle + "%");
+ this._labelRightElement.style.setProperty("right", (100 - this._percentages.end) + "%");
+ }
+ },
+
+ refresh: function(calculator)
+ {
+ var percentages = calculator.computeBarGraphPercentages(this.resource);
+ var labels = calculator.computeBarGraphLabels(this.resource);
+
+ this._percentages = percentages;
+
+ this._barAreaElement.removeStyleClass("hidden");
+
+ if (!this._graphElement.hasStyleClass("resources-category-" + this.resource.category.name)) {
+ this._graphElement.removeMatchingStyleClasses("resources-category-\\w+");
+ this._graphElement.addStyleClass("resources-category-" + this.resource.category.name);
+ }
+
+ this._barLeftElement.style.setProperty("left", percentages.start + "%");
+ this._barLeftElement.style.setProperty("right", (100 - percentages.end) + "%");
+
+ this._barRightElement.style.setProperty("left", percentages.middle + "%");
+ this._barRightElement.style.setProperty("right", (100 - percentages.end) + "%");
+
+ this._labelLeftElement.textContent = labels.left;
+ this._labelRightElement.textContent = labels.right;
+
+ var tooltip = (labels.tooltip || "");
+ this._barLeftElement.title = tooltip;
+ this._labelLeftElement.title = tooltip;
+ this._labelRightElement.title = tooltip;
+ this._barRightElement.title = tooltip;
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScopeChainSidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScopeChainSidebarPane.js
new file mode 100644
index 0000000..3875324
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScopeChainSidebarPane.js
@@ -0,0 +1,146 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ScopeChainSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Scope Variables"));
+ this._expandedProperties = [];
+}
+
+WebInspector.ScopeChainSidebarPane.prototype = {
+ update: function(callFrame)
+ {
+ this.bodyElement.removeChildren();
+
+ this.sections = [];
+ this.callFrame = callFrame;
+
+ if (!callFrame) {
+ var infoElement = document.createElement("div");
+ infoElement.className = "info";
+ infoElement.textContent = WebInspector.UIString("Not Paused");
+ this.bodyElement.appendChild(infoElement);
+ return;
+ }
+
+ var foundLocalScope = false;
+ var scopeChain = callFrame.scopeChain;
+ for (var i = 0; i < scopeChain.length; ++i) {
+ var scopeObjectProxy = scopeChain[i];
+ var title = null;
+ var subtitle = scopeObjectProxy.description;
+ var emptyPlaceholder = null;
+ var extraProperties = null;
+
+ if (scopeObjectProxy.isLocal) {
+ if (scopeObjectProxy.thisObject) {
+ extraProperties = [ new WebInspector.ObjectPropertyProxy("this", scopeObjectProxy.thisObject) ];
+ title = WebInspector.UIString("Local");
+ } else
+ title = WebInspector.UIString("Closure");
+ emptyPlaceholder = WebInspector.UIString("No Variables");
+ subtitle = null;
+ foundLocalScope = true;
+ } else if (i === (scopeChain.length - 1))
+ title = WebInspector.UIString("Global");
+ else if (scopeObjectProxy.isElement)
+ title = WebInspector.UIString("Event Target");
+ else if (scopeObjectProxy.isDocument)
+ title = WebInspector.UIString("Event Document");
+ else if (scopeObjectProxy.isWithBlock)
+ title = WebInspector.UIString("With Block");
+
+ if (!title || title === subtitle)
+ subtitle = null;
+
+ var section = new WebInspector.ObjectPropertiesSection(scopeObjectProxy, title, subtitle, emptyPlaceholder, true, extraProperties, WebInspector.ScopeVariableTreeElement);
+ section.editInSelectedCallFrameWhenPaused = true;
+ section.pane = this;
+
+ if (!foundLocalScope || scopeObjectProxy.isLocal)
+ section.expanded = true;
+
+ this.sections.push(section);
+ this.bodyElement.appendChild(section.element);
+ }
+ }
+}
+
+WebInspector.ScopeChainSidebarPane.prototype.__proto__ = WebInspector.SidebarPane.prototype;
+
+WebInspector.ScopeVariableTreeElement = function(property)
+{
+ WebInspector.ObjectPropertyTreeElement.call(this, property);
+}
+
+WebInspector.ScopeVariableTreeElement.prototype = {
+ onattach: function()
+ {
+ WebInspector.ObjectPropertyTreeElement.prototype.onattach.call(this);
+ if (this.hasChildren && this.propertyIdentifier in this.treeOutline.section.pane._expandedProperties)
+ this.expand();
+ },
+
+ onexpand: function()
+ {
+ this.treeOutline.section.pane._expandedProperties[this.propertyIdentifier] = true;
+ },
+
+ oncollapse: function()
+ {
+ delete this.treeOutline.section.pane._expandedProperties[this.propertyIdentifier];
+ },
+
+ get propertyIdentifier()
+ {
+ if ("_propertyIdentifier" in this)
+ return this._propertyIdentifier;
+ var section = this.treeOutline.section;
+ this._propertyIdentifier = section.title + ":" + (section.subtitle ? section.subtitle + ":" : "") + this.propertyPath;
+ return this._propertyIdentifier;
+ },
+
+ get propertyPath()
+ {
+ if ("_propertyPath" in this)
+ return this._propertyPath;
+
+ var current = this;
+ var result;
+
+ do {
+ if (result)
+ result = current.property.name + "." + result;
+ else
+ result = current.property.name;
+ current = current.parent;
+ } while (current && !current.root);
+
+ this._propertyPath = result;
+ return result;
+ }
+}
+
+WebInspector.ScopeVariableTreeElement.prototype.__proto__ = WebInspector.ObjectPropertyTreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Script.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Script.js
new file mode 100644
index 0000000..e6413a9
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/Script.js
@@ -0,0 +1,50 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.Script = function(sourceID, sourceURL, source, startingLine, errorLine, errorMessage)
+{
+ this.sourceID = sourceID;
+ this.sourceURL = sourceURL;
+ this.source = source;
+ this.startingLine = startingLine;
+ this.errorLine = errorLine;
+ this.errorMessage = errorMessage;
+
+ // if no URL, look for "//@ sourceURL=" decorator
+ // note that this sourceURL comment decorator is behavior that FireBug added
+ // in it's 1.1 release as noted in the release notes:
+ // http://fbug.googlecode.com/svn/branches/firebug1.1/docs/ReleaseNotes_1.1.txt
+ if (!sourceURL) {
+ // use of [ \t] rather than \s is to prevent \n from matching
+ var pattern = /^\s*\/\/[ \t]*@[ \t]*sourceURL[ \t]*=[ \t]*(\S+).*$/m;
+ var match = pattern.exec(source);
+
+ if (match)
+ this.sourceURL = WebInspector.UIString("(program): %s", match[1]);
+ }
+}
+
+WebInspector.Script.prototype = {
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptView.js
new file mode 100644
index 0000000..124190c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptView.js
@@ -0,0 +1,103 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ScriptView = function(script)
+{
+ WebInspector.View.call(this);
+
+ this.element.addStyleClass("script-view");
+
+ this.script = script;
+
+ this._frameNeedsSetup = true;
+ this._sourceFrameSetup = false;
+
+ this.sourceFrame = new WebInspector.SourceFrame(null, this._addBreakpoint.bind(this));
+
+ this.element.appendChild(this.sourceFrame.element);
+}
+
+WebInspector.ScriptView.prototype = {
+ show: function(parentElement)
+ {
+ WebInspector.View.prototype.show.call(this, parentElement);
+ this.setupSourceFrameIfNeeded();
+ },
+
+ hide: function()
+ {
+ WebInspector.View.prototype.hide.call(this);
+ this._currentSearchResultIndex = -1;
+ },
+
+ setupSourceFrameIfNeeded: function()
+ {
+ if (!this._frameNeedsSetup)
+ return;
+
+ this.attach();
+
+ if (!InspectorController.addSourceToFrame("text/javascript", this.script.source, this.sourceFrame.element))
+ return;
+
+ delete this._frameNeedsSetup;
+
+ this.sourceFrame.addEventListener("syntax highlighting complete", this._syntaxHighlightingComplete, this);
+ this.sourceFrame.syntaxHighlightJavascript();
+ },
+
+ attach: function()
+ {
+ if (!this.element.parentNode)
+ document.getElementById("script-resource-views").appendChild(this.element);
+ },
+
+ _addBreakpoint: function(line)
+ {
+ var breakpoint = new WebInspector.Breakpoint(this.script.sourceURL, line, this.script.sourceID);
+ WebInspector.panels.scripts.addBreakpoint(breakpoint);
+ },
+
+ // The follow methods are pulled from SourceView, since they are
+ // generic and work with ScriptView just fine.
+
+ revealLine: WebInspector.SourceView.prototype.revealLine,
+ highlightLine: WebInspector.SourceView.prototype.highlightLine,
+ addMessage: WebInspector.SourceView.prototype.addMessage,
+ clearMessages: WebInspector.SourceView.prototype.clearMessages,
+ searchCanceled: WebInspector.SourceView.prototype.searchCanceled,
+ performSearch: WebInspector.SourceView.prototype.performSearch,
+ jumpToFirstSearchResult: WebInspector.SourceView.prototype.jumpToFirstSearchResult,
+ jumpToLastSearchResult: WebInspector.SourceView.prototype.jumpToLastSearchResult,
+ jumpToNextSearchResult: WebInspector.SourceView.prototype.jumpToNextSearchResult,
+ jumpToPreviousSearchResult: WebInspector.SourceView.prototype.jumpToPreviousSearchResult,
+ showingFirstSearchResult: WebInspector.SourceView.prototype.showingFirstSearchResult,
+ showingLastSearchResult: WebInspector.SourceView.prototype.showingLastSearchResult,
+ _jumpToSearchResult: WebInspector.SourceView.prototype._jumpToSearchResult,
+ _sourceFrameSetupFinished: WebInspector.SourceView.prototype._sourceFrameSetupFinished,
+ _syntaxHighlightingComplete: WebInspector.SourceView.prototype._syntaxHighlightingComplete
+}
+
+WebInspector.ScriptView.prototype.__proto__ = WebInspector.View.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptsPanel.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptsPanel.js
new file mode 100644
index 0000000..04f27bb
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/ScriptsPanel.js
@@ -0,0 +1,918 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.ScriptsPanel = function()
+{
+ WebInspector.Panel.call(this);
+
+ this.element.addStyleClass("scripts");
+
+ this.topStatusBar = document.createElement("div");
+ this.topStatusBar.className = "status-bar";
+ this.topStatusBar.id = "scripts-status-bar";
+ this.element.appendChild(this.topStatusBar);
+
+ this.backButton = document.createElement("button");
+ this.backButton.className = "status-bar-item";
+ this.backButton.id = "scripts-back";
+ this.backButton.title = WebInspector.UIString("Show the previous script resource.");
+ this.backButton.disabled = true;
+ this.backButton.appendChild(document.createElement("img"));
+ this.backButton.addEventListener("click", this._goBack.bind(this), false);
+ this.topStatusBar.appendChild(this.backButton);
+
+ this.forwardButton = document.createElement("button");
+ this.forwardButton.className = "status-bar-item";
+ this.forwardButton.id = "scripts-forward";
+ this.forwardButton.title = WebInspector.UIString("Show the next script resource.");
+ this.forwardButton.disabled = true;
+ this.forwardButton.appendChild(document.createElement("img"));
+ this.forwardButton.addEventListener("click", this._goForward.bind(this), false);
+ this.topStatusBar.appendChild(this.forwardButton);
+
+ this.filesSelectElement = document.createElement("select");
+ this.filesSelectElement.className = "status-bar-item";
+ this.filesSelectElement.id = "scripts-files";
+ this.filesSelectElement.addEventListener("change", this._changeVisibleFile.bind(this), false);
+ this.filesSelectElement.handleKeyEvent = this.handleKeyEvent.bind(this);
+ this.topStatusBar.appendChild(this.filesSelectElement);
+
+ this.functionsSelectElement = document.createElement("select");
+ this.functionsSelectElement.className = "status-bar-item";
+ this.functionsSelectElement.id = "scripts-functions";
+
+ // FIXME: append the functions select element to the top status bar when it is implemented.
+ // this.topStatusBar.appendChild(this.functionsSelectElement);
+
+ this.sidebarButtonsElement = document.createElement("div");
+ this.sidebarButtonsElement.id = "scripts-sidebar-buttons";
+ this.topStatusBar.appendChild(this.sidebarButtonsElement);
+
+ this.pauseButton = document.createElement("button");
+ this.pauseButton.className = "status-bar-item";
+ this.pauseButton.id = "scripts-pause";
+ this.pauseButton.title = WebInspector.UIString("Pause script execution.");
+ this.pauseButton.disabled = true;
+ this.pauseButton.appendChild(document.createElement("img"));
+ this.pauseButton.addEventListener("click", this._togglePause.bind(this), false);
+ this.sidebarButtonsElement.appendChild(this.pauseButton);
+
+ this.stepOverButton = document.createElement("button");
+ this.stepOverButton.className = "status-bar-item";
+ this.stepOverButton.id = "scripts-step-over";
+ this.stepOverButton.title = WebInspector.UIString("Step over next function call.");
+ this.stepOverButton.disabled = true;
+ this.stepOverButton.addEventListener("click", this._stepOverClicked.bind(this), false);
+ this.stepOverButton.appendChild(document.createElement("img"));
+ this.sidebarButtonsElement.appendChild(this.stepOverButton);
+
+ this.stepIntoButton = document.createElement("button");
+ this.stepIntoButton.className = "status-bar-item";
+ this.stepIntoButton.id = "scripts-step-into";
+ this.stepIntoButton.title = WebInspector.UIString("Step into next function call.");
+ this.stepIntoButton.disabled = true;
+ this.stepIntoButton.addEventListener("click", this._stepIntoClicked.bind(this), false);
+ this.stepIntoButton.appendChild(document.createElement("img"));
+ this.sidebarButtonsElement.appendChild(this.stepIntoButton);
+
+ this.stepOutButton = document.createElement("button");
+ this.stepOutButton.className = "status-bar-item";
+ this.stepOutButton.id = "scripts-step-out";
+ this.stepOutButton.title = WebInspector.UIString("Step out of current function.");
+ this.stepOutButton.disabled = true;
+ this.stepOutButton.addEventListener("click", this._stepOutClicked.bind(this), false);
+ this.stepOutButton.appendChild(document.createElement("img"));
+ this.sidebarButtonsElement.appendChild(this.stepOutButton);
+
+ this.debuggerStatusElement = document.createElement("div");
+ this.debuggerStatusElement.id = "scripts-debugger-status";
+ this.sidebarButtonsElement.appendChild(this.debuggerStatusElement);
+
+ this.viewsContainerElement = document.createElement("div");
+ this.viewsContainerElement.id = "script-resource-views";
+
+ this.sidebarElement = document.createElement("div");
+ this.sidebarElement.id = "scripts-sidebar";
+
+ this.sidebarResizeElement = document.createElement("div");
+ this.sidebarResizeElement.className = "sidebar-resizer-vertical";
+ this.sidebarResizeElement.addEventListener("mousedown", this._startSidebarResizeDrag.bind(this), false);
+
+ this.sidebarResizeWidgetElement = document.createElement("div");
+ this.sidebarResizeWidgetElement.id = "scripts-sidebar-resizer-widget";
+ this.sidebarResizeWidgetElement.addEventListener("mousedown", this._startSidebarResizeDrag.bind(this), false);
+ this.topStatusBar.appendChild(this.sidebarResizeWidgetElement);
+
+ this.sidebarPanes = {};
+ this.sidebarPanes.watchExpressions = new WebInspector.WatchExpressionsSidebarPane();
+ this.sidebarPanes.callstack = new WebInspector.CallStackSidebarPane();
+ this.sidebarPanes.scopechain = new WebInspector.ScopeChainSidebarPane();
+ this.sidebarPanes.breakpoints = new WebInspector.BreakpointsSidebarPane();
+
+ for (var pane in this.sidebarPanes)
+ this.sidebarElement.appendChild(this.sidebarPanes[pane].element);
+
+ this.sidebarPanes.callstack.expanded = true;
+ this.sidebarPanes.callstack.addEventListener("call frame selected", this._callFrameSelected, this);
+
+ this.sidebarPanes.scopechain.expanded = true;
+ this.sidebarPanes.breakpoints.expanded = true;
+
+ var panelEnablerHeading = WebInspector.UIString("You need to enable debugging before you can use the Scripts panel.");
+ var panelEnablerDisclaimer = WebInspector.UIString("Enabling debugging will make scripts run slower.");
+ var panelEnablerButton = WebInspector.UIString("Enable Debugging");
+
+ this.panelEnablerView = new WebInspector.PanelEnablerView("scripts", panelEnablerHeading, panelEnablerDisclaimer, panelEnablerButton);
+ this.panelEnablerView.addEventListener("enable clicked", this._enableDebugging, this);
+
+ this.element.appendChild(this.panelEnablerView.element);
+ this.element.appendChild(this.viewsContainerElement);
+ this.element.appendChild(this.sidebarElement);
+ this.element.appendChild(this.sidebarResizeElement);
+
+ this.enableToggleButton = new WebInspector.StatusBarButton("", "enable-toggle-status-bar-item");
+ this.enableToggleButton.addEventListener("click", this._toggleDebugging.bind(this), false);
+
+ this.pauseOnExceptionButton = new WebInspector.StatusBarButton("", "scripts-pause-on-exceptions-status-bar-item");
+ this.pauseOnExceptionButton.addEventListener("click", this._togglePauseOnExceptions.bind(this), false);
+
+ this._breakpointsURLMap = {};
+
+ this._shortcuts = {};
+
+ var isMac = InspectorController.platform().indexOf("mac-") === 0;
+ var platformSpecificModifier = isMac ? WebInspector.KeyboardShortcut.Modifiers.Meta : WebInspector.KeyboardShortcut.Modifiers.Ctrl;
+
+ // Continue.
+ var handler = this.pauseButton.click.bind(this.pauseButton);
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.F8);
+ this._shortcuts[shortcut] = handler;
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.Slash, platformSpecificModifier);
+ this._shortcuts[shortcut] = handler;
+
+ // Step over.
+ var handler = this.stepOverButton.click.bind(this.stepOverButton);
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.F10);
+ this._shortcuts[shortcut] = handler;
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.SingleQuote, platformSpecificModifier);
+ this._shortcuts[shortcut] = handler;
+
+ // Step into.
+ var handler = this.stepIntoButton.click.bind(this.stepIntoButton);
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.F11);
+ this._shortcuts[shortcut] = handler;
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.Semicolon, platformSpecificModifier);
+ this._shortcuts[shortcut] = handler;
+
+ // Step out.
+ var handler = this.stepOutButton.click.bind(this.stepOutButton);
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.F11, WebInspector.KeyboardShortcut.Modifiers.Shift);
+ this._shortcuts[shortcut] = handler;
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(WebInspector.KeyboardShortcut.KeyCodes.Semicolon, WebInspector.KeyboardShortcut.Modifiers.Shift, platformSpecificModifier);
+ this._shortcuts[shortcut] = handler;
+
+ this.reset();
+}
+
+WebInspector.ScriptsPanel.prototype = {
+ toolbarItemClass: "scripts",
+
+ get toolbarItemLabel()
+ {
+ return WebInspector.UIString("Scripts");
+ },
+
+ get statusBarItems()
+ {
+ return [this.enableToggleButton.element, this.pauseOnExceptionButton.element];
+ },
+
+ get paused()
+ {
+ return this._paused;
+ },
+
+ show: function()
+ {
+ WebInspector.Panel.prototype.show.call(this);
+ this.sidebarResizeElement.style.right = (this.sidebarElement.offsetWidth - 3) + "px";
+
+ if (this.visibleView) {
+ if (this.visibleView instanceof WebInspector.ResourceView)
+ this.visibleView.headersVisible = false;
+ this.visibleView.show(this.viewsContainerElement);
+ }
+
+ // Hide any views that are visible that are not this panel's current visible view.
+ // This can happen when a ResourceView is visible in the Resources panel then switched
+ // to the this panel.
+ for (var sourceID in this._sourceIDMap) {
+ var scriptOrResource = this._sourceIDMap[sourceID];
+ var view = this._sourceViewForScriptOrResource(scriptOrResource);
+ if (!view || view === this.visibleView)
+ continue;
+ view.visible = false;
+ }
+ if (this._attachDebuggerWhenShown) {
+ InspectorController.enableDebugger(false);
+ delete this._attachDebuggerWhenShown;
+ }
+ },
+
+ get searchableViews()
+ {
+ var views = [];
+
+ const visibleView = this.visibleView;
+ if (visibleView && visibleView.performSearch) {
+ visibleView.alreadySearching = true;
+ views.push(visibleView);
+ }
+
+ for (var sourceID in this._sourceIDMap) {
+ var scriptOrResource = this._sourceIDMap[sourceID];
+ var view = this._sourceViewForScriptOrResource(scriptOrResource);
+ if (!view || !view.performSearch || view.alreadySearching)
+ continue;
+
+ view.alreadySearching = true;
+ views.push(view);
+ }
+
+ for (var i = 0; i < views.length; ++i)
+ delete views[i].alreadySearching;
+
+ return views;
+ },
+
+ addScript: function(sourceID, sourceURL, source, startingLine, errorLine, errorMessage)
+ {
+ var script = new WebInspector.Script(sourceID, sourceURL, source, startingLine, errorLine, errorMessage);
+
+ if (sourceURL in WebInspector.resourceURLMap) {
+ var resource = WebInspector.resourceURLMap[sourceURL];
+ resource.addScript(script);
+ }
+
+ if (sourceURL in this._breakpointsURLMap && sourceID) {
+ var breakpoints = this._breakpointsURLMap[sourceURL];
+ var breakpointsLength = breakpoints.length;
+ for (var i = 0; i < breakpointsLength; ++i) {
+ var breakpoint = breakpoints[i];
+ if (startingLine <= breakpoint.line) {
+ breakpoint.sourceID = sourceID;
+ if (breakpoint.enabled)
+ InspectorController.addBreakpoint(breakpoint.sourceID, breakpoint.line, breakpoint.condition);
+ }
+ }
+ }
+
+ if (sourceID)
+ this._sourceIDMap[sourceID] = (resource || script);
+
+ this._addScriptToFilesMenu(script);
+ },
+
+ scriptOrResourceForID: function(id)
+ {
+ return this._sourceIDMap[id];
+ },
+
+ addBreakpoint: function(breakpoint)
+ {
+ this.sidebarPanes.breakpoints.addBreakpoint(breakpoint);
+
+ var sourceFrame;
+ if (breakpoint.url) {
+ if (!(breakpoint.url in this._breakpointsURLMap))
+ this._breakpointsURLMap[breakpoint.url] = [];
+ this._breakpointsURLMap[breakpoint.url].unshift(breakpoint);
+
+ if (breakpoint.url in WebInspector.resourceURLMap) {
+ var resource = WebInspector.resourceURLMap[breakpoint.url];
+ sourceFrame = this._sourceFrameForScriptOrResource(resource);
+ }
+ }
+
+ if (breakpoint.sourceID && !sourceFrame) {
+ var object = this._sourceIDMap[breakpoint.sourceID]
+ sourceFrame = this._sourceFrameForScriptOrResource(object);
+ }
+
+ if (sourceFrame)
+ sourceFrame.addBreakpoint(breakpoint);
+ },
+
+ removeBreakpoint: function(breakpoint)
+ {
+ this.sidebarPanes.breakpoints.removeBreakpoint(breakpoint);
+
+ var sourceFrame;
+ if (breakpoint.url && breakpoint.url in this._breakpointsURLMap) {
+ var breakpoints = this._breakpointsURLMap[breakpoint.url];
+ breakpoints.remove(breakpoint);
+ if (!breakpoints.length)
+ delete this._breakpointsURLMap[breakpoint.url];
+
+ if (breakpoint.url in WebInspector.resourceURLMap) {
+ var resource = WebInspector.resourceURLMap[breakpoint.url];
+ sourceFrame = this._sourceFrameForScriptOrResource(resource);
+ }
+ }
+
+ if (breakpoint.sourceID && !sourceFrame) {
+ var object = this._sourceIDMap[breakpoint.sourceID]
+ sourceFrame = this._sourceFrameForScriptOrResource(object);
+ }
+
+ if (sourceFrame)
+ sourceFrame.removeBreakpoint(breakpoint);
+ },
+
+ evaluateInSelectedCallFrame: function(code, updateInterface, callback)
+ {
+ var selectedCallFrame = this.sidebarPanes.callstack.selectedCallFrame;
+ if (!this._paused || !selectedCallFrame)
+ return;
+
+ if (typeof updateInterface === "undefined")
+ updateInterface = true;
+
+ var self = this;
+ function updatingCallbackWrapper(result, exception)
+ {
+ callback(result, exception);
+ if (updateInterface)
+ self.sidebarPanes.scopechain.update(selectedCallFrame);
+ }
+ this.doEvalInCallFrame(selectedCallFrame, code, updatingCallbackWrapper);
+ },
+
+ doEvalInCallFrame: function(callFrame, code, callback)
+ {
+ function evalCallback(result)
+ {
+ if (result)
+ callback(result.value, result.isException);
+ }
+ InjectedScriptAccess.evaluateInCallFrame(callFrame.id, code, evalCallback);
+ },
+
+ variablesInSelectedCallFrame: function()
+ {
+ var selectedCallFrame = this.sidebarPanes.callstack.selectedCallFrame;
+ if (!this._paused || !selectedCallFrame)
+ return {};
+
+ var result = {};
+ var scopeChain = selectedCallFrame.scopeChain;
+ for (var i = 0; i < scopeChain.length; ++i) {
+ var scopeObjectProperties = scopeChain[i].properties;
+ for (var j = 0; j < scopeObjectProperties.length; ++j)
+ result[scopeObjectProperties[j]] = true;
+ }
+ return result;
+ },
+
+ debuggerPaused: function(callFrames)
+ {
+ this._paused = true;
+ this._waitingToPause = false;
+ this._stepping = false;
+
+ this._updateDebuggerButtons();
+
+ this.sidebarPanes.callstack.update(callFrames, this._sourceIDMap);
+ this.sidebarPanes.callstack.selectedCallFrame = callFrames[0];
+
+ WebInspector.currentPanel = this;
+ window.focus();
+ },
+
+ debuggerResumed: function()
+ {
+ this._paused = false;
+ this._waitingToPause = false;
+ this._stepping = false;
+
+ this._clearInterface();
+ },
+
+ attachDebuggerWhenShown: function()
+ {
+ if (this.element.parentElement) {
+ InspectorController.enableDebugger(false);
+ } else {
+ this._attachDebuggerWhenShown = true;
+ }
+ },
+
+ debuggerWasEnabled: function()
+ {
+ this.reset();
+ },
+
+ debuggerWasDisabled: function()
+ {
+ this.reset();
+ },
+
+ reset: function()
+ {
+ this.visibleView = null;
+
+ delete this.currentQuery;
+ this.searchCanceled();
+
+ if (!InspectorController.debuggerEnabled()) {
+ this._paused = false;
+ this._waitingToPause = false;
+ this._stepping = false;
+ }
+
+ this._clearInterface();
+
+ this._backForwardList = [];
+ this._currentBackForwardIndex = -1;
+ this._updateBackAndForwardButtons();
+
+ this._scriptsForURLsInFilesSelect = {};
+ this.filesSelectElement.removeChildren();
+ this.functionsSelectElement.removeChildren();
+ this.viewsContainerElement.removeChildren();
+
+ if (this._sourceIDMap) {
+ for (var sourceID in this._sourceIDMap) {
+ var object = this._sourceIDMap[sourceID];
+ if (object instanceof WebInspector.Resource)
+ object.removeAllScripts();
+ }
+ }
+
+ this._sourceIDMap = {};
+
+ this.sidebarPanes.watchExpressions.refreshExpressions();
+ },
+
+ get visibleView()
+ {
+ return this._visibleView;
+ },
+
+ set visibleView(x)
+ {
+ if (this._visibleView === x)
+ return;
+
+ if (this._visibleView)
+ this._visibleView.hide();
+
+ this._visibleView = x;
+
+ if (x)
+ x.show(this.viewsContainerElement);
+ },
+
+ canShowResource: function(resource)
+ {
+ return resource && resource.scripts.length && InspectorController.debuggerEnabled();
+ },
+
+ showScript: function(script, line)
+ {
+ this._showScriptOrResource(script, line, true);
+ },
+
+ showResource: function(resource, line)
+ {
+ this._showScriptOrResource(resource, line, true);
+ },
+
+ showView: function(view)
+ {
+ if (!view)
+ return;
+ this._showScriptOrResource((view.resource || view.script));
+ },
+
+ handleKeyEvent: function(event)
+ {
+ var shortcut = WebInspector.KeyboardShortcut.makeKeyFromEvent(event);
+ var handler = this._shortcuts[shortcut];
+ if (handler) {
+ handler(event);
+ event.preventDefault();
+ event.handled = true;
+ } else {
+ this.sidebarPanes.callstack.handleKeyEvent(event);
+ }
+ },
+
+ scriptViewForScript: function(script)
+ {
+ if (!script)
+ return null;
+ if (!script._scriptView)
+ script._scriptView = new WebInspector.ScriptView(script);
+ return script._scriptView;
+ },
+
+ sourceFrameForScript: function(script)
+ {
+ var view = this.scriptViewForScript(script);
+ if (!view)
+ return null;
+
+ // Setting up the source frame requires that we be attached.
+ if (!this.element.parentNode)
+ this.attach();
+
+ view.setupSourceFrameIfNeeded();
+ return view.sourceFrame;
+ },
+
+ _sourceViewForScriptOrResource: function(scriptOrResource)
+ {
+ if (scriptOrResource instanceof WebInspector.Resource) {
+ if (!WebInspector.panels.resources)
+ return null;
+ return WebInspector.panels.resources.resourceViewForResource(scriptOrResource);
+ }
+ if (scriptOrResource instanceof WebInspector.Script)
+ return this.scriptViewForScript(scriptOrResource);
+ },
+
+ _sourceFrameForScriptOrResource: function(scriptOrResource)
+ {
+ if (scriptOrResource instanceof WebInspector.Resource) {
+ if (!WebInspector.panels.resources)
+ return null;
+ return WebInspector.panels.resources.sourceFrameForResource(scriptOrResource);
+ }
+ if (scriptOrResource instanceof WebInspector.Script)
+ return this.sourceFrameForScript(scriptOrResource);
+ },
+
+ _showScriptOrResource: function(scriptOrResource, line, shouldHighlightLine, fromBackForwardAction)
+ {
+ if (!scriptOrResource)
+ return;
+
+ var view;
+ if (scriptOrResource instanceof WebInspector.Resource) {
+ if (!WebInspector.panels.resources)
+ return null;
+ view = WebInspector.panels.resources.resourceViewForResource(scriptOrResource);
+ view.headersVisible = false;
+
+ if (scriptOrResource.url in this._breakpointsURLMap) {
+ var sourceFrame = this._sourceFrameForScriptOrResource(scriptOrResource);
+ if (sourceFrame && !sourceFrame.breakpoints.length) {
+ var breakpoints = this._breakpointsURLMap[scriptOrResource.url];
+ var breakpointsLength = breakpoints.length;
+ for (var i = 0; i < breakpointsLength; ++i)
+ sourceFrame.addBreakpoint(breakpoints[i]);
+ }
+ }
+ } else if (scriptOrResource instanceof WebInspector.Script)
+ view = this.scriptViewForScript(scriptOrResource);
+
+ if (!view)
+ return;
+
+ if (!fromBackForwardAction) {
+ var oldIndex = this._currentBackForwardIndex;
+ if (oldIndex >= 0)
+ this._backForwardList.splice(oldIndex + 1, this._backForwardList.length - oldIndex);
+
+ // Check for a previous entry of the same object in _backForwardList.
+ // If one is found, remove it and update _currentBackForwardIndex to match.
+ var previousEntryIndex = this._backForwardList.indexOf(scriptOrResource);
+ if (previousEntryIndex !== -1) {
+ this._backForwardList.splice(previousEntryIndex, 1);
+ --this._currentBackForwardIndex;
+ }
+
+ this._backForwardList.push(scriptOrResource);
+ ++this._currentBackForwardIndex;
+
+ this._updateBackAndForwardButtons();
+ }
+
+ this.visibleView = view;
+
+ if (line) {
+ if (view.revealLine)
+ view.revealLine(line);
+ if (view.highlightLine && shouldHighlightLine)
+ view.highlightLine(line);
+ }
+
+ var option;
+ if (scriptOrResource instanceof WebInspector.Script) {
+ option = scriptOrResource.filesSelectOption;
+ console.assert(option);
+ } else {
+ var url = scriptOrResource.url;
+ var script = this._scriptsForURLsInFilesSelect[url];
+ if (script)
+ option = script.filesSelectOption;
+ }
+
+ if (option)
+ this.filesSelectElement.selectedIndex = option.index;
+ },
+
+ _addScriptToFilesMenu: function(script)
+ {
+ if (script.resource && this._scriptsForURLsInFilesSelect[script.sourceURL])
+ return;
+
+ this._scriptsForURLsInFilesSelect[script.sourceURL] = script;
+
+ var select = this.filesSelectElement;
+
+ var option = document.createElement("option");
+ option.representedObject = (script.resource || script);
+ option.text = (script.sourceURL ? WebInspector.displayNameForURL(script.sourceURL) : WebInspector.UIString("(program)"));
+
+ function optionCompare(a, b)
+ {
+ var aTitle = a.text.toLowerCase();
+ var bTitle = b.text.toLowerCase();
+ if (aTitle < bTitle)
+ return -1;
+ else if (aTitle > bTitle)
+ return 1;
+
+ var aSourceID = a.representedObject.sourceID;
+ var bSourceID = b.representedObject.sourceID;
+ if (aSourceID < bSourceID)
+ return -1;
+ else if (aSourceID > bSourceID)
+ return 1;
+ return 0;
+ }
+
+ var insertionIndex = insertionIndexForObjectInListSortedByFunction(option, select.childNodes, optionCompare);
+ if (insertionIndex < 0)
+ select.appendChild(option);
+ else
+ select.insertBefore(option, select.childNodes.item(insertionIndex));
+
+ script.filesSelectOption = option;
+
+ // Call _showScriptOrResource if the option we just appended ended up being selected.
+ // This will happen for the first item added to the menu.
+ if (select.options[select.selectedIndex] === option)
+ this._showScriptOrResource(option.representedObject);
+ },
+
+ _clearCurrentExecutionLine: function()
+ {
+ if (this._executionSourceFrame)
+ this._executionSourceFrame.executionLine = 0;
+ delete this._executionSourceFrame;
+ },
+
+ _callFrameSelected: function()
+ {
+ this._clearCurrentExecutionLine();
+
+ var callStackPane = this.sidebarPanes.callstack;
+ var currentFrame = callStackPane.selectedCallFrame;
+ if (!currentFrame)
+ return;
+
+ this.sidebarPanes.scopechain.update(currentFrame);
+ this.sidebarPanes.watchExpressions.refreshExpressions();
+
+ var scriptOrResource = this._sourceIDMap[currentFrame.sourceID];
+ this._showScriptOrResource(scriptOrResource, currentFrame.line);
+
+ this._executionSourceFrame = this._sourceFrameForScriptOrResource(scriptOrResource);
+ if (this._executionSourceFrame)
+ this._executionSourceFrame.executionLine = currentFrame.line;
+ },
+
+ _changeVisibleFile: function(event)
+ {
+ var select = this.filesSelectElement;
+ this._showScriptOrResource(select.options[select.selectedIndex].representedObject);
+ },
+
+ _startSidebarResizeDrag: function(event)
+ {
+ WebInspector.elementDragStart(this.sidebarElement, this._sidebarResizeDrag.bind(this), this._endSidebarResizeDrag.bind(this), event, "col-resize");
+
+ if (event.target === this.sidebarResizeWidgetElement)
+ this._dragOffset = (event.target.offsetWidth - (event.pageX - event.target.totalOffsetLeft));
+ else
+ this._dragOffset = 0;
+ },
+
+ _endSidebarResizeDrag: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+
+ delete this._dragOffset;
+ },
+
+ _sidebarResizeDrag: function(event)
+ {
+ var x = event.pageX + this._dragOffset;
+ var newWidth = Number.constrain(window.innerWidth - x, Preferences.minScriptsSidebarWidth, window.innerWidth * 0.66);
+
+ this.sidebarElement.style.width = newWidth + "px";
+ this.sidebarButtonsElement.style.width = newWidth + "px";
+ this.viewsContainerElement.style.right = newWidth + "px";
+ this.sidebarResizeWidgetElement.style.right = newWidth + "px";
+ this.sidebarResizeElement.style.right = (newWidth - 3) + "px";
+
+ event.preventDefault();
+ },
+
+ _updatePauseOnExceptionsButton: function()
+ {
+ if (InspectorController.pauseOnExceptions()) {
+ this.pauseOnExceptionButton.title = WebInspector.UIString("Don't pause on exceptions.");
+ this.pauseOnExceptionButton.toggled = true;
+ } else {
+ this.pauseOnExceptionButton.title = WebInspector.UIString("Pause on exceptions.");
+ this.pauseOnExceptionButton.toggled = false;
+ }
+ },
+
+ _updateDebuggerButtons: function()
+ {
+ if (InspectorController.debuggerEnabled()) {
+ this.enableToggleButton.title = WebInspector.UIString("Debugging enabled. Click to disable.");
+ this.enableToggleButton.toggled = true;
+ this.pauseOnExceptionButton.visible = true;
+ this.panelEnablerView.visible = false;
+ } else {
+ this.enableToggleButton.title = WebInspector.UIString("Debugging disabled. Click to enable.");
+ this.enableToggleButton.toggled = false;
+ this.pauseOnExceptionButton.visible = false;
+ this.panelEnablerView.visible = true;
+ }
+
+ this._updatePauseOnExceptionsButton();
+
+ if (this._paused) {
+ this.pauseButton.addStyleClass("paused");
+
+ this.pauseButton.disabled = false;
+ this.stepOverButton.disabled = false;
+ this.stepIntoButton.disabled = false;
+ this.stepOutButton.disabled = false;
+
+ this.debuggerStatusElement.textContent = WebInspector.UIString("Paused");
+ } else {
+ this.pauseButton.removeStyleClass("paused");
+
+ this.pauseButton.disabled = this._waitingToPause;
+ this.stepOverButton.disabled = true;
+ this.stepIntoButton.disabled = true;
+ this.stepOutButton.disabled = true;
+
+ if (this._waitingToPause)
+ this.debuggerStatusElement.textContent = WebInspector.UIString("Pausing");
+ else if (this._stepping)
+ this.debuggerStatusElement.textContent = WebInspector.UIString("Stepping");
+ else
+ this.debuggerStatusElement.textContent = "";
+ }
+ },
+
+ _updateBackAndForwardButtons: function()
+ {
+ this.backButton.disabled = this._currentBackForwardIndex <= 0;
+ this.forwardButton.disabled = this._currentBackForwardIndex >= (this._backForwardList.length - 1);
+ },
+
+ _clearInterface: function()
+ {
+ this.sidebarPanes.callstack.update(null);
+ this.sidebarPanes.scopechain.update(null);
+
+ this._clearCurrentExecutionLine();
+ this._updateDebuggerButtons();
+ },
+
+ _goBack: function()
+ {
+ if (this._currentBackForwardIndex <= 0) {
+ console.error("Can't go back from index " + this._currentBackForwardIndex);
+ return;
+ }
+
+ this._showScriptOrResource(this._backForwardList[--this._currentBackForwardIndex], null, false, true);
+ this._updateBackAndForwardButtons();
+ },
+
+ _goForward: function()
+ {
+ if (this._currentBackForwardIndex >= this._backForwardList.length - 1) {
+ console.error("Can't go forward from index " + this._currentBackForwardIndex);
+ return;
+ }
+
+ this._showScriptOrResource(this._backForwardList[++this._currentBackForwardIndex], null, false, true);
+ this._updateBackAndForwardButtons();
+ },
+
+ _enableDebugging: function()
+ {
+ if (InspectorController.debuggerEnabled())
+ return;
+ this._toggleDebugging(this.panelEnablerView.alwaysEnabled);
+ },
+
+ _toggleDebugging: function(optionalAlways)
+ {
+ this._paused = false;
+ this._waitingToPause = false;
+ this._stepping = false;
+
+ if (InspectorController.debuggerEnabled())
+ InspectorController.disableDebugger(true);
+ else
+ InspectorController.enableDebugger(!!optionalAlways);
+ },
+
+ _togglePauseOnExceptions: function()
+ {
+ InspectorController.setPauseOnExceptions(!InspectorController.pauseOnExceptions());
+ this._updatePauseOnExceptionsButton();
+ },
+
+ _togglePause: function()
+ {
+ if (this._paused) {
+ this._paused = false;
+ this._waitingToPause = false;
+ InspectorController.resumeDebugger();
+ } else {
+ this._stepping = false;
+ this._waitingToPause = true;
+ InspectorController.pauseInDebugger();
+ }
+
+ this._clearInterface();
+ },
+
+ _stepOverClicked: function()
+ {
+ this._paused = false;
+ this._stepping = true;
+
+ this._clearInterface();
+
+ InspectorController.stepOverStatementInDebugger();
+ },
+
+ _stepIntoClicked: function()
+ {
+ this._paused = false;
+ this._stepping = true;
+
+ this._clearInterface();
+
+ InspectorController.stepIntoStatementInDebugger();
+ },
+
+ _stepOutClicked: function()
+ {
+ this._paused = false;
+ this._stepping = true;
+
+ this._clearInterface();
+
+ InspectorController.stepOutOfFunctionInDebugger();
+ }
+}
+
+WebInspector.ScriptsPanel.prototype.__proto__ = WebInspector.Panel.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarPane.js
new file mode 100644
index 0000000..af9e5f9
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarPane.js
@@ -0,0 +1,125 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.SidebarPane = function(title)
+{
+ this.element = document.createElement("div");
+ this.element.className = "pane";
+
+ this.titleElement = document.createElement("div");
+ this.titleElement.className = "title";
+ this.titleElement.addEventListener("click", this.toggleExpanded.bind(this), false);
+
+ this.bodyElement = document.createElement("div");
+ this.bodyElement.className = "body";
+
+ this.element.appendChild(this.titleElement);
+ this.element.appendChild(this.bodyElement);
+
+ this.title = title;
+ this.growbarVisible = false;
+ this.expanded = false;
+}
+
+WebInspector.SidebarPane.prototype = {
+ get title()
+ {
+ return this._title;
+ },
+
+ set title(x)
+ {
+ if (this._title === x)
+ return;
+ this._title = x;
+ this.titleElement.textContent = x;
+ },
+
+ get growbarVisible()
+ {
+ return this._growbarVisible;
+ },
+
+ set growbarVisible(x)
+ {
+ if (this._growbarVisible === x)
+ return;
+
+ this._growbarVisible = x;
+
+ if (x && !this._growbarElement) {
+ this._growbarElement = document.createElement("div");
+ this._growbarElement.className = "growbar";
+ this.element.appendChild(this._growbarElement);
+ } else if (!x && this._growbarElement) {
+ if (this._growbarElement.parentNode)
+ this._growbarElement.parentNode(this._growbarElement);
+ delete this._growbarElement;
+ }
+ },
+
+ get expanded()
+ {
+ return this._expanded;
+ },
+
+ set expanded(x)
+ {
+ if (x)
+ this.expand();
+ else
+ this.collapse();
+ },
+
+ expand: function()
+ {
+ if (this._expanded)
+ return;
+ this._expanded = true;
+ this.element.addStyleClass("expanded");
+ if (this.onexpand)
+ this.onexpand(this);
+ },
+
+ collapse: function()
+ {
+ if (!this._expanded)
+ return;
+ this._expanded = false;
+ this.element.removeStyleClass("expanded");
+ if (this.oncollapse)
+ this.oncollapse(this);
+ },
+
+ toggleExpanded: function()
+ {
+ this.expanded = !this.expanded;
+ }
+}
+
+WebInspector.SidebarPane.prototype.__proto__ = WebInspector.Object.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarTreeElement.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarTreeElement.js
new file mode 100644
index 0000000..c08b0ef
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SidebarTreeElement.js
@@ -0,0 +1,201 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.SidebarSectionTreeElement = function(title, representedObject, hasChildren)
+{
+ TreeElement.call(this, title.escapeHTML(), representedObject || {}, hasChildren);
+}
+
+WebInspector.SidebarSectionTreeElement.prototype = {
+ selectable: false,
+
+ get smallChildren()
+ {
+ return this._smallChildren;
+ },
+
+ set smallChildren(x)
+ {
+ if (this._smallChildren === x)
+ return;
+
+ this._smallChildren = x;
+
+ if (this._smallChildren)
+ this._childrenListNode.addStyleClass("small");
+ else
+ this._childrenListNode.removeStyleClass("small");
+ },
+
+ onattach: function()
+ {
+ this._listItemNode.addStyleClass("sidebar-tree-section");
+ },
+
+ onreveal: function()
+ {
+ if (this.listItemElement)
+ this.listItemElement.scrollIntoViewIfNeeded(false);
+ }
+}
+
+WebInspector.SidebarSectionTreeElement.prototype.__proto__ = TreeElement.prototype;
+
+WebInspector.SidebarTreeElement = function(className, title, subtitle, representedObject, hasChildren)
+{
+ TreeElement.call(this, "", representedObject || {}, hasChildren);
+
+ if (hasChildren) {
+ this.disclosureButton = document.createElement("button");
+ this.disclosureButton.className = "disclosure-button";
+ }
+
+ if (!this.iconElement) {
+ this.iconElement = document.createElement("img");
+ this.iconElement.className = "icon";
+ }
+
+ this.statusElement = document.createElement("div");
+ this.statusElement.className = "status";
+
+ this.titlesElement = document.createElement("div");
+ this.titlesElement.className = "titles";
+
+ this.titleElement = document.createElement("span");
+ this.titleElement.className = "title";
+ this.titlesElement.appendChild(this.titleElement);
+
+ this.subtitleElement = document.createElement("span");
+ this.subtitleElement.className = "subtitle";
+ this.titlesElement.appendChild(this.subtitleElement);
+
+ this.className = className;
+ this.mainTitle = title;
+ this.subtitle = subtitle;
+}
+
+WebInspector.SidebarTreeElement.prototype = {
+ get small()
+ {
+ return this._small;
+ },
+
+ set small(x)
+ {
+ this._small = x;
+
+ if (this._listItemNode) {
+ if (this._small)
+ this._listItemNode.addStyleClass("small");
+ else
+ this._listItemNode.removeStyleClass("small");
+ }
+ },
+
+ get mainTitle()
+ {
+ return this._mainTitle;
+ },
+
+ set mainTitle(x)
+ {
+ this._mainTitle = x;
+ this.refreshTitles();
+ },
+
+ get subtitle()
+ {
+ return this._subtitle;
+ },
+
+ set subtitle(x)
+ {
+ this._subtitle = x;
+ this.refreshTitles();
+ },
+
+ get bubbleText()
+ {
+ return this._bubbleText;
+ },
+
+ set bubbleText(x)
+ {
+ if (!this.bubbleElement) {
+ this.bubbleElement = document.createElement("div");
+ this.bubbleElement.className = "bubble";
+ this.statusElement.appendChild(this.bubbleElement);
+ }
+
+ this._bubbleText = x;
+ this.bubbleElement.textContent = x;
+ },
+
+ refreshTitles: function()
+ {
+ var mainTitle = this.mainTitle;
+ if (this.titleElement.textContent !== mainTitle)
+ this.titleElement.textContent = mainTitle;
+
+ var subtitle = this.subtitle;
+ if (subtitle) {
+ if (this.subtitleElement.textContent !== subtitle)
+ this.subtitleElement.textContent = subtitle;
+ this.titlesElement.removeStyleClass("no-subtitle");
+ } else
+ this.titlesElement.addStyleClass("no-subtitle");
+ },
+
+ isEventWithinDisclosureTriangle: function(event)
+ {
+ return event.target === this.disclosureButton;
+ },
+
+ onattach: function()
+ {
+ this._listItemNode.addStyleClass("sidebar-tree-item");
+
+ if (this.className)
+ this._listItemNode.addStyleClass(this.className);
+
+ if (this.small)
+ this._listItemNode.addStyleClass("small");
+
+ if (this.hasChildren && this.disclosureButton)
+ this._listItemNode.appendChild(this.disclosureButton);
+
+ this._listItemNode.appendChild(this.iconElement);
+ this._listItemNode.appendChild(this.statusElement);
+ this._listItemNode.appendChild(this.titlesElement);
+ },
+
+ onreveal: function()
+ {
+ if (this._listItemNode)
+ this._listItemNode.scrollIntoViewIfNeeded(false);
+ }
+}
+
+WebInspector.SidebarTreeElement.prototype.__proto__ = TreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceFrame.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceFrame.js
new file mode 100644
index 0000000..e364cb2
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceFrame.js
@@ -0,0 +1,906 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.SourceFrame = function(element, addBreakpointDelegate)
+{
+ this.messages = [];
+ this.breakpoints = [];
+ this._shortcuts = {};
+
+ this.addBreakpointDelegate = addBreakpointDelegate;
+
+ this.element = element || document.createElement("iframe");
+ this.element.addStyleClass("source-view-frame");
+ this.element.setAttribute("viewsource", "true");
+
+ this.element.addEventListener("load", this._loaded.bind(this), false);
+}
+
+WebInspector.SourceFrame.prototype = {
+ get executionLine()
+ {
+ return this._executionLine;
+ },
+
+ set executionLine(x)
+ {
+ if (this._executionLine === x)
+ return;
+
+ var previousLine = this._executionLine;
+ this._executionLine = x;
+
+ this._updateExecutionLine(previousLine);
+ },
+
+ get autoSizesToFitContentHeight()
+ {
+ return this._autoSizesToFitContentHeight;
+ },
+
+ set autoSizesToFitContentHeight(x)
+ {
+ if (this._autoSizesToFitContentHeight === x)
+ return;
+
+ this._autoSizesToFitContentHeight = x;
+
+ if (this._autoSizesToFitContentHeight) {
+ this._windowResizeListener = this._windowResized.bind(this);
+ window.addEventListener("resize", this._windowResizeListener, false);
+ this.sizeToFitContentHeight();
+ } else {
+ this.element.style.removeProperty("height");
+ if (this.element.contentDocument)
+ this.element.contentDocument.body.removeStyleClass("webkit-height-sized-to-fit");
+ window.removeEventListener("resize", this._windowResizeListener, false);
+ delete this._windowResizeListener;
+ }
+ },
+
+ sourceRow: function(lineNumber)
+ {
+ if (!lineNumber || !this.element.contentDocument)
+ return;
+
+ var table = this.element.contentDocument.getElementsByTagName("table")[0];
+ if (!table)
+ return;
+
+ var rows = table.rows;
+
+ // Line numbers are a 1-based index, but the rows collection is 0-based.
+ --lineNumber;
+
+ return rows[lineNumber];
+ },
+
+ lineNumberForSourceRow: function(sourceRow)
+ {
+ // Line numbers are a 1-based index, but the rows collection is 0-based.
+ var lineNumber = 0;
+ while (sourceRow) {
+ ++lineNumber;
+ sourceRow = sourceRow.previousSibling;
+ }
+
+ return lineNumber;
+ },
+
+ revealLine: function(lineNumber)
+ {
+ if (!this._isContentLoaded()) {
+ this._lineNumberToReveal = lineNumber;
+ return;
+ }
+
+ var row = this.sourceRow(lineNumber);
+ if (row)
+ row.scrollIntoViewIfNeeded(true);
+ },
+
+ addBreakpoint: function(breakpoint)
+ {
+ this.breakpoints.push(breakpoint);
+ breakpoint.addEventListener("enabled", this._breakpointEnableChanged, this);
+ breakpoint.addEventListener("disabled", this._breakpointEnableChanged, this);
+ this._addBreakpointToSource(breakpoint);
+ },
+
+ removeBreakpoint: function(breakpoint)
+ {
+ this.breakpoints.remove(breakpoint);
+ breakpoint.removeEventListener("enabled", null, this);
+ breakpoint.removeEventListener("disabled", null, this);
+ this._removeBreakpointFromSource(breakpoint);
+ },
+
+ addMessage: function(msg)
+ {
+ // Don't add the message if there is no message or valid line or if the msg isn't an error or warning.
+ if (!msg.message || msg.line <= 0 || !msg.isErrorOrWarning())
+ return;
+ this.messages.push(msg);
+ this._addMessageToSource(msg);
+ },
+
+ clearMessages: function()
+ {
+ this.messages = [];
+
+ if (!this.element.contentDocument)
+ return;
+
+ var bubbles = this.element.contentDocument.querySelectorAll(".webkit-html-message-bubble");
+ if (!bubbles)
+ return;
+
+ for (var i = 0; i < bubbles.length; ++i) {
+ var bubble = bubbles[i];
+ bubble.parentNode.removeChild(bubble);
+ }
+ },
+
+ sizeToFitContentHeight: function()
+ {
+ if (this.element.contentDocument) {
+ this.element.style.setProperty("height", this.element.contentDocument.body.offsetHeight + "px");
+ this.element.contentDocument.body.addStyleClass("webkit-height-sized-to-fit");
+ }
+ },
+
+ _highlightLineEnds: function(event)
+ {
+ event.target.parentNode.removeStyleClass("webkit-highlighted-line");
+ },
+
+ highlightLine: function(lineNumber)
+ {
+ if (!this._isContentLoaded()) {
+ this._lineNumberToHighlight = lineNumber;
+ return;
+ }
+
+ var sourceRow = this.sourceRow(lineNumber);
+ if (!sourceRow)
+ return;
+ var line = sourceRow.getElementsByClassName('webkit-line-content')[0];
+ // Trick to reset the animation if the user clicks on the same link
+ // Using a timeout to avoid coalesced style updates
+ line.style.setProperty("-webkit-animation-name", "none");
+ setTimeout(function () {
+ line.style.removeProperty("-webkit-animation-name");
+ sourceRow.addStyleClass("webkit-highlighted-line");
+ }, 0);
+ },
+
+ _loaded: function()
+ {
+ WebInspector.addMainEventListeners(this.element.contentDocument);
+ this.element.contentDocument.addEventListener("contextmenu", this._documentContextMenu.bind(this), true);
+ this.element.contentDocument.addEventListener("mousedown", this._documentMouseDown.bind(this), true);
+ this.element.contentDocument.addEventListener("keydown", this._documentKeyDown.bind(this), true);
+ this.element.contentDocument.addEventListener("keyup", WebInspector.documentKeyUp.bind(WebInspector), true);
+ this.element.contentDocument.addEventListener("webkitAnimationEnd", this._highlightLineEnds.bind(this), false);
+
+ // Register 'eval' shortcut.
+ var isMac = InspectorController.platform().indexOf("mac-") === 0;
+ var platformSpecificModifier = isMac ? WebInspector.KeyboardShortcut.Modifiers.Meta : WebInspector.KeyboardShortcut.Modifiers.Ctrl;
+ var shortcut = WebInspector.KeyboardShortcut.makeKey(69 /* 'E' */, platformSpecificModifier | WebInspector.KeyboardShortcut.Modifiers.Shift);
+ this._shortcuts[shortcut] = this._evalSelectionInCallFrame.bind(this);
+
+ var headElement = this.element.contentDocument.getElementsByTagName("head")[0];
+ if (!headElement) {
+ headElement = this.element.contentDocument.createElement("head");
+ this.element.contentDocument.documentElement.insertBefore(headElement, this.element.contentDocument.documentElement.firstChild);
+ }
+
+ var styleElement = this.element.contentDocument.createElement("style");
+ headElement.appendChild(styleElement);
+
+ // Add these style rules here since they are specific to the Inspector. They also behave oddly and not
+ // all properties apply if added to view-source.css (becuase it is a user agent sheet.)
+ var styleText = ".webkit-line-number { background-repeat: no-repeat; background-position: right 1px; }\n";
+ styleText += ".webkit-execution-line .webkit-line-number { color: transparent; background-image: -webkit-canvas(program-counter); }\n";
+
+ styleText += ".webkit-breakpoint .webkit-line-number { color: white; background-image: -webkit-canvas(breakpoint); }\n";
+ styleText += ".webkit-breakpoint-disabled .webkit-line-number { color: white; background-image: -webkit-canvas(breakpoint-disabled); }\n";
+ styleText += ".webkit-breakpoint.webkit-execution-line .webkit-line-number { color: transparent; background-image: -webkit-canvas(breakpoint-program-counter); }\n";
+ styleText += ".webkit-breakpoint-disabled.webkit-execution-line .webkit-line-number { color: transparent; background-image: -webkit-canvas(breakpoint-disabled-program-counter); }\n";
+
+ styleText += ".webkit-breakpoint.webkit-breakpoint-conditional .webkit-line-number { color: white; background-image: -webkit-canvas(breakpoint-conditional); }\n";
+ styleText += ".webkit-breakpoint-disabled.webkit-breakpoint-conditional .webkit-line-number { color: white; background-image: -webkit-canvas(breakpoint-disabled-conditional); }\n";
+ styleText += ".webkit-breakpoint.webkit-breakpoint-conditional.webkit-execution-line .webkit-line-number { color: transparent; background-image: -webkit-canvas(breakpoint-conditional-program-counter); }\n";
+ styleText += ".webkit-breakpoint-disabled.webkit-breakpoint-conditional.webkit-execution-line .webkit-line-number { color: transparent; background-image: -webkit-canvas(breakpoint-disabled-conditional-program-counter); }\n";
+
+ styleText += ".webkit-execution-line .webkit-line-content { background-color: rgb(171, 191, 254); outline: 1px solid rgb(64, 115, 244); }\n";
+ styleText += ".webkit-height-sized-to-fit { overflow-y: hidden }\n";
+ styleText += ".webkit-line-content { background-color: white; }\n";
+ styleText += "@-webkit-keyframes fadeout {from {background-color: rgb(255, 255, 120);} to { background-color: white;}}\n";
+ styleText += ".webkit-highlighted-line .webkit-line-content { background-color: rgb(255, 255, 120); -webkit-animation: 'fadeout' 2s 500ms}\n";
+ styleText += ".webkit-javascript-comment { color: rgb(0, 116, 0); }\n";
+ styleText += ".webkit-javascript-keyword { color: rgb(170, 13, 145); }\n";
+ styleText += ".webkit-javascript-number { color: rgb(28, 0, 207); }\n";
+ styleText += ".webkit-javascript-string, .webkit-javascript-regexp { color: rgb(196, 26, 22); }\n";
+
+ // TODO: Move these styles into inspector.css once https://bugs.webkit.org/show_bug.cgi?id=28913 is fixed and popup moved into the top frame.
+ styleText += ".popup-content { position: absolute; z-index: 10000; padding: 4px; background-color: rgb(203, 226, 255); -webkit-border-radius: 7px; border: 2px solid rgb(169, 172, 203); }";
+ styleText += ".popup-glasspane { position: absolute; top: 0; left: 0; height: 100%; width: 100%; opacity: 0; z-index: 9900; }";
+ styleText += ".popup-message { background-color: transparent; font-family: Lucida Grande, sans-serif; font-weight: normal; font-size: 11px; text-align: left; text-shadow: none; color: rgb(85, 85, 85); cursor: default; margin: 0 0 2px 0; }";
+ styleText += ".popup-content.breakpoint-condition { width: 90%; }";
+ styleText += ".popup-content input#bp-condition { font-family: monospace; margin: 0; border: 1px inset rgb(190, 190, 190) !important; width: 100%; box-shadow: none !important; outline: none !important; -webkit-user-modify: read-write; }";
+ // This class is already in inspector.css
+ styleText += ".hidden { display: none !important; }";
+
+ styleElement.textContent = styleText;
+
+ this._needsProgramCounterImage = true;
+ this._needsBreakpointImages = true;
+
+ this.element.contentWindow.Element.prototype.addStyleClass = Element.prototype.addStyleClass;
+ this.element.contentWindow.Element.prototype.removeStyleClass = Element.prototype.removeStyleClass;
+ this.element.contentWindow.Element.prototype.positionAt = Element.prototype.positionAt;
+ this.element.contentWindow.Element.prototype.removeMatchingStyleClasses = Element.prototype.removeMatchingStyleClasses;
+ this.element.contentWindow.Element.prototype.hasStyleClass = Element.prototype.hasStyleClass;
+ this.element.contentWindow.Element.prototype.pageOffsetRelativeToWindow = Element.prototype.pageOffsetRelativeToWindow;
+ this.element.contentWindow.Element.prototype.__defineGetter__("totalOffsetLeft", Element.prototype.__lookupGetter__("totalOffsetLeft"));
+ this.element.contentWindow.Element.prototype.__defineGetter__("totalOffsetTop", Element.prototype.__lookupGetter__("totalOffsetTop"));
+ this.element.contentWindow.Node.prototype.enclosingNodeOrSelfWithNodeName = Node.prototype.enclosingNodeOrSelfWithNodeName;
+ this.element.contentWindow.Node.prototype.enclosingNodeOrSelfWithNodeNameInArray = Node.prototype.enclosingNodeOrSelfWithNodeNameInArray;
+
+ this._addExistingMessagesToSource();
+ this._addExistingBreakpointsToSource();
+ this._updateExecutionLine();
+ if (this._executionLine)
+ this.revealLine(this._executionLine);
+
+ if (this.autoSizesToFitContentHeight)
+ this.sizeToFitContentHeight();
+
+ if (this._lineNumberToReveal) {
+ this.revealLine(this._lineNumberToReveal);
+ delete this._lineNumberToReveal;
+ }
+
+ if (this._lineNumberToHighlight) {
+ this.highlightLine(this._lineNumberToHighlight);
+ delete this._lineNumberToHighlight;
+ }
+
+ this.dispatchEventToListeners("content loaded");
+ },
+
+ _isContentLoaded: function() {
+ var doc = this.element.contentDocument;
+ return doc && doc.getElementsByTagName("table")[0];
+ },
+
+ _windowResized: function(event)
+ {
+ if (!this._autoSizesToFitContentHeight)
+ return;
+ this.sizeToFitContentHeight();
+ },
+
+ _documentContextMenu: function(event)
+ {
+ if (!event.target.hasStyleClass("webkit-line-number"))
+ return;
+ var sourceRow = event.target.enclosingNodeOrSelfWithNodeName("tr");
+ if (!sourceRow._breakpointObject && this.addBreakpointDelegate)
+ this.addBreakpointDelegate(this.lineNumberForSourceRow(sourceRow));
+
+ var breakpoint = sourceRow._breakpointObject;
+ if (!breakpoint)
+ return;
+
+ this._editBreakpointCondition(event.target, sourceRow, breakpoint);
+ event.preventDefault();
+ },
+
+ _documentMouseDown: function(event)
+ {
+ if (!event.target.hasStyleClass("webkit-line-number"))
+ return;
+ if (event.button != 0 || event.altKey || event.ctrlKey || event.metaKey || event.shiftKey)
+ return;
+ var sourceRow = event.target.enclosingNodeOrSelfWithNodeName("tr");
+ if (sourceRow._breakpointObject && sourceRow._breakpointObject.enabled)
+ sourceRow._breakpointObject.enabled = false;
+ else if (sourceRow._breakpointObject)
+ WebInspector.panels.scripts.removeBreakpoint(sourceRow._breakpointObject);
+ else if (this.addBreakpointDelegate)
+ this.addBreakpointDelegate(this.lineNumberForSourceRow(sourceRow));
+ },
+
+ _editBreakpointCondition: function(eventTarget, sourceRow, breakpoint)
+ {
+ // TODO: Migrate the popup to the top-level document and remove the blur listener from conditionElement once https://bugs.webkit.org/show_bug.cgi?id=28913 is fixed.
+ var popupDocument = this.element.contentDocument;
+ this._showBreakpointConditionPopup(eventTarget, breakpoint.line, popupDocument);
+
+ function committed(element, newText)
+ {
+ breakpoint.condition = newText;
+ if (breakpoint.condition)
+ sourceRow.addStyleClass("webkit-breakpoint-conditional");
+ else
+ sourceRow.removeStyleClass("webkit-breakpoint-conditional");
+ dismissed.call(this);
+ }
+
+ function dismissed()
+ {
+ this._popup.hide();
+ delete this._conditionEditorElement;
+ }
+
+ var dismissedHandler = dismissed.bind(this);
+ this._conditionEditorElement.addEventListener("blur", dismissedHandler, false);
+
+ WebInspector.startEditing(this._conditionEditorElement, committed.bind(this), dismissedHandler);
+ this._conditionEditorElement.value = breakpoint.condition;
+ this._conditionEditorElement.select();
+ },
+
+ _showBreakpointConditionPopup: function(clickedElement, lineNumber, popupDocument)
+ {
+ var popupContentElement = this._createPopupElement(lineNumber, popupDocument);
+ var lineElement = clickedElement.enclosingNodeOrSelfWithNodeName("td").nextSibling;
+ if (this._popup) {
+ this._popup.hide();
+ this._popup.element = popupContentElement;
+ } else {
+ this._popup = new WebInspector.Popup(popupContentElement);
+ this._popup.autoHide = true;
+ }
+ this._popup.anchor = lineElement;
+ this._popup.show();
+ },
+
+ _createPopupElement: function(lineNumber, popupDocument)
+ {
+ var popupContentElement = popupDocument.createElement("div");
+ popupContentElement.className = "popup-content breakpoint-condition";
+
+ var labelElement = document.createElement("label");
+ labelElement.className = "popup-message";
+ labelElement.htmlFor = "bp-condition";
+ labelElement.appendChild(document.createTextNode(WebInspector.UIString("The breakpoint on line %d will stop only if this expression is true:", lineNumber)));
+ popupContentElement.appendChild(labelElement);
+
+ var editorElement = document.createElement("input");
+ editorElement.id = "bp-condition";
+ editorElement.type = "text"
+ popupContentElement.appendChild(editorElement);
+ this._conditionEditorElement = editorElement;
+
+ return popupContentElement;
+ },
+
+ _documentKeyDown: function(event)
+ {
+ var shortcut = WebInspector.KeyboardShortcut.makeKeyFromEvent(event);
+ var handler = this._shortcuts[shortcut];
+ if (handler) {
+ handler(event);
+ event.preventDefault();
+ } else {
+ WebInspector.documentKeyDown(event);
+ }
+ },
+
+ _evalSelectionInCallFrame: function(event)
+ {
+ if (!WebInspector.panels.scripts || !WebInspector.panels.scripts.paused)
+ return;
+
+ var selection = this.element.contentWindow.getSelection();
+ if (!selection.rangeCount)
+ return;
+
+ var expression = selection.getRangeAt(0).toString().trimWhitespace();
+ WebInspector.panels.scripts.evaluateInSelectedCallFrame(expression, false, function(result, exception) {
+ WebInspector.showConsole();
+ var commandMessage = new WebInspector.ConsoleCommand(expression);
+ WebInspector.console.addMessage(commandMessage);
+ WebInspector.console.addMessage(new WebInspector.ConsoleCommandResult(result, exception, commandMessage));
+ });
+ },
+
+ _breakpointEnableChanged: function(event)
+ {
+ var breakpoint = event.target;
+ var sourceRow = this.sourceRow(breakpoint.line);
+ if (!sourceRow)
+ return;
+
+ sourceRow.addStyleClass("webkit-breakpoint");
+
+ if (breakpoint.enabled)
+ sourceRow.removeStyleClass("webkit-breakpoint-disabled");
+ else
+ sourceRow.addStyleClass("webkit-breakpoint-disabled");
+ },
+
+ _updateExecutionLine: function(previousLine)
+ {
+ if (previousLine) {
+ var sourceRow = this.sourceRow(previousLine);
+ if (sourceRow)
+ sourceRow.removeStyleClass("webkit-execution-line");
+ }
+
+ if (!this._executionLine)
+ return;
+
+ this._drawProgramCounterImageIfNeeded();
+
+ var sourceRow = this.sourceRow(this._executionLine);
+ if (sourceRow)
+ sourceRow.addStyleClass("webkit-execution-line");
+ },
+
+ _addExistingBreakpointsToSource: function()
+ {
+ var length = this.breakpoints.length;
+ for (var i = 0; i < length; ++i)
+ this._addBreakpointToSource(this.breakpoints[i]);
+ },
+
+ _addBreakpointToSource: function(breakpoint)
+ {
+ var sourceRow = this.sourceRow(breakpoint.line);
+ if (!sourceRow)
+ return;
+
+ breakpoint.sourceText = sourceRow.getElementsByClassName('webkit-line-content')[0].textContent;
+
+ this._drawBreakpointImagesIfNeeded();
+
+ sourceRow._breakpointObject = breakpoint;
+
+ sourceRow.addStyleClass("webkit-breakpoint");
+ if (!breakpoint.enabled)
+ sourceRow.addStyleClass("webkit-breakpoint-disabled");
+ if (breakpoint.condition)
+ sourceRow.addStyleClass("webkit-breakpoint-conditional");
+ },
+
+ _removeBreakpointFromSource: function(breakpoint)
+ {
+ var sourceRow = this.sourceRow(breakpoint.line);
+ if (!sourceRow)
+ return;
+
+ delete sourceRow._breakpointObject;
+
+ sourceRow.removeStyleClass("webkit-breakpoint");
+ sourceRow.removeStyleClass("webkit-breakpoint-disabled");
+ sourceRow.removeStyleClass("webkit-breakpoint-conditional");
+ },
+
+ _incrementMessageRepeatCount: function(msg, repeatDelta)
+ {
+ if (!msg._resourceMessageLineElement)
+ return;
+
+ if (!msg._resourceMessageRepeatCountElement) {
+ var repeatedElement = document.createElement("span");
+ msg._resourceMessageLineElement.appendChild(repeatedElement);
+ msg._resourceMessageRepeatCountElement = repeatedElement;
+ }
+
+ msg.repeatCount += repeatDelta;
+ msg._resourceMessageRepeatCountElement.textContent = WebInspector.UIString(" (repeated %d times)", msg.repeatCount);
+ },
+
+ _addExistingMessagesToSource: function()
+ {
+ var length = this.messages.length;
+ for (var i = 0; i < length; ++i)
+ this._addMessageToSource(this.messages[i]);
+ },
+
+ _addMessageToSource: function(msg)
+ {
+ var row = this.sourceRow(msg.line);
+ if (!row)
+ return;
+
+ var cell = row.cells[1];
+ if (!cell)
+ return;
+
+ var messageBubbleElement = cell.lastChild;
+ if (!messageBubbleElement || messageBubbleElement.nodeType !== Node.ELEMENT_NODE || !messageBubbleElement.hasStyleClass("webkit-html-message-bubble")) {
+ messageBubbleElement = this.element.contentDocument.createElement("div");
+ messageBubbleElement.className = "webkit-html-message-bubble";
+ cell.appendChild(messageBubbleElement);
+ }
+
+ if (!row.messages)
+ row.messages = [];
+
+ for (var i = 0; i < row.messages.length; ++i) {
+ if (row.messages[i].isEqual(msg, true)) {
+ this._incrementMessageRepeatCount(row.messages[i], msg.repeatDelta);
+ return;
+ }
+ }
+
+ row.messages.push(msg);
+
+ var imageURL;
+ switch (msg.level) {
+ case WebInspector.ConsoleMessage.MessageLevel.Error:
+ messageBubbleElement.addStyleClass("webkit-html-error-message");
+ imageURL = "Images/errorIcon.png";
+ break;
+ case WebInspector.ConsoleMessage.MessageLevel.Warning:
+ messageBubbleElement.addStyleClass("webkit-html-warning-message");
+ imageURL = "Images/warningIcon.png";
+ break;
+ }
+
+ var messageLineElement = this.element.contentDocument.createElement("div");
+ messageLineElement.className = "webkit-html-message-line";
+ messageBubbleElement.appendChild(messageLineElement);
+
+ // Create the image element in the Inspector's document so we can use relative image URLs.
+ var image = document.createElement("img");
+ image.src = imageURL;
+ image.className = "webkit-html-message-icon";
+
+ // Adopt the image element since it wasn't created in element's contentDocument.
+ image = this.element.contentDocument.adoptNode(image);
+ messageLineElement.appendChild(image);
+ messageLineElement.appendChild(this.element.contentDocument.createTextNode(msg.message));
+
+ msg._resourceMessageLineElement = messageLineElement;
+ },
+
+ _drawProgramCounterInContext: function(ctx, glow)
+ {
+ if (glow)
+ ctx.save();
+
+ ctx.beginPath();
+ ctx.moveTo(17, 2);
+ ctx.lineTo(19, 2);
+ ctx.lineTo(19, 0);
+ ctx.lineTo(21, 0);
+ ctx.lineTo(26, 5.5);
+ ctx.lineTo(21, 11);
+ ctx.lineTo(19, 11);
+ ctx.lineTo(19, 9);
+ ctx.lineTo(17, 9);
+ ctx.closePath();
+ ctx.fillStyle = "rgb(142, 5, 4)";
+
+ if (glow) {
+ ctx.shadowBlur = 4;
+ ctx.shadowColor = "rgb(255, 255, 255)";
+ ctx.shadowOffsetX = -1;
+ ctx.shadowOffsetY = 0;
+ }
+
+ ctx.fill();
+ ctx.fill(); // Fill twice to get a good shadow and darker anti-aliased pixels.
+
+ if (glow)
+ ctx.restore();
+ },
+
+ _drawProgramCounterImageIfNeeded: function()
+ {
+ if (!this._needsProgramCounterImage || !this.element.contentDocument)
+ return;
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "program-counter", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ this._drawProgramCounterInContext(ctx, true);
+
+ delete this._needsProgramCounterImage;
+ },
+
+ _drawBreakpointImagesIfNeeded: function(conditional)
+ {
+ if (!this._needsBreakpointImages || !this.element.contentDocument)
+ return;
+
+ function drawBreakpoint(ctx, disabled, conditional)
+ {
+ ctx.beginPath();
+ ctx.moveTo(0, 2);
+ ctx.lineTo(2, 0);
+ ctx.lineTo(21, 0);
+ ctx.lineTo(26, 5.5);
+ ctx.lineTo(21, 11);
+ ctx.lineTo(2, 11);
+ ctx.lineTo(0, 9);
+ ctx.closePath();
+ ctx.fillStyle = conditional ? "rgb(217, 142, 1)" : "rgb(1, 142, 217)";
+ ctx.strokeStyle = conditional ? "rgb(205, 103, 0)" : "rgb(0, 103, 205)";
+ ctx.lineWidth = 3;
+ ctx.fill();
+ ctx.save();
+ ctx.clip();
+ ctx.stroke();
+ ctx.restore();
+
+ if (!disabled)
+ return;
+
+ ctx.save();
+ ctx.globalCompositeOperation = "destination-out";
+ ctx.fillStyle = "rgba(0, 0, 0, 0.5)";
+ ctx.fillRect(0, 0, 26, 11);
+ ctx.restore();
+ }
+
+
+ // Unconditional breakpoints.
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx);
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint-program-counter", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx);
+ ctx.clearRect(20, 0, 6, 11);
+ this._drawProgramCounterInContext(ctx, true);
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint-disabled", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx, true);
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint-disabled-program-counter", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx, true);
+ ctx.clearRect(20, 0, 6, 11);
+ this._drawProgramCounterInContext(ctx, true);
+
+
+ // Conditional breakpoints.
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint-conditional", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx, false, true);
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint-conditional-program-counter", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx, false, true);
+ ctx.clearRect(20, 0, 6, 11);
+ this._drawProgramCounterInContext(ctx, true);
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint-disabled-conditional", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx, true, true);
+
+ var ctx = this.element.contentDocument.getCSSCanvasContext("2d", "breakpoint-disabled-conditional-program-counter", 26, 11);
+ ctx.clearRect(0, 0, 26, 11);
+ drawBreakpoint(ctx, true, true);
+ ctx.clearRect(20, 0, 6, 11);
+ this._drawProgramCounterInContext(ctx, true);
+
+ delete this._needsBreakpointImages;
+ },
+
+ syntaxHighlightJavascript: function()
+ {
+ var table = this.element.contentDocument.getElementsByTagName("table")[0];
+ if (!table)
+ return;
+
+ function deleteContinueFlags(cell)
+ {
+ if (!cell)
+ return;
+ delete cell._commentContinues;
+ delete cell._singleQuoteStringContinues;
+ delete cell._doubleQuoteStringContinues;
+ delete cell._regexpContinues;
+ }
+
+ function createSpan(content, className)
+ {
+ var span = document.createElement("span");
+ span.className = className;
+ span.appendChild(document.createTextNode(content));
+ return span;
+ }
+
+ function generateFinder(regex, matchNumber, className)
+ {
+ return function(str) {
+ var match = regex.exec(str);
+ if (!match)
+ return null;
+ previousMatchLength = match[matchNumber].length;
+ return createSpan(match[matchNumber], className);
+ };
+ }
+
+ var findNumber = generateFinder(/^(-?(\d+\.?\d*([eE][+-]\d+)?|0[xX]\h+|Infinity)|NaN)(?:\W|$)/, 1, "webkit-javascript-number");
+ var findKeyword = generateFinder(/^(null|true|false|break|case|catch|const|default|finally|for|instanceof|new|var|continue|function|return|void|delete|if|this|do|while|else|in|switch|throw|try|typeof|with|debugger|class|enum|export|extends|import|super|get|set)(?:\W|$)/, 1, "webkit-javascript-keyword");
+ var findSingleLineString = generateFinder(/^"(?:[^"\\]|\\.)*"|^'([^'\\]|\\.)*'/, 0, "webkit-javascript-string"); // " this quote keeps Xcode happy
+ var findMultilineCommentStart = generateFinder(/^\/\*.*$/, 0, "webkit-javascript-comment");
+ var findMultilineCommentEnd = generateFinder(/^.*?\*\//, 0, "webkit-javascript-comment");
+ var findMultilineSingleQuoteStringStart = generateFinder(/^'(?:[^'\\]|\\.)*\\$/, 0, "webkit-javascript-string");
+ var findMultilineSingleQuoteStringEnd = generateFinder(/^(?:[^'\\]|\\.)*?'/, 0, "webkit-javascript-string");
+ var findMultilineDoubleQuoteStringStart = generateFinder(/^"(?:[^"\\]|\\.)*\\$/, 0, "webkit-javascript-string");
+ var findMultilineDoubleQuoteStringEnd = generateFinder(/^(?:[^"\\]|\\.)*?"/, 0, "webkit-javascript-string");
+ var findMultilineRegExpEnd = generateFinder(/^(?:[^\/\\]|\\.)*?\/([gim]{0,3})/, 0, "webkit-javascript-regexp");
+ var findSingleLineComment = generateFinder(/^\/\/.*|^\/\*.*?\*\//, 0, "webkit-javascript-comment");
+
+ function findMultilineRegExpStart(str)
+ {
+ var match = /^\/(?:[^\/\\]|\\.)*\\$/.exec(str);
+ if (!match || !/\\|\$|\.[\?\*\+]|[^\|]\|[^\|]/.test(match[0]))
+ return null;
+ var node = createSpan(match[0], "webkit-javascript-regexp");
+ previousMatchLength = match[0].length;
+ return node;
+ }
+
+ function findSingleLineRegExp(str)
+ {
+ var match = /^(\/(?:[^\/\\]|\\.)*\/([gim]{0,3}))(.?)/.exec(str);
+ if (!match || !(match[2].length > 0 || /\\|\$|\.[\?\*\+]|[^\|]\|[^\|]/.test(match[1]) || /\.|;|,/.test(match[3])))
+ return null;
+ var node = createSpan(match[1], "webkit-javascript-regexp");
+ previousMatchLength = match[1].length;
+ return node;
+ }
+
+ function syntaxHighlightJavascriptLine(line, prevLine)
+ {
+ var messageBubble = line.lastChild;
+ if (messageBubble && messageBubble.nodeType === Node.ELEMENT_NODE && messageBubble.hasStyleClass("webkit-html-message-bubble"))
+ line.removeChild(messageBubble);
+ else
+ messageBubble = null;
+
+ var code = line.textContent;
+
+ while (line.firstChild)
+ line.removeChild(line.firstChild);
+
+ var token;
+ var tmp = 0;
+ var i = 0;
+ previousMatchLength = 0;
+
+ if (prevLine) {
+ if (prevLine._commentContinues) {
+ if (!(token = findMultilineCommentEnd(code))) {
+ token = createSpan(code, "webkit-javascript-comment");
+ line._commentContinues = true;
+ }
+ } else if (prevLine._singleQuoteStringContinues) {
+ if (!(token = findMultilineSingleQuoteStringEnd(code))) {
+ token = createSpan(code, "webkit-javascript-string");
+ line._singleQuoteStringContinues = true;
+ }
+ } else if (prevLine._doubleQuoteStringContinues) {
+ if (!(token = findMultilineDoubleQuoteStringEnd(code))) {
+ token = createSpan(code, "webkit-javascript-string");
+ line._doubleQuoteStringContinues = true;
+ }
+ } else if (prevLine._regexpContinues) {
+ if (!(token = findMultilineRegExpEnd(code))) {
+ token = createSpan(code, "webkit-javascript-regexp");
+ line._regexpContinues = true;
+ }
+ }
+ if (token) {
+ i += previousMatchLength ? previousMatchLength : code.length;
+ tmp = i;
+ line.appendChild(token);
+ }
+ }
+
+ for ( ; i < code.length; ++i) {
+ var codeFragment = code.substr(i);
+ var prevChar = code[i - 1];
+ token = findSingleLineComment(codeFragment);
+ if (!token) {
+ if ((token = findMultilineCommentStart(codeFragment)))
+ line._commentContinues = true;
+ else if (!prevChar || /^\W/.test(prevChar)) {
+ token = findNumber(codeFragment, code[i - 1]) ||
+ findKeyword(codeFragment, code[i - 1]) ||
+ findSingleLineString(codeFragment) ||
+ findSingleLineRegExp(codeFragment);
+ if (!token) {
+ if (token = findMultilineSingleQuoteStringStart(codeFragment))
+ line._singleQuoteStringContinues = true;
+ else if (token = findMultilineDoubleQuoteStringStart(codeFragment))
+ line._doubleQuoteStringContinues = true;
+ else if (token = findMultilineRegExpStart(codeFragment))
+ line._regexpContinues = true;
+ }
+ }
+ }
+
+ if (token) {
+ if (tmp !== i)
+ line.appendChild(document.createTextNode(code.substring(tmp, i)));
+ line.appendChild(token);
+ i += previousMatchLength - 1;
+ tmp = i + 1;
+ }
+ }
+
+ if (tmp < code.length)
+ line.appendChild(document.createTextNode(code.substring(tmp, i)));
+
+ if (messageBubble)
+ line.appendChild(messageBubble);
+ }
+
+ var i = 0;
+ var rows = table.rows;
+ var rowsLength = rows.length;
+ var previousCell = null;
+ var previousMatchLength = 0;
+ var sourceFrame = this;
+
+ // Split up the work into chunks so we don't block the
+ // UI thread while processing.
+
+ function processChunk()
+ {
+ for (var end = Math.min(i + 10, rowsLength); i < end; ++i) {
+ var row = rows[i];
+ if (!row)
+ continue;
+ var cell = row.cells[1];
+ if (!cell)
+ continue;
+ syntaxHighlightJavascriptLine(cell, previousCell);
+ if (i < (end - 1))
+ deleteContinueFlags(previousCell);
+ previousCell = cell;
+ }
+
+ if (i >= rowsLength && processChunkInterval) {
+ deleteContinueFlags(previousCell);
+ clearInterval(processChunkInterval);
+
+ sourceFrame.dispatchEventToListeners("syntax highlighting complete");
+ }
+ }
+
+ processChunk();
+
+ var processChunkInterval = setInterval(processChunk, 25);
+ }
+}
+
+WebInspector.SourceFrame.prototype.__proto__ = WebInspector.Object.prototype;
+
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceView.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceView.js
new file mode 100644
index 0000000..97a5bd5
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SourceView.js
@@ -0,0 +1,311 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.SourceView = function(resource)
+{
+ // Set the sourceFrame first since WebInspector.ResourceView will set headersVisible
+ // and our override of headersVisible needs the sourceFrame.
+ this.sourceFrame = new WebInspector.SourceFrame(null, this._addBreakpoint.bind(this));
+
+ WebInspector.ResourceView.call(this, resource);
+
+ resource.addEventListener("finished", this._resourceLoadingFinished, this);
+
+ this.element.addStyleClass("source");
+
+ this._frameNeedsSetup = true;
+
+ this.contentElement.appendChild(this.sourceFrame.element);
+
+ var gutterElement = document.createElement("div");
+ gutterElement.className = "webkit-line-gutter-backdrop";
+ this.element.appendChild(gutterElement);
+}
+
+WebInspector.SourceView.prototype = {
+ set headersVisible(x)
+ {
+ if (x === this._headersVisible)
+ return;
+
+ var superSetter = WebInspector.ResourceView.prototype.__lookupSetter__("headersVisible");
+ if (superSetter)
+ superSetter.call(this, x);
+
+ this.sourceFrame.autoSizesToFitContentHeight = x;
+ },
+
+ show: function(parentElement)
+ {
+ WebInspector.ResourceView.prototype.show.call(this, parentElement);
+ this.setupSourceFrameIfNeeded();
+ },
+
+ hide: function()
+ {
+ WebInspector.View.prototype.hide.call(this);
+ this._currentSearchResultIndex = -1;
+ },
+
+ resize: function()
+ {
+ if (this.sourceFrame.autoSizesToFitContentHeight)
+ this.sourceFrame.sizeToFitContentHeight();
+ },
+
+ detach: function()
+ {
+ WebInspector.ResourceView.prototype.detach.call(this);
+
+ // FIXME: We need to mark the frame for setup on detach because the frame DOM is cleared
+ // when it is removed from the document. Is this a bug?
+ this._frameNeedsSetup = true;
+ this._sourceFrameSetup = false;
+ },
+
+ setupSourceFrameIfNeeded: function()
+ {
+ if (!this._frameNeedsSetup)
+ return;
+
+ this.attach();
+
+ delete this._frameNeedsSetup;
+ this.sourceFrame.addEventListener("content loaded", this._contentLoaded, this);
+ InspectorController.addResourceSourceToFrame(this.resource.identifier, this.sourceFrame.element);
+ },
+
+ _contentLoaded: function()
+ {
+ delete this._frameNeedsSetup;
+ this.sourceFrame.removeEventListener("content loaded", this._contentLoaded, this);
+
+ if (this.resource.type === WebInspector.Resource.Type.Script
+ || this.resource.mimeType === 'application/json'
+ || this.resource.mimeType === 'application/javascript'
+ || /\.js(on)?$/.test(this.resource.lastPathComponent) ) {
+ this.sourceFrame.addEventListener("syntax highlighting complete", this._syntaxHighlightingComplete, this);
+ this.sourceFrame.syntaxHighlightJavascript();
+ } else
+ this._sourceFrameSetupFinished();
+ },
+
+ _resourceLoadingFinished: function(event)
+ {
+ this._frameNeedsSetup = true;
+ this._sourceFrameSetup = false;
+ if (this.visible)
+ this.setupSourceFrameIfNeeded();
+ this.resource.removeEventListener("finished", this._resourceLoadingFinished, this);
+ },
+
+ _addBreakpoint: function(line)
+ {
+ var sourceID = null;
+ var closestStartingLine = 0;
+ var scripts = this.resource.scripts;
+ for (var i = 0; i < scripts.length; ++i) {
+ var script = scripts[i];
+ if (script.startingLine <= line && script.startingLine >= closestStartingLine) {
+ closestStartingLine = script.startingLine;
+ sourceID = script.sourceID;
+ }
+ }
+
+ if (WebInspector.panels.scripts) {
+ var breakpoint = new WebInspector.Breakpoint(this.resource.url, line, sourceID);
+ WebInspector.panels.scripts.addBreakpoint(breakpoint);
+ }
+ },
+
+ // The rest of the methods in this prototype need to be generic enough to work with a ScriptView.
+ // The ScriptView prototype pulls these methods into it's prototype to avoid duplicate code.
+
+ searchCanceled: function()
+ {
+ this._currentSearchResultIndex = -1;
+ this._searchResults = [];
+ delete this._delayedFindSearchMatches;
+ },
+
+ performSearch: function(query, finishedCallback)
+ {
+ // Call searchCanceled since it will reset everything we need before doing a new search.
+ this.searchCanceled();
+
+ var lineQueryRegex = /(^|\s)(?:#|line:\s*)(\d+)(\s|$)/i;
+ var lineQueryMatch = query.match(lineQueryRegex);
+ if (lineQueryMatch) {
+ var lineToSearch = parseInt(lineQueryMatch[2]);
+
+ // If there was a space before and after the line query part, replace with a space.
+ // Otherwise replace with an empty string to eat the prefix or postfix space.
+ var lineQueryReplacement = (lineQueryMatch[1] && lineQueryMatch[3] ? " " : "");
+ var filterlessQuery = query.replace(lineQueryRegex, lineQueryReplacement);
+ }
+
+ this._searchFinishedCallback = finishedCallback;
+
+ function findSearchMatches(query, finishedCallback)
+ {
+ if (isNaN(lineToSearch)) {
+ // Search the whole document since there was no line to search.
+ this._searchResults = (InspectorController.search(this.sourceFrame.element.contentDocument, query) || []);
+ } else {
+ var sourceRow = this.sourceFrame.sourceRow(lineToSearch);
+ if (sourceRow) {
+ if (filterlessQuery) {
+ // There is still a query string, so search for that string in the line.
+ this._searchResults = (InspectorController.search(sourceRow, filterlessQuery) || []);
+ } else {
+ // Match the whole line, since there was no remaining query string to match.
+ var rowRange = this.sourceFrame.element.contentDocument.createRange();
+ rowRange.selectNodeContents(sourceRow);
+ this._searchResults = [rowRange];
+ }
+ }
+
+ // Attempt to search for the whole query, just incase it matches a color like "#333".
+ var wholeQueryMatches = InspectorController.search(this.sourceFrame.element.contentDocument, query);
+ if (wholeQueryMatches)
+ this._searchResults = this._searchResults.concat(wholeQueryMatches);
+ }
+
+ if (this._searchResults)
+ finishedCallback(this, this._searchResults.length);
+ }
+
+ if (!this._sourceFrameSetup) {
+ // The search is performed in _sourceFrameSetupFinished by calling _delayedFindSearchMatches.
+ this._delayedFindSearchMatches = findSearchMatches.bind(this, query, finishedCallback);
+ this.setupSourceFrameIfNeeded();
+ return;
+ }
+
+ findSearchMatches.call(this, query, finishedCallback);
+ },
+
+ jumpToFirstSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ this._currentSearchResultIndex = 0;
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ jumpToLastSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ this._currentSearchResultIndex = (this._searchResults.length - 1);
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ jumpToNextSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ if (++this._currentSearchResultIndex >= this._searchResults.length)
+ this._currentSearchResultIndex = 0;
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ jumpToPreviousSearchResult: function()
+ {
+ if (!this._searchResults || !this._searchResults.length)
+ return;
+ if (--this._currentSearchResultIndex < 0)
+ this._currentSearchResultIndex = (this._searchResults.length - 1);
+ this._jumpToSearchResult(this._currentSearchResultIndex);
+ },
+
+ showingFirstSearchResult: function()
+ {
+ return (this._currentSearchResultIndex === 0);
+ },
+
+ showingLastSearchResult: function()
+ {
+ return (this._searchResults && this._currentSearchResultIndex === (this._searchResults.length - 1));
+ },
+
+ revealLine: function(lineNumber)
+ {
+ this.setupSourceFrameIfNeeded();
+ this.sourceFrame.revealLine(lineNumber);
+ },
+
+ highlightLine: function(lineNumber)
+ {
+ this.setupSourceFrameIfNeeded();
+ this.sourceFrame.highlightLine(lineNumber);
+ },
+
+ addMessage: function(msg)
+ {
+ this.sourceFrame.addMessage(msg);
+ },
+
+ clearMessages: function()
+ {
+ this.sourceFrame.clearMessages();
+ },
+
+ _jumpToSearchResult: function(index)
+ {
+ var foundRange = this._searchResults[index];
+ if (!foundRange)
+ return;
+
+ var selection = this.sourceFrame.element.contentWindow.getSelection();
+ selection.removeAllRanges();
+ selection.addRange(foundRange);
+
+ if (foundRange.startContainer.scrollIntoViewIfNeeded)
+ foundRange.startContainer.scrollIntoViewIfNeeded(true);
+ else if (foundRange.startContainer.parentNode)
+ foundRange.startContainer.parentNode.scrollIntoViewIfNeeded(true);
+ },
+
+ _sourceFrameSetupFinished: function()
+ {
+ this._sourceFrameSetup = true;
+ if (this._delayedFindSearchMatches) {
+ this._delayedFindSearchMatches();
+ delete this._delayedFindSearchMatches;
+ }
+ },
+
+ _syntaxHighlightingComplete: function(event)
+ {
+ this._sourceFrameSetupFinished();
+ this.sourceFrame.removeEventListener("syntax highlighting complete", null, this);
+ }
+}
+
+WebInspector.SourceView.prototype.__proto__ = WebInspector.ResourceView.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StatusBarButton.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StatusBarButton.js
new file mode 100644
index 0000000..5c69ed5
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StatusBarButton.js
@@ -0,0 +1,118 @@
+/*
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.StatusBarButton = function(title, className)
+{
+ this.element = document.createElement("button");
+ this.element.className = className + " status-bar-item";
+ this.element.addEventListener("click", this._clicked.bind(this), false);
+
+ this.glyph = document.createElement("div");
+ this.glyph.className = "glyph";
+ this.element.appendChild(this.glyph);
+
+ this.glyphShadow = document.createElement("div");
+ this.glyphShadow.className = "glyph shadow";
+ this.element.appendChild(this.glyphShadow);
+
+ this.title = title;
+ this.disabled = false;
+ this._toggled = false;
+ this._visible = true;
+}
+
+WebInspector.StatusBarButton.prototype = {
+ _clicked: function()
+ {
+ this.dispatchEventToListeners("click");
+ },
+
+ get disabled()
+ {
+ return this._disabled;
+ },
+
+ set disabled(x)
+ {
+ if (this._disabled === x)
+ return;
+ this._disabled = x;
+ this.element.disabled = x;
+ },
+
+ get title()
+ {
+ return this._title;
+ },
+
+ set title(x)
+ {
+ if (this._title === x)
+ return;
+ this._title = x;
+ this.element.title = x;
+ },
+
+ get toggled()
+ {
+ return this._toggled;
+ },
+
+ set toggled(x)
+ {
+ if (this._toggled === x)
+ return;
+
+ if (x)
+ this.element.addStyleClass("toggled-on");
+ else
+ this.element.removeStyleClass("toggled-on");
+ this._toggled = x;
+ },
+
+ get visible()
+ {
+ return this._visible;
+ },
+
+ set visible(x)
+ {
+ if (this._visible === x)
+ return;
+
+ if (x)
+ this.element.removeStyleClass("hidden");
+ else
+ this.element.addStyleClass("hidden");
+ this._visible = x;
+ }
+}
+
+WebInspector.StatusBarButton.prototype.__proto__ = WebInspector.Object.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StoragePanel.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StoragePanel.js
new file mode 100644
index 0000000..aed0d06
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StoragePanel.js
@@ -0,0 +1,685 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.StoragePanel = function(database)
+{
+ WebInspector.Panel.call(this);
+
+ this.sidebarElement = document.createElement("div");
+ this.sidebarElement.id = "storage-sidebar";
+ this.sidebarElement.className = "sidebar";
+ this.element.appendChild(this.sidebarElement);
+
+ this.sidebarResizeElement = document.createElement("div");
+ this.sidebarResizeElement.className = "sidebar-resizer-vertical";
+ this.sidebarResizeElement.addEventListener("mousedown", this._startSidebarDragging.bind(this), false);
+ this.element.appendChild(this.sidebarResizeElement);
+
+ this.sidebarTreeElement = document.createElement("ol");
+ this.sidebarTreeElement.className = "sidebar-tree";
+ this.sidebarElement.appendChild(this.sidebarTreeElement);
+
+ this.sidebarTree = new TreeOutline(this.sidebarTreeElement);
+
+ this.databasesListTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("DATABASES"), {}, true);
+ this.sidebarTree.appendChild(this.databasesListTreeElement);
+ this.databasesListTreeElement.expand();
+
+ this.localStorageListTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("LOCAL STORAGE"), {}, true);
+ this.sidebarTree.appendChild(this.localStorageListTreeElement);
+ this.localStorageListTreeElement.expand();
+
+ this.sessionStorageListTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("SESSION STORAGE"), {}, true);
+ this.sidebarTree.appendChild(this.sessionStorageListTreeElement);
+ this.sessionStorageListTreeElement.expand();
+
+ this.cookieListTreeElement = new WebInspector.SidebarSectionTreeElement(WebInspector.UIString("COOKIES"), {}, true);
+ this.sidebarTree.appendChild(this.cookieListTreeElement);
+ this.cookieListTreeElement.expand();
+
+ this.cookieTreeElement = new WebInspector.CookieSidebarTreeElement();
+ this.cookieListTreeElement.appendChild(this.cookieTreeElement);
+
+ this.storageViews = document.createElement("div");
+ this.storageViews.id = "storage-views";
+ this.element.appendChild(this.storageViews);
+
+ this.storageViewStatusBarItemsContainer = document.createElement("div");
+ this.storageViewStatusBarItemsContainer.id = "storage-view-status-bar-items";
+
+ this.reset();
+}
+
+WebInspector.StoragePanel.prototype = {
+ toolbarItemClass: "storage",
+
+ get toolbarItemLabel()
+ {
+ return WebInspector.UIString("Storage");
+ },
+
+ get statusBarItems()
+ {
+ return [this.storageViewStatusBarItemsContainer];
+ },
+
+ show: function()
+ {
+ WebInspector.Panel.prototype.show.call(this);
+ this._updateSidebarWidth();
+ this._registerStorageEventListener();
+ },
+
+ reset: function()
+ {
+ if (this._databases) {
+ var databasesLength = this._databases.length;
+ for (var i = 0; i < databasesLength; ++i) {
+ var database = this._databases[i];
+
+ delete database._tableViews;
+ delete database._queryView;
+ }
+ }
+
+ this._databases = [];
+
+ this._unregisterStorageEventListener();
+
+ if (this._domStorage) {
+ var domStorageLength = this._domStorage.length;
+ for (var i = 0; i < domStorageLength; ++i) {
+ var domStorage = this._domStorage[i];
+
+ delete domStorage._domStorageView;
+ }
+ }
+
+ this._domStorage = [];
+
+ delete this._cookieView;
+
+ this.databasesListTreeElement.removeChildren();
+ this.localStorageListTreeElement.removeChildren();
+ this.sessionStorageListTreeElement.removeChildren();
+ this.storageViews.removeChildren();
+
+ this.storageViewStatusBarItemsContainer.removeChildren();
+
+ if (this.sidebarTree.selectedTreeElement)
+ this.sidebarTree.selectedTreeElement.deselect();
+ },
+
+ handleKeyEvent: function(event)
+ {
+ this.sidebarTree.handleKeyEvent(event);
+ },
+
+ addDatabase: function(database)
+ {
+ this._databases.push(database);
+
+ var databaseTreeElement = new WebInspector.DatabaseSidebarTreeElement(database);
+ database._databasesTreeElement = databaseTreeElement;
+ this.databasesListTreeElement.appendChild(databaseTreeElement);
+ },
+
+ addDOMStorage: function(domStorage)
+ {
+ this._domStorage.push(domStorage);
+ var domStorageTreeElement = new WebInspector.DOMStorageSidebarTreeElement(domStorage, (domStorage.isLocalStorage ? "local-storage" : "session-storage"));
+ domStorage._domStorageTreeElement = domStorageTreeElement;
+ if (domStorage.isLocalStorage)
+ this.localStorageListTreeElement.appendChild(domStorageTreeElement);
+ else
+ this.sessionStorageListTreeElement.appendChild(domStorageTreeElement);
+ },
+
+ selectDatabase: function(db)
+ {
+ var database;
+ for (var i = 0, len = this._databases.length; i < len; ++i) {
+ database = this._databases[i];
+ if (database.isDatabase(db)) {
+ this.showDatabase(database);
+ database._databasesTreeElement.select();
+ return;
+ }
+ }
+ },
+
+ selectDOMStorage: function(s)
+ {
+ var isLocalStorage = (s === InspectorController.inspectedWindow().localStorage);
+ for (var i = 0, len = this._domStorage.length; i < len; ++i) {
+ var storage = this._domStorage[i];
+ if ( isLocalStorage === storage.isLocalStorage ) {
+ this.showDOMStorage(storage);
+ storage._domStorageTreeElement.select();
+ return;
+ }
+ }
+ },
+
+ showDatabase: function(database, tableName)
+ {
+ if (!database)
+ return;
+
+ if (this.visibleView)
+ this.visibleView.hide();
+
+ var view;
+ if (tableName) {
+ if (!("_tableViews" in database))
+ database._tableViews = {};
+ view = database._tableViews[tableName];
+ if (!view) {
+ view = new WebInspector.DatabaseTableView(database, tableName);
+ database._tableViews[tableName] = view;
+ }
+ } else {
+ view = database._queryView;
+ if (!view) {
+ view = new WebInspector.DatabaseQueryView(database);
+ database._queryView = view;
+ }
+ }
+
+ view.show(this.storageViews);
+
+ this.visibleView = view;
+
+ this.storageViewStatusBarItemsContainer.removeChildren();
+ var statusBarItems = view.statusBarItems || [];
+ for (var i = 0; i < statusBarItems.length; ++i)
+ this.storageViewStatusBarItemsContainer.appendChild(statusBarItems[i].element);
+ },
+
+ showDOMStorage: function(domStorage)
+ {
+ if (!domStorage)
+ return;
+
+ if (this.visibleView)
+ this.visibleView.hide();
+
+ var view;
+ view = domStorage._domStorageView;
+ if (!view) {
+ view = new WebInspector.DOMStorageItemsView(domStorage);
+ domStorage._domStorageView = view;
+ }
+
+ view.show(this.storageViews);
+
+ this.visibleView = view;
+
+ this.storageViewStatusBarItemsContainer.removeChildren();
+ var statusBarItems = view.statusBarItems;
+ for (var i = 0; i < statusBarItems.length; ++i)
+ this.storageViewStatusBarItemsContainer.appendChild(statusBarItems[i]);
+ },
+
+ showCookies: function()
+ {
+ if (this.visibleView)
+ this.visibleView.hide();
+
+ var view = this._cookieView;
+ if (!view) {
+ view = new WebInspector.CookieItemsView();
+ this._cookieView = view;
+ }
+
+ view.show(this.storageViews);
+
+ this.visibleView = view;
+
+ this.storageViewStatusBarItemsContainer.removeChildren();
+ var statusBarItems = view.statusBarItems;
+ for (var i = 0; i < statusBarItems.length; ++i)
+ this.storageViewStatusBarItemsContainer.appendChild(statusBarItems[i]);
+ },
+
+ closeVisibleView: function()
+ {
+ if (this.visibleView)
+ this.visibleView.hide();
+ delete this.visibleView;
+ },
+
+ updateDatabaseTables: function(database)
+ {
+ if (!database || !database._databasesTreeElement)
+ return;
+
+ database._databasesTreeElement.shouldRefreshChildren = true;
+
+ if (!("_tableViews" in database))
+ return;
+
+ var tableNamesHash = {};
+ var self = this;
+ function tableNamesCallback(tableNames)
+ {
+ var tableNamesLength = tableNames.length;
+ for (var i = 0; i < tableNamesLength; ++i)
+ tableNamesHash[tableNames[i]] = true;
+
+ for (var tableName in database._tableViews) {
+ if (!(tableName in tableNamesHash)) {
+ if (self.visibleView === database._tableViews[tableName])
+ self.closeVisibleView();
+ delete database._tableViews[tableName];
+ }
+ }
+ }
+ database.getTableNames(tableNamesCallback);
+ },
+
+ dataGridForResult: function(result)
+ {
+ if (!result.rows.length)
+ return null;
+
+ var columns = {};
+
+ var rows = result.rows;
+ for (var columnIdentifier in rows.item(0)) {
+ var column = {};
+ column.width = columnIdentifier.length;
+ column.title = columnIdentifier;
+
+ columns[columnIdentifier] = column;
+ }
+
+ var nodes = [];
+ var length = rows.length;
+ for (var i = 0; i < length; ++i) {
+ var data = {};
+
+ var row = rows.item(i);
+ for (var columnIdentifier in row) {
+ // FIXME: (Bug 19439) We should specially format SQL NULL here
+ // (which is represented by JavaScript null here, and turned
+ // into the string "null" by the String() function).
+ var text = String(row[columnIdentifier]);
+ data[columnIdentifier] = text;
+ if (text.length > columns[columnIdentifier].width)
+ columns[columnIdentifier].width = text.length;
+ }
+
+ var node = new WebInspector.DataGridNode(data, false);
+ node.selectable = false;
+ nodes.push(node);
+ }
+
+ var totalColumnWidths = 0;
+ for (var columnIdentifier in columns)
+ totalColumnWidths += columns[columnIdentifier].width;
+
+ // Calculate the percentage width for the columns.
+ const minimumPrecent = 5;
+ var recoupPercent = 0;
+ for (var columnIdentifier in columns) {
+ var width = columns[columnIdentifier].width;
+ width = Math.round((width / totalColumnWidths) * 100);
+ if (width < minimumPrecent) {
+ recoupPercent += (minimumPrecent - width);
+ width = minimumPrecent;
+ }
+
+ columns[columnIdentifier].width = width;
+ }
+
+ // Enforce the minimum percentage width.
+ while (recoupPercent > 0) {
+ for (var columnIdentifier in columns) {
+ if (columns[columnIdentifier].width > minimumPrecent) {
+ --columns[columnIdentifier].width;
+ --recoupPercent;
+ if (!recoupPercent)
+ break;
+ }
+ }
+ }
+
+ // Change the width property to a string suitable for a style width.
+ for (var columnIdentifier in columns)
+ columns[columnIdentifier].width += "%";
+
+ var dataGrid = new WebInspector.DataGrid(columns);
+ var length = nodes.length;
+ for (var i = 0; i < length; ++i)
+ dataGrid.appendChild(nodes[i]);
+
+ return dataGrid;
+ },
+
+ dataGridForDOMStorage: function(domStorage)
+ {
+ if (!domStorage.length)
+ return null;
+
+ var columns = {};
+ columns[0] = {};
+ columns[1] = {};
+ columns[0].title = WebInspector.UIString("Key");
+ columns[0].width = columns[0].title.length;
+ columns[1].title = WebInspector.UIString("Value");
+ columns[1].width = columns[1].title.length;
+
+ var nodes = [];
+
+ var length = domStorage.length;
+ for (var index = 0; index < domStorage.length; index++) {
+ var data = {};
+
+ var key = String(domStorage.key(index));
+ data[0] = key;
+ if (key.length > columns[0].width)
+ columns[0].width = key.length;
+
+ var value = String(domStorage.getItem(key));
+ data[1] = value;
+ if (value.length > columns[1].width)
+ columns[1].width = value.length;
+ var node = new WebInspector.DataGridNode(data, false);
+ node.selectable = true;
+ nodes.push(node);
+ }
+
+ var totalColumnWidths = columns[0].width + columns[1].width;
+ var width = Math.round((columns[0].width * 100) / totalColumnWidths);
+ const minimumPrecent = 10;
+ if (width < minimumPrecent)
+ width = minimumPrecent;
+ if (width > 100 - minimumPrecent)
+ width = 100 - minimumPrecent;
+ columns[0].width = width;
+ columns[1].width = 100 - width;
+ columns[0].width += "%";
+ columns[1].width += "%";
+
+ var dataGrid = new WebInspector.DOMStorageDataGrid(columns);
+ var length = nodes.length;
+ for (var i = 0; i < length; ++i)
+ dataGrid.appendChild(nodes[i]);
+ dataGrid.addCreationNode(false);
+ if (length > 0)
+ nodes[0].selected = true;
+ return dataGrid;
+ },
+
+ resize: function()
+ {
+ var visibleView = this.visibleView;
+ if (visibleView && "resize" in visibleView)
+ visibleView.resize();
+ },
+
+ _registerStorageEventListener: function()
+ {
+ var inspectedWindow = InspectorController.inspectedWindow();
+ if (!inspectedWindow || !inspectedWindow.document)
+ return;
+
+ this._storageEventListener = InspectorController.wrapCallback(this._storageEvent.bind(this));
+ inspectedWindow.addEventListener("storage", this._storageEventListener, true);
+ },
+
+ _unregisterStorageEventListener: function()
+ {
+ if (!this._storageEventListener)
+ return;
+
+ var inspectedWindow = InspectorController.inspectedWindow();
+ if (!inspectedWindow || !inspectedWindow.document)
+ return;
+
+ inspectedWindow.removeEventListener("storage", this._storageEventListener, true);
+ delete this._storageEventListener;
+ },
+
+ _storageEvent: function(event)
+ {
+ if (!this._domStorage)
+ return;
+
+ var isLocalStorage = (event.storageArea === InspectorController.inspectedWindow().localStorage);
+ var domStorageLength = this._domStorage.length;
+ for (var i = 0; i < domStorageLength; ++i) {
+ var domStorage = this._domStorage[i];
+ if (isLocalStorage === domStorage.isLocalStorage) {
+ var view = domStorage._domStorageView;
+ if (this.visibleView && view === this.visibleView)
+ domStorage._domStorageView.update();
+ }
+ }
+ },
+
+ _startSidebarDragging: function(event)
+ {
+ WebInspector.elementDragStart(this.sidebarResizeElement, this._sidebarDragging.bind(this), this._endSidebarDragging.bind(this), event, "col-resize");
+ },
+
+ _sidebarDragging: function(event)
+ {
+ this._updateSidebarWidth(event.pageX);
+
+ event.preventDefault();
+ },
+
+ _endSidebarDragging: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+ },
+
+ _updateSidebarWidth: function(width)
+ {
+ if (this.sidebarElement.offsetWidth <= 0) {
+ // The stylesheet hasn't loaded yet or the window is closed,
+ // so we can't calculate what is need. Return early.
+ return;
+ }
+
+ if (!("_currentSidebarWidth" in this))
+ this._currentSidebarWidth = this.sidebarElement.offsetWidth;
+
+ if (typeof width === "undefined")
+ width = this._currentSidebarWidth;
+
+ width = Number.constrain(width, Preferences.minSidebarWidth, window.innerWidth / 2);
+
+ this._currentSidebarWidth = width;
+
+ this.sidebarElement.style.width = width + "px";
+ this.storageViews.style.left = width + "px";
+ this.storageViewStatusBarItemsContainer.style.left = width + "px";
+ this.sidebarResizeElement.style.left = (width - 3) + "px";
+
+ var visibleView = this.visibleView;
+ if (visibleView && "resize" in visibleView)
+ visibleView.resize();
+ }
+}
+
+WebInspector.StoragePanel.prototype.__proto__ = WebInspector.Panel.prototype;
+
+WebInspector.DatabaseSidebarTreeElement = function(database)
+{
+ this.database = database;
+
+ WebInspector.SidebarTreeElement.call(this, "database-sidebar-tree-item", "", "", database, true);
+
+ this.refreshTitles();
+}
+
+WebInspector.DatabaseSidebarTreeElement.prototype = {
+ onselect: function()
+ {
+ WebInspector.panels.storage.showDatabase(this.database);
+ },
+
+ oncollapse: function()
+ {
+ // Request a refresh after every collapse so the next
+ // expand will have an updated table list.
+ this.shouldRefreshChildren = true;
+ },
+
+ onpopulate: function()
+ {
+ this.removeChildren();
+
+ var self = this;
+ function tableNamesCallback(tableNames)
+ {
+ var tableNamesLength = tableNames.length;
+ for (var i = 0; i < tableNamesLength; ++i)
+ self.appendChild(new WebInspector.SidebarDatabaseTableTreeElement(self.database, tableNames[i]));
+ }
+ this.database.getTableNames(tableNamesCallback);
+ },
+
+ get mainTitle()
+ {
+ return this.database.name;
+ },
+
+ set mainTitle(x)
+ {
+ // Do nothing.
+ },
+
+ get subtitle()
+ {
+ return this.database.displayDomain;
+ },
+
+ set subtitle(x)
+ {
+ // Do nothing.
+ }
+}
+
+WebInspector.DatabaseSidebarTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
+
+WebInspector.SidebarDatabaseTableTreeElement = function(database, tableName)
+{
+ this.database = database;
+ this.tableName = tableName;
+
+ WebInspector.SidebarTreeElement.call(this, "database-table-sidebar-tree-item small", tableName, "", null, false);
+}
+
+WebInspector.SidebarDatabaseTableTreeElement.prototype = {
+ onselect: function()
+ {
+ WebInspector.panels.storage.showDatabase(this.database, this.tableName);
+ }
+}
+
+WebInspector.SidebarDatabaseTableTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
+
+WebInspector.DOMStorageSidebarTreeElement = function(domStorage, className)
+{
+
+ this.domStorage = domStorage;
+
+ WebInspector.SidebarTreeElement.call(this, "domstorage-sidebar-tree-item " + className, domStorage, "", null, false);
+
+ this.refreshTitles();
+}
+
+WebInspector.DOMStorageSidebarTreeElement.prototype = {
+ onselect: function()
+ {
+ WebInspector.panels.storage.showDOMStorage(this.domStorage);
+ },
+
+ get mainTitle()
+ {
+ return this.domStorage.domain;
+ },
+
+ set mainTitle(x)
+ {
+ // Do nothing.
+ },
+
+ get subtitle()
+ {
+ return ""; //this.database.displayDomain;
+ },
+
+ set subtitle(x)
+ {
+ // Do nothing.
+ }
+}
+
+WebInspector.DOMStorageSidebarTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
+
+WebInspector.CookieSidebarTreeElement = function()
+{
+ WebInspector.SidebarTreeElement.call(this, "cookie-sidebar-tree-item", null, "", null, false);
+
+ this.refreshTitles();
+}
+
+WebInspector.CookieSidebarTreeElement.prototype = {
+ onselect: function()
+ {
+ WebInspector.panels.storage.showCookies();
+ },
+
+ get mainTitle()
+ {
+ return WebInspector.UIString("Cookies");
+ },
+
+ set mainTitle(x)
+ {
+ // Do nothing.
+ },
+
+ get subtitle()
+ {
+ return "";
+ },
+
+ set subtitle(x)
+ {
+ // Do nothing.
+ }
+}
+
+WebInspector.CookieSidebarTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StylesSidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StylesSidebarPane.js
new file mode 100644
index 0000000..6185aff
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/StylesSidebarPane.js
@@ -0,0 +1,1373 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.StylesSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Styles"));
+
+ this.settingsSelectElement = document.createElement("select");
+
+ var option = document.createElement("option");
+ option.value = "hex";
+ option.action = this._changeColorFormat.bind(this);
+ if (Preferences.colorFormat === "hex")
+ option.selected = true;
+ option.label = WebInspector.UIString("Hex Colors");
+ this.settingsSelectElement.appendChild(option);
+
+ option = document.createElement("option");
+ option.value = "rgb";
+ option.action = this._changeColorFormat.bind(this);
+ if (Preferences.colorFormat === "rgb")
+ option.selected = true;
+ option.label = WebInspector.UIString("RGB Colors");
+ this.settingsSelectElement.appendChild(option);
+
+ option = document.createElement("option");
+ option.value = "hsl";
+ option.action = this._changeColorFormat.bind(this);
+ if (Preferences.colorFormat === "hsl")
+ option.selected = true;
+ option.label = WebInspector.UIString("HSL Colors");
+ this.settingsSelectElement.appendChild(option);
+
+ this.settingsSelectElement.appendChild(document.createElement("hr"));
+
+ option = document.createElement("option");
+ option.action = this._createNewRule.bind(this);
+ option.label = WebInspector.UIString("New Style Rule");
+ this.settingsSelectElement.appendChild(option);
+
+ this.settingsSelectElement.addEventListener("click", function(event) { event.stopPropagation() }, false);
+ this.settingsSelectElement.addEventListener("change", this._changeSetting.bind(this), false);
+
+ this.titleElement.appendChild(this.settingsSelectElement);
+}
+
+WebInspector.StylesSidebarPane.prototype = {
+ update: function(node, editedSection, forceUpdate)
+ {
+ var refresh = false;
+
+ if (forceUpdate)
+ delete this.node;
+
+ if (!forceUpdate && (!node || node === this.node))
+ refresh = true;
+
+ if (node && node.nodeType === Node.TEXT_NODE && node.parentNode)
+ node = node.parentNode;
+
+ if (node && node.nodeType !== Node.ELEMENT_NODE)
+ node = null;
+
+ if (node)
+ this.node = node;
+ else
+ node = this.node;
+
+ var body = this.bodyElement;
+ if (!refresh || !node) {
+ body.removeChildren();
+ this.sections = [];
+ }
+
+ if (!node)
+ return;
+
+ var self = this;
+ function callback(styles)
+ {
+ if (!styles)
+ return;
+ node._setStyles(styles.computedStyle, styles.inlineStyle, styles.styleAttributes, styles.matchedCSSRules);
+ self._update(refresh, body, node, editedSection, forceUpdate);
+ }
+
+ InjectedScriptAccess.getStyles(node.id, !Preferences.showUserAgentStyles, callback);
+ },
+
+ _update: function(refresh, body, node, editedSection, forceUpdate)
+ {
+ if (!refresh) {
+ body.removeChildren();
+ this.sections = [];
+ }
+
+ var styleRules = [];
+
+ if (refresh) {
+ for (var i = 0; i < this.sections.length; ++i) {
+ var section = this.sections[i];
+ if (section instanceof WebInspector.BlankStylePropertiesSection)
+ continue;
+ if (section.computedStyle)
+ section.styleRule.style = node.ownerDocument.defaultView.getComputedStyle(node);
+ var styleRule = { section: section, style: section.styleRule.style, computedStyle: section.computedStyle, rule: section.rule };
+ styleRules.push(styleRule);
+ }
+ } else {
+ var computedStyle = node.ownerDocument.defaultView.getComputedStyle(node);
+ styleRules.push({ computedStyle: true, selectorText: WebInspector.UIString("Computed Style"), style: computedStyle, editable: false });
+
+ var nodeName = node.nodeName.toLowerCase();
+ for (var i = 0; i < node.attributes.length; ++i) {
+ var attr = node.attributes[i];
+ if (attr.style) {
+ var attrStyle = { style: attr.style, editable: false };
+ attrStyle.subtitle = WebInspector.UIString("element’s “%s†attribute", attr.name);
+ attrStyle.selectorText = nodeName + "[" + attr.name;
+ if (attr.value.length)
+ attrStyle.selectorText += "=" + attr.value;
+ attrStyle.selectorText += "]";
+ styleRules.push(attrStyle);
+ }
+ }
+
+ // Always Show element's Style Attributes
+ if (node.nodeType === Node.ELEMENT_NODE) {
+ var inlineStyle = { selectorText: WebInspector.UIString("Style Attribute"), style: node.style, isAttribute: true };
+ inlineStyle.subtitle = WebInspector.UIString("element’s “%s†attribute", "style");
+ styleRules.push(inlineStyle);
+ }
+
+ var matchedStyleRules = node.ownerDocument.defaultView.getMatchedCSSRules(node, "", !Preferences.showUserAgentStyles);
+ if (matchedStyleRules) {
+ // Add rules in reverse order to match the cascade order.
+ for (var i = (matchedStyleRules.length - 1); i >= 0; --i) {
+ var rule = matchedStyleRules[i];
+ styleRules.push({ style: rule.style, selectorText: rule.selectorText, parentStyleSheet: rule.parentStyleSheet, rule: rule });
+ }
+ }
+ }
+
+ function deleteDisabledProperty(style, name)
+ {
+ if (!style || !name)
+ return;
+ if (style.__disabledPropertyValues)
+ delete style.__disabledPropertyValues[name];
+ if (style.__disabledPropertyPriorities)
+ delete style.__disabledPropertyPriorities[name];
+ if (style.__disabledProperties)
+ delete style.__disabledProperties[name];
+ }
+
+ var usedProperties = {};
+ var disabledComputedProperties = {};
+ var priorityUsed = false;
+
+ // Walk the style rules and make a list of all used and overloaded properties.
+ for (var i = 0; i < styleRules.length; ++i) {
+ var styleRule = styleRules[i];
+ if (styleRule.computedStyle)
+ continue;
+ if (styleRule.section && styleRule.section.noAffect)
+ continue;
+
+ styleRule.usedProperties = {};
+
+ var style = styleRule.style;
+ for (var j = 0; j < style.length; ++j) {
+ var name = style[j];
+
+ if (!priorityUsed && style.getPropertyPriority(name).length)
+ priorityUsed = true;
+
+ // If the property name is already used by another rule then this rule's
+ // property is overloaded, so don't add it to the rule's usedProperties.
+ if (!(name in usedProperties))
+ styleRule.usedProperties[name] = true;
+
+ if (name === "font") {
+ // The font property is not reported as a shorthand. Report finding the individual
+ // properties so they are visible in computed style.
+ // FIXME: remove this when http://bugs.webkit.org/show_bug.cgi?id=15598 is fixed.
+ styleRule.usedProperties["font-family"] = true;
+ styleRule.usedProperties["font-size"] = true;
+ styleRule.usedProperties["font-style"] = true;
+ styleRule.usedProperties["font-variant"] = true;
+ styleRule.usedProperties["font-weight"] = true;
+ styleRule.usedProperties["line-height"] = true;
+ }
+
+ // Delete any disabled properties, since the property does exist.
+ // This prevents it from showing twice.
+ deleteDisabledProperty(style, name);
+ deleteDisabledProperty(style, style.getPropertyShorthand(name));
+ }
+
+ // Add all the properties found in this style to the used properties list.
+ // Do this here so only future rules are affect by properties used in this rule.
+ for (var name in styleRules[i].usedProperties)
+ usedProperties[name] = true;
+
+ // Remember all disabled properties so they show up in computed style.
+ if (style.__disabledProperties)
+ for (var name in style.__disabledProperties)
+ disabledComputedProperties[name] = true;
+ }
+
+ if (priorityUsed) {
+ // Walk the properties again and account for !important.
+ var foundPriorityProperties = [];
+
+ // Walk in reverse to match the order !important overrides.
+ for (var i = (styleRules.length - 1); i >= 0; --i) {
+ if (styleRules[i].computedStyle)
+ continue;
+
+ var style = styleRules[i].style;
+ var uniqueProperties = style.uniqueStyleProperties;
+ for (var j = 0; j < uniqueProperties.length; ++j) {
+ var name = uniqueProperties[j];
+ if (style.getPropertyPriority(name).length) {
+ if (!(name in foundPriorityProperties))
+ styleRules[i].usedProperties[name] = true;
+ else
+ delete styleRules[i].usedProperties[name];
+ foundPriorityProperties[name] = true;
+ } else if (name in foundPriorityProperties)
+ delete styleRules[i].usedProperties[name];
+ }
+ }
+ }
+
+ if (refresh) {
+ // Walk the style rules and update the sections with new overloaded and used properties.
+ for (var i = 0; i < styleRules.length; ++i) {
+ var styleRule = styleRules[i];
+ var section = styleRule.section;
+ if (styleRule.computedStyle)
+ section.disabledComputedProperties = disabledComputedProperties;
+ section._usedProperties = (styleRule.usedProperties || usedProperties);
+ section.update((section === editedSection) || styleRule.computedStyle);
+ }
+ } else {
+ // Make a property section for each style rule.
+ for (var i = 0; i < styleRules.length; ++i) {
+ var styleRule = styleRules[i];
+ var subtitle = styleRule.subtitle;
+ delete styleRule.subtitle;
+
+ var computedStyle = styleRule.computedStyle;
+ delete styleRule.computedStyle;
+
+ var ruleUsedProperties = styleRule.usedProperties;
+ delete styleRule.usedProperties;
+
+ var editable = styleRule.editable;
+ delete styleRule.editable;
+
+ var isAttribute = styleRule.isAttribute;
+ delete styleRule.isAttribute;
+
+ // Default editable to true if it was omitted.
+ if (typeof editable === "undefined")
+ editable = true;
+
+ var section = new WebInspector.StylePropertiesSection(styleRule, subtitle, computedStyle, (ruleUsedProperties || usedProperties), editable);
+ if (computedStyle)
+ section.disabledComputedProperties = disabledComputedProperties;
+ section.pane = this;
+
+ if (Preferences.styleRulesExpandedState && section.identifier in Preferences.styleRulesExpandedState)
+ section.expanded = Preferences.styleRulesExpandedState[section.identifier];
+ else if (computedStyle)
+ section.collapse(true);
+ else if (isAttribute && styleRule.style.length === 0)
+ section.collapse(true);
+ else
+ section.expand(true);
+
+ body.appendChild(section.element);
+ this.sections.push(section);
+ }
+ }
+ },
+
+ _changeSetting: function(event)
+ {
+ var options = this.settingsSelectElement.options;
+ var selectedOption = options[this.settingsSelectElement.selectedIndex];
+ selectedOption.action(event);
+
+ // Select the correct color format setting again, since it needs to be selected.
+ var selectedIndex = 0;
+ for (var i = 0; i < options.length; ++i) {
+ if (options[i].value === Preferences.colorFormat) {
+ selectedIndex = i;
+ break;
+ }
+ }
+
+ this.settingsSelectElement.selectedIndex = selectedIndex;
+ },
+
+ _changeColorFormat: function(event)
+ {
+ var selectedOption = this.settingsSelectElement[this.settingsSelectElement.selectedIndex];
+ Preferences.colorFormat = selectedOption.value;
+
+ InspectorController.setSetting("color-format", Preferences.colorFormat);
+
+ for (var i = 0; i < this.sections.length; ++i)
+ this.sections[i].update(true);
+ },
+
+ _createNewRule: function(event)
+ {
+ this.addBlankSection().startEditingSelector();
+ },
+
+ addBlankSection: function()
+ {
+ var blankSection = new WebInspector.BlankStylePropertiesSection(this.appropriateSelectorForNode());
+ blankSection.pane = this;
+
+ var elementStyleSection = this.sections[1];
+ this.bodyElement.insertBefore(blankSection.element, elementStyleSection.element.nextSibling);
+
+ this.sections.splice(2, 0, blankSection);
+
+ return blankSection;
+ },
+
+ removeSection: function(section)
+ {
+ var index = this.sections.indexOf(section);
+ if (index === -1)
+ return;
+ this.sections.splice(index, 1);
+ if (section.element.parentNode)
+ section.element.parentNode.removeChild(section.element);
+ },
+
+ appropriateSelectorForNode: function()
+ {
+ var node = this.node;
+ if (!node)
+ return "";
+
+ var id = node.getAttribute("id");
+ if (id)
+ return "#" + id;
+
+ var className = node.getAttribute("class");
+ if (className)
+ return "." + className.replace(/\s+/, ".");
+
+ var nodeName = node.nodeName.toLowerCase();
+ if (nodeName === "input" && node.getAttribute("type"))
+ return nodeName + "[type=\"" + node.getAttribute("type") + "\"]";
+
+ return nodeName;
+ }
+}
+
+WebInspector.StylesSidebarPane.prototype.__proto__ = WebInspector.SidebarPane.prototype;
+
+WebInspector.StylePropertiesSection = function(styleRule, subtitle, computedStyle, usedProperties, editable)
+{
+ WebInspector.PropertiesSection.call(this, styleRule.selectorText);
+
+ this.titleElement.addEventListener("click", function(e) { e.stopPropagation(); }, false);
+ this.titleElement.addEventListener("dblclick", this._dblclickSelector.bind(this), false);
+ this.element.addEventListener("dblclick", this._dblclickEmptySpace.bind(this), false);
+
+ this.styleRule = styleRule;
+ this.rule = this.styleRule.rule;
+ this.computedStyle = computedStyle;
+ this.editable = (editable && !computedStyle);
+
+ // Prevent editing the user agent and user rules.
+ var isUserAgent = this.rule && this.rule.isUserAgent;
+ var isUser = this.rule && this.rule.isUser;
+ var isViaInspector = this.rule && this.rule.isViaInspector;
+
+ if (isUserAgent || isUser)
+ this.editable = false;
+
+ this._usedProperties = usedProperties;
+
+ if (computedStyle) {
+ this.element.addStyleClass("computed-style");
+
+ if (Preferences.showInheritedComputedStyleProperties)
+ this.element.addStyleClass("show-inherited");
+
+ var showInheritedLabel = document.createElement("label");
+ var showInheritedInput = document.createElement("input");
+ showInheritedInput.type = "checkbox";
+ showInheritedInput.checked = Preferences.showInheritedComputedStyleProperties;
+
+ var computedStyleSection = this;
+ var showInheritedToggleFunction = function(event) {
+ Preferences.showInheritedComputedStyleProperties = showInheritedInput.checked;
+ if (Preferences.showInheritedComputedStyleProperties)
+ computedStyleSection.element.addStyleClass("show-inherited");
+ else
+ computedStyleSection.element.removeStyleClass("show-inherited");
+ event.stopPropagation();
+ };
+
+ showInheritedLabel.addEventListener("click", showInheritedToggleFunction, false);
+
+ showInheritedLabel.appendChild(showInheritedInput);
+ showInheritedLabel.appendChild(document.createTextNode(WebInspector.UIString("Show inherited")));
+ this.subtitleElement.appendChild(showInheritedLabel);
+ } else {
+ if (!subtitle) {
+ if (this.styleRule.parentStyleSheet && this.styleRule.parentStyleSheet.href) {
+ var url = this.styleRule.parentStyleSheet.href;
+ subtitle = WebInspector.linkifyURL(url, WebInspector.displayNameForURL(url));
+ this.subtitleElement.addStyleClass("file");
+ } else if (isUserAgent)
+ subtitle = WebInspector.UIString("user agent stylesheet");
+ else if (isUser)
+ subtitle = WebInspector.UIString("user stylesheet");
+ else if (isViaInspector)
+ subtitle = WebInspector.UIString("via inspector");
+ else
+ subtitle = WebInspector.UIString("inline stylesheet");
+ }
+
+ this.subtitle = subtitle;
+ }
+
+ this.identifier = styleRule.selectorText;
+ if (this.subtitle)
+ this.identifier += ":" + this.subtitleElement.textContent;
+}
+
+WebInspector.StylePropertiesSection.prototype = {
+ get usedProperties()
+ {
+ return this._usedProperties || {};
+ },
+
+ set usedProperties(x)
+ {
+ this._usedProperties = x;
+ this.update();
+ },
+
+ expand: function(dontRememberState)
+ {
+ WebInspector.PropertiesSection.prototype.expand.call(this);
+ if (dontRememberState)
+ return;
+
+ if (!Preferences.styleRulesExpandedState)
+ Preferences.styleRulesExpandedState = {};
+ Preferences.styleRulesExpandedState[this.identifier] = true;
+ },
+
+ collapse: function(dontRememberState)
+ {
+ WebInspector.PropertiesSection.prototype.collapse.call(this);
+ if (dontRememberState)
+ return;
+
+ if (!Preferences.styleRulesExpandedState)
+ Preferences.styleRulesExpandedState = {};
+ Preferences.styleRulesExpandedState[this.identifier] = false;
+ },
+
+ isPropertyInherited: function(property)
+ {
+ if (!this.computedStyle || !this._usedProperties || this.noAffect)
+ return false;
+ // These properties should always show for Computed Style.
+ var alwaysShowComputedProperties = { "display": true, "height": true, "width": true };
+ return !(property in this.usedProperties) && !(property in alwaysShowComputedProperties) && !(property in this.disabledComputedProperties);
+ },
+
+ isPropertyOverloaded: function(property, shorthand)
+ {
+ if (this.computedStyle || !this._usedProperties || this.noAffect)
+ return false;
+
+ var used = (property in this.usedProperties);
+ if (used || !shorthand)
+ return !used;
+
+ // Find out if any of the individual longhand properties of the shorthand
+ // are used, if none are then the shorthand is overloaded too.
+ var longhandProperties = this.styleRule.style.getLonghandProperties(property);
+ for (var j = 0; j < longhandProperties.length; ++j) {
+ var individualProperty = longhandProperties[j];
+ if (individualProperty in this.usedProperties)
+ return false;
+ }
+
+ return true;
+ },
+
+ isInspectorStylesheet: function()
+ {
+ return (this.styleRule.parentStyleSheet === WebInspector.panels.elements.stylesheet);
+ },
+
+ update: function(full)
+ {
+ if (full || this.computedStyle) {
+ this.propertiesTreeOutline.removeChildren();
+ this.populated = false;
+ } else {
+ var child = this.propertiesTreeOutline.children[0];
+ while (child) {
+ child.overloaded = this.isPropertyOverloaded(child.name, child.shorthand);
+ child = child.traverseNextTreeElement(false, null, true);
+ }
+ }
+
+ if (this._afterUpdate) {
+ this._afterUpdate(this);
+ delete this._afterUpdate;
+ }
+ },
+
+ onpopulate: function()
+ {
+ var style = this.styleRule.style;
+
+ var foundShorthands = {};
+ var uniqueProperties = style.uniqueStyleProperties;
+ var disabledProperties = style.__disabledPropertyValues || {};
+
+ for (var name in disabledProperties)
+ uniqueProperties.push(name);
+
+ uniqueProperties.sort();
+
+ for (var i = 0; i < uniqueProperties.length; ++i) {
+ var name = uniqueProperties[i];
+ var disabled = name in disabledProperties;
+ if (!disabled && this.disabledComputedProperties && !(name in this.usedProperties) && name in this.disabledComputedProperties)
+ disabled = true;
+
+ var shorthand = !disabled ? style.getPropertyShorthand(name) : null;
+
+ if (shorthand && shorthand in foundShorthands)
+ continue;
+
+ if (shorthand) {
+ foundShorthands[shorthand] = true;
+ name = shorthand;
+ }
+
+ var isShorthand = (shorthand ? true : false);
+ var inherited = this.isPropertyInherited(name);
+ var overloaded = this.isPropertyOverloaded(name, isShorthand);
+
+ var item = new WebInspector.StylePropertyTreeElement(this.styleRule, style, name, isShorthand, inherited, overloaded, disabled);
+ this.propertiesTreeOutline.appendChild(item);
+ }
+ },
+
+ findTreeElementWithName: function(name)
+ {
+ var treeElement = this.propertiesTreeOutline.children[0];
+ while (treeElement) {
+ if (treeElement.name === name)
+ return treeElement;
+ treeElement = treeElement.traverseNextTreeElement(true, null, true);
+ }
+ return null;
+ },
+
+ addNewBlankProperty: function()
+ {
+ var item = new WebInspector.StylePropertyTreeElement(this.styleRule, this.styleRule.style, "", false, false, false, false);
+ this.propertiesTreeOutline.appendChild(item);
+ item.listItemElement.textContent = "";
+ item._newProperty = true;
+ return item;
+ },
+
+ _dblclickEmptySpace: function(event)
+ {
+ this.expand();
+ this.addNewBlankProperty().startEditing();
+ },
+
+ _dblclickSelector: function(event)
+ {
+ if (!this.editable)
+ return;
+
+ if (!this.rule && this.propertiesTreeOutline.children.length === 0) {
+ this.expand();
+ this.addNewBlankProperty().startEditing();
+ return;
+ }
+
+ if (!this.rule)
+ return;
+
+ this.startEditingSelector();
+ event.stopPropagation();
+ },
+
+ startEditingSelector: function()
+ {
+ var element = this.titleElement;
+ if (WebInspector.isBeingEdited(element))
+ return;
+
+ WebInspector.startEditing(this.titleElement, this.editingSelectorCommitted.bind(this), this.editingSelectorCancelled.bind(this), null);
+ window.getSelection().setBaseAndExtent(element, 0, element, 1);
+ },
+
+ editingSelectorCommitted: function(element, newContent, oldContent, context, moveDirection)
+ {
+ function moveToNextIfNeeded() {
+ if (!moveDirection || moveDirection !== "forward")
+ return;
+
+ this.expand();
+ if (this.propertiesTreeOutline.children.length === 0)
+ this.addNewBlankProperty().startEditing();
+ else {
+ var item = this.propertiesTreeOutline.children[0]
+ item.startEditing(item.valueElement);
+ }
+ }
+
+ if (newContent === oldContent)
+ return moveToNextIfNeeded.call(this);
+
+ var self = this;
+ function callback(result)
+ {
+ if (!result) {
+ // Invalid Syntax for a Selector
+ moveToNextIfNeeded.call(self);
+ return;
+ }
+
+ var newRulePayload = result[0];
+ var doesAffectSelectedNode = result[1];
+ if (!doesAffectSelectedNode) {
+ self.noAffect = true;
+ self.element.addStyleClass("no-affect");
+ } else {
+ delete self.noAffect;
+ self.element.removeStyleClass("no-affect");
+ }
+
+ var newRule = WebInspector.CSSStyleDeclaration.parseRule(newRulePayload);
+ self.rule = newRule;
+ self.styleRule = { section: self, style: newRule.style, selectorText: newRule.selectorText, parentStyleSheet: newRule.parentStyleSheet, rule: newRule };
+
+ var oldIdentifier = this.identifier;
+ self.identifier = newRule.selectorText + ":" + self.subtitleElement.textContent;
+
+ self.pane.update();
+
+ WebInspector.panels.elements.renameSelector(oldIdentifier, this.identifier, oldContent, newContent);
+
+ moveToNextIfNeeded.call(self);
+ }
+
+ InjectedScriptAccess.applyStyleRuleText(this.rule.id, newContent, this.pane.node.id, callback);
+ },
+
+ editingSelectorCancelled: function()
+ {
+ // Do nothing, this is overridden by BlankStylePropertiesSection.
+ }
+}
+
+WebInspector.StylePropertiesSection.prototype.__proto__ = WebInspector.PropertiesSection.prototype;
+
+WebInspector.BlankStylePropertiesSection = function(defaultSelectorText)
+{
+ WebInspector.StylePropertiesSection.call(this, {selectorText: defaultSelectorText, rule: {isViaInspector: true}}, "", false, {}, false);
+
+ this.element.addStyleClass("blank-section");
+}
+
+WebInspector.BlankStylePropertiesSection.prototype = {
+ expand: function()
+ {
+ // Do nothing, blank sections are not expandable.
+ },
+
+ editingSelectorCommitted: function(element, newContent, oldContent, context)
+ {
+ var self = this;
+ function callback(result)
+ {
+ if (!result) {
+ // Invalid Syntax for a Selector
+ self.editingSelectorCancelled();
+ return;
+ }
+
+ var rule = result[0];
+ var doesSelectorAffectSelectedNode = result[1];
+
+ var styleRule = WebInspector.CSSStyleDeclaration.parseRule(rule);
+ styleRule.rule = rule;
+
+ self.makeNormal(styleRule);
+
+ if (!doesSelectorAffectSelectedNode) {
+ self.noAffect = true;
+ self.element.addStyleClass("no-affect");
+ }
+
+ self.subtitleElement.textContent = WebInspector.UIString("via inspector");
+ self.expand();
+
+ self.addNewBlankProperty().startEditing();
+ }
+
+ InjectedScriptAccess.addStyleSelector(newContent, this.pane.node.id, callback);
+ },
+
+ editingSelectorCancelled: function()
+ {
+ this.pane.removeSection(this);
+ },
+
+ makeNormal: function(styleRule)
+ {
+ this.element.removeStyleClass("blank-section");
+
+ this.styleRule = styleRule;
+ this.rule = styleRule.rule;
+ this.computedStyle = false;
+ this.editable = true;
+ this.identifier = styleRule.selectorText + ":via inspector";
+
+ this.__proto__ = WebInspector.StylePropertiesSection.prototype;
+ }
+}
+
+WebInspector.BlankStylePropertiesSection.prototype.__proto__ = WebInspector.StylePropertiesSection.prototype;
+
+WebInspector.StylePropertyTreeElement = function(styleRule, style, name, shorthand, inherited, overloaded, disabled)
+{
+ this._styleRule = styleRule;
+ this.style = style;
+ this.name = name;
+ this.shorthand = shorthand;
+ this._inherited = inherited;
+ this._overloaded = overloaded;
+ this._disabled = disabled;
+
+ // Pass an empty title, the title gets made later in onattach.
+ TreeElement.call(this, "", null, shorthand);
+}
+
+WebInspector.StylePropertyTreeElement.prototype = {
+ get inherited()
+ {
+ return this._inherited;
+ },
+
+ set inherited(x)
+ {
+ if (x === this._inherited)
+ return;
+ this._inherited = x;
+ this.updateState();
+ },
+
+ get overloaded()
+ {
+ return this._overloaded;
+ },
+
+ set overloaded(x)
+ {
+ if (x === this._overloaded)
+ return;
+ this._overloaded = x;
+ this.updateState();
+ },
+
+ get disabled()
+ {
+ return this._disabled;
+ },
+
+ set disabled(x)
+ {
+ if (x === this._disabled)
+ return;
+ this._disabled = x;
+ this.updateState();
+ },
+
+ get priority()
+ {
+ if (this.disabled && this.style.__disabledPropertyPriorities && this.name in this.style.__disabledPropertyPriorities)
+ return this.style.__disabledPropertyPriorities[this.name];
+ return (this.shorthand ? this.style.getShorthandPriority(this.name) : this.style.getPropertyPriority(this.name));
+ },
+
+ get value()
+ {
+ if (this.disabled && this.style.__disabledPropertyValues && this.name in this.style.__disabledPropertyValues)
+ return this.style.__disabledPropertyValues[this.name];
+ return (this.shorthand ? this.style.getShorthandValue(this.name) : this.style.getPropertyValue(this.name));
+ },
+
+ onattach: function()
+ {
+ this.updateTitle();
+ },
+
+ updateTitle: function()
+ {
+ var priority = this.priority;
+ var value = this.value;
+
+ if (priority && !priority.length)
+ delete priority;
+ if (priority)
+ priority = "!" + priority;
+
+ this.updateState();
+
+ var enabledCheckboxElement = document.createElement("input");
+ enabledCheckboxElement.className = "enabled-button";
+ enabledCheckboxElement.type = "checkbox";
+ enabledCheckboxElement.checked = !this.disabled;
+ enabledCheckboxElement.addEventListener("change", this.toggleEnabled.bind(this), false);
+
+ var nameElement = document.createElement("span");
+ nameElement.className = "name";
+ nameElement.textContent = this.name;
+ this.nameElement = nameElement;
+
+ var valueElement = document.createElement("span");
+ valueElement.className = "value";
+ this.valueElement = valueElement;
+
+ if (value) {
+ function processValue(regex, processor, nextProcessor, valueText)
+ {
+ var container = document.createDocumentFragment();
+
+ var items = valueText.replace(regex, "\0$1\0").split("\0");
+ for (var i = 0; i < items.length; ++i) {
+ if ((i % 2) === 0) {
+ if (nextProcessor)
+ container.appendChild(nextProcessor(items[i]));
+ else
+ container.appendChild(document.createTextNode(items[i]));
+ } else {
+ var processedNode = processor(items[i]);
+ if (processedNode)
+ container.appendChild(processedNode);
+ }
+ }
+
+ return container;
+ }
+
+ function linkifyURL(url)
+ {
+ var container = document.createDocumentFragment();
+ container.appendChild(document.createTextNode("url("));
+ container.appendChild(WebInspector.linkifyURLAsNode(url, url, null, (url in WebInspector.resourceURLMap)));
+ container.appendChild(document.createTextNode(")"));
+ return container;
+ }
+
+ function processColor(text)
+ {
+ try {
+ var color = new WebInspector.Color(text);
+ } catch (e) {
+ return document.createTextNode(text);
+ }
+
+ var swatchElement = document.createElement("span");
+ swatchElement.title = WebInspector.UIString("Click to change color format");
+ swatchElement.className = "swatch";
+ swatchElement.style.setProperty("background-color", text);
+
+ swatchElement.addEventListener("click", changeColorDisplay, false);
+ swatchElement.addEventListener("dblclick", function(event) { event.stopPropagation() }, false);
+
+ var format;
+ if (Preferences.showColorNicknames && color.nickname)
+ format = "nickname";
+ else if (Preferences.colorFormat === "rgb")
+ format = (color.simple ? "rgb" : "rgba");
+ else if (Preferences.colorFormat === "hsl")
+ format = (color.simple ? "hsl" : "hsla");
+ else if (color.simple)
+ format = (color.hasShortHex() ? "shorthex" : "hex");
+ else
+ format = "rgba";
+
+ var colorValueElement = document.createElement("span");
+ colorValueElement.textContent = color.toString(format);
+
+ function changeColorDisplay(event)
+ {
+ switch (format) {
+ case "rgb":
+ format = "hsl";
+ break;
+
+ case "shorthex":
+ format = "hex";
+ break;
+
+ case "hex":
+ format = "rgb";
+ break;
+
+ case "nickname":
+ if (color.simple) {
+ if (color.hasShortHex())
+ format = "shorthex";
+ else
+ format = "hex";
+ break;
+ }
+
+ format = "rgba";
+ break;
+
+ case "hsl":
+ if (color.nickname)
+ format = "nickname";
+ else if (color.hasShortHex())
+ format = "shorthex";
+ else
+ format = "hex";
+ break;
+
+ case "rgba":
+ format = "hsla";
+ break;
+
+ case "hsla":
+ if (color.nickname)
+ format = "nickname";
+ else
+ format = "rgba";
+ break;
+ }
+
+ colorValueElement.textContent = color.toString(format);
+ }
+
+ var container = document.createDocumentFragment();
+ container.appendChild(swatchElement);
+ container.appendChild(colorValueElement);
+ return container;
+ }
+
+ var colorRegex = /((?:rgb|hsl)a?\([^)]+\)|#[0-9a-fA-F]{6}|#[0-9a-fA-F]{3}|\b\w+\b)/g;
+ var colorProcessor = processValue.bind(window, colorRegex, processColor, null);
+
+ valueElement.appendChild(processValue(/url\(([^)]+)\)/g, linkifyURL, colorProcessor, value));
+ }
+
+ if (priority) {
+ var priorityElement = document.createElement("span");
+ priorityElement.className = "priority";
+ priorityElement.textContent = priority;
+ }
+
+ this.listItemElement.removeChildren();
+
+ // Append the checkbox for root elements of an editable section.
+ if (this.treeOutline.section && this.treeOutline.section.editable && this.parent.root)
+ this.listItemElement.appendChild(enabledCheckboxElement);
+ this.listItemElement.appendChild(nameElement);
+ this.listItemElement.appendChild(document.createTextNode(": "));
+ this.listItemElement.appendChild(valueElement);
+
+ if (priorityElement) {
+ this.listItemElement.appendChild(document.createTextNode(" "));
+ this.listItemElement.appendChild(priorityElement);
+ }
+
+ this.listItemElement.appendChild(document.createTextNode(";"));
+
+ this.tooltip = this.name + ": " + valueElement.textContent + (priority ? " " + priority : "");
+ },
+
+ updateAll: function(updateAllRules)
+ {
+ if (updateAllRules && this.treeOutline.section && this.treeOutline.section.pane)
+ this.treeOutline.section.pane.update(null, this.treeOutline.section);
+ else if (this.treeOutline.section)
+ this.treeOutline.section.update(true);
+ else
+ this.updateTitle(); // FIXME: this will not show new properties. But we don't hit his case yet.
+ },
+
+ toggleEnabled: function(event)
+ {
+ var disabled = !event.target.checked;
+
+ var self = this;
+ function callback(newPayload)
+ {
+ if (!newPayload)
+ return;
+
+ self.style = WebInspector.CSSStyleDeclaration.parseStyle(newPayload);
+ self._styleRule.style = self.style;
+
+ // Set the disabled property here, since the code above replies on it not changing
+ // until after the value and priority are retrieved.
+ self.disabled = disabled;
+
+ if (self.treeOutline.section && self.treeOutline.section.pane)
+ self.treeOutline.section.pane.dispatchEventToListeners("style property toggled");
+
+ self.updateAll(true);
+ }
+
+ InjectedScriptAccess.toggleStyleEnabled(this.style.id, this.name, disabled, callback);
+ },
+
+ updateState: function()
+ {
+ if (!this.listItemElement)
+ return;
+
+ if (this.style.isPropertyImplicit(this.name) || this.value === "initial")
+ this.listItemElement.addStyleClass("implicit");
+ else
+ this.listItemElement.removeStyleClass("implicit");
+
+ if (this.inherited)
+ this.listItemElement.addStyleClass("inherited");
+ else
+ this.listItemElement.removeStyleClass("inherited");
+
+ if (this.overloaded)
+ this.listItemElement.addStyleClass("overloaded");
+ else
+ this.listItemElement.removeStyleClass("overloaded");
+
+ if (this.disabled)
+ this.listItemElement.addStyleClass("disabled");
+ else
+ this.listItemElement.removeStyleClass("disabled");
+ },
+
+ onpopulate: function()
+ {
+ // Only populate once and if this property is a shorthand.
+ if (this.children.length || !this.shorthand)
+ return;
+
+ var longhandProperties = this.style.getLonghandProperties(this.name);
+ for (var i = 0; i < longhandProperties.length; ++i) {
+ var name = longhandProperties[i];
+
+ if (this.treeOutline.section) {
+ var inherited = this.treeOutline.section.isPropertyInherited(name);
+ var overloaded = this.treeOutline.section.isPropertyOverloaded(name);
+ }
+
+ var item = new WebInspector.StylePropertyTreeElement(this._styleRule, this.style, name, false, inherited, overloaded);
+ this.appendChild(item);
+ }
+ },
+
+ ondblclick: function(element, event)
+ {
+ this.startEditing(event.target);
+ event.stopPropagation();
+ },
+
+ startEditing: function(selectElement)
+ {
+ // FIXME: we don't allow editing of longhand properties under a shorthand right now.
+ if (this.parent.shorthand)
+ return;
+
+ if (WebInspector.isBeingEdited(this.listItemElement) || (this.treeOutline.section && !this.treeOutline.section.editable))
+ return;
+
+ var context = { expanded: this.expanded, hasChildren: this.hasChildren };
+
+ // Lie about our children to prevent expanding on double click and to collapse shorthands.
+ this.hasChildren = false;
+
+ if (!selectElement)
+ selectElement = this.listItemElement;
+
+ this.listItemElement.handleKeyEvent = this.editingKeyDown.bind(this);
+
+ WebInspector.startEditing(this.listItemElement, this.editingCommitted.bind(this), this.editingCancelled.bind(this), context);
+ window.getSelection().setBaseAndExtent(selectElement, 0, selectElement, 1);
+ },
+
+ editingKeyDown: function(event)
+ {
+ var arrowKeyPressed = (event.keyIdentifier === "Up" || event.keyIdentifier === "Down");
+ var pageKeyPressed = (event.keyIdentifier === "PageUp" || event.keyIdentifier === "PageDown");
+ if (!arrowKeyPressed && !pageKeyPressed)
+ return;
+
+ var selection = window.getSelection();
+ if (!selection.rangeCount)
+ return;
+
+ var selectionRange = selection.getRangeAt(0);
+ if (selectionRange.commonAncestorContainer !== this.listItemElement && !selectionRange.commonAncestorContainer.isDescendant(this.listItemElement))
+ return;
+
+ const styleValueDelimeters = " \t\n\"':;,/()";
+ var wordRange = selectionRange.startContainer.rangeOfWord(selectionRange.startOffset, styleValueDelimeters, this.listItemElement);
+ var wordString = wordRange.toString();
+ var replacementString = wordString;
+
+ var matches = /(.*?)(-?\d+(?:\.\d+)?)(.*)/.exec(wordString);
+ if (matches && matches.length) {
+ var prefix = matches[1];
+ var number = parseFloat(matches[2]);
+ var suffix = matches[3];
+
+ // If the number is near zero or the number is one and the direction will take it near zero.
+ var numberNearZero = (number < 1 && number > -1);
+ if (number === 1 && event.keyIdentifier === "Down")
+ numberNearZero = true;
+ else if (number === -1 && event.keyIdentifier === "Up")
+ numberNearZero = true;
+
+ if (numberNearZero && event.altKey && arrowKeyPressed) {
+ if (event.keyIdentifier === "Down")
+ number = Math.ceil(number - 1);
+ else
+ number = Math.floor(number + 1);
+ } else {
+ // Jump by 10 when shift is down or jump by 0.1 when near zero or Alt/Option is down.
+ // Also jump by 10 for page up and down, or by 100 if shift is held with a page key.
+ var changeAmount = 1;
+ if (event.shiftKey && pageKeyPressed)
+ changeAmount = 100;
+ else if (event.shiftKey || pageKeyPressed)
+ changeAmount = 10;
+ else if (event.altKey || numberNearZero)
+ changeAmount = 0.1;
+
+ if (event.keyIdentifier === "Down" || event.keyIdentifier === "PageDown")
+ changeAmount *= -1;
+
+ // Make the new number and constrain it to a precision of 6, this matches numbers the engine returns.
+ // Use the Number constructor to forget the fixed precision, so 1.100000 will print as 1.1.
+ number = Number((number + changeAmount).toFixed(6));
+ }
+
+ replacementString = prefix + number + suffix;
+ } else {
+ // FIXME: this should cycle through known keywords for the current property name.
+ return;
+ }
+
+ var replacementTextNode = document.createTextNode(replacementString);
+
+ wordRange.deleteContents();
+ wordRange.insertNode(replacementTextNode);
+
+ var finalSelectionRange = document.createRange();
+ finalSelectionRange.setStart(replacementTextNode, 0);
+ finalSelectionRange.setEnd(replacementTextNode, replacementString.length);
+
+ selection.removeAllRanges();
+ selection.addRange(finalSelectionRange);
+
+ event.preventDefault();
+ event.handled = true;
+
+ if (!this.originalCSSText) {
+ // Remember the rule's original CSS text, so it can be restored
+ // if the editing is canceled and before each apply.
+ this.originalCSSText = this.style.styleTextWithShorthands();
+ } else {
+ // Restore the original CSS text before applying user changes. This is needed to prevent
+ // new properties from sticking around if the user adds one, then removes it.
+ InjectedScriptAccess.setStyleText(this.style.id, this.originalCSSText);
+ }
+
+ this.applyStyleText(this.listItemElement.textContent);
+ },
+
+ editingEnded: function(context)
+ {
+ this.hasChildren = context.hasChildren;
+ if (context.expanded)
+ this.expand();
+ delete this.listItemElement.handleKeyEvent;
+ delete this.originalCSSText;
+ },
+
+ editingCancelled: function(element, context)
+ {
+ if (this._newProperty)
+ this.treeOutline.removeChild(this);
+ else if (this.originalCSSText) {
+ InjectedScriptAccess.setStyleText(this.style.id, this.originalCSSText);
+
+ if (this.treeOutline.section && this.treeOutline.section.pane)
+ this.treeOutline.section.pane.dispatchEventToListeners("style edited");
+
+ this.updateAll();
+ } else
+ this.updateTitle();
+
+ this.editingEnded(context);
+ },
+
+ editingCommitted: function(element, userInput, previousContent, context, moveDirection)
+ {
+ this.editingEnded(context);
+
+ // Determine where to move to before making changes
+ var newProperty, moveToPropertyName, moveToSelector;
+ var moveTo = (moveDirection === "forward" ? this.nextSibling : this.previousSibling);
+ if (moveTo)
+ moveToPropertyName = moveTo.name;
+ else if (moveDirection === "forward")
+ newProperty = true;
+ else if (moveDirection === "backward" && this.treeOutline.section.rule)
+ moveToSelector = true;
+
+ // Make the Changes and trigger the moveToNextCallback after updating
+ var blankInput = /^\s*$/.test(userInput);
+ if (userInput !== previousContent || (this._newProperty && blankInput)) { // only if something changed, or adding a new style and it was blank
+ this.treeOutline.section._afterUpdate = moveToNextCallback.bind(this, this._newProperty, !blankInput);
+ this.applyStyleText(userInput, true);
+ } else
+ moveToNextCallback(this._newProperty, false, this.treeOutline.section, false);
+
+ // The Callback to start editing the next property
+ function moveToNextCallback(alreadyNew, valueChanged, section)
+ {
+ if (!moveDirection)
+ return;
+
+ // User just tabbed through without changes
+ if (moveTo && moveTo.parent) {
+ moveTo.startEditing(moveTo.valueElement);
+ return;
+ }
+
+ // User has made a change then tabbed, wiping all the original treeElements,
+ // recalculate the new treeElement for the same property we were going to edit next
+ if (moveTo && !moveTo.parent) {
+ var treeElement = section.findTreeElementWithName(moveToPropertyName);
+ if (treeElement)
+ treeElement.startEditing(treeElement.valueElement);
+ return;
+ }
+
+ // Create a new attribute in this section
+ if (newProperty) {
+ if (alreadyNew && !valueChanged)
+ return;
+
+ var item = section.addNewBlankProperty();
+ item.startEditing();
+ return;
+ }
+
+ if (moveToSelector)
+ section.startEditingSelector();
+ }
+ },
+
+ applyStyleText: function(styleText, updateInterface)
+ {
+ var section = this.treeOutline.section;
+ var elementsPanel = WebInspector.panels.elements;
+ var styleTextLength = styleText.trimWhitespace().length;
+ if (!styleTextLength && updateInterface) {
+ if (this._newProperty) {
+ // The user deleted everything, so remove the tree element and update.
+ this.parent.removeChild(this);
+ return;
+ } else {
+ delete section._afterUpdate;
+ }
+ }
+
+ var self = this;
+ function callback(result)
+ {
+ if (!result) {
+ // The user typed something, but it didn't parse. Just abort and restore
+ // the original title for this property. If this was a new attribute and
+ // we couldn't parse, then just remove it.
+ if (self._newProperty) {
+ self.parent.removeChild(self);
+ return;
+ }
+ if (updateInterface)
+ self.updateTitle();
+ return;
+ }
+
+ var newPayload = result[0];
+ var changedProperties = result[1];
+ elementsPanel.removeStyleChange(section.identifier, self.style, self.name);
+
+ if (!styleTextLength) {
+ // Do remove ourselves from UI when the property removal is confirmed.
+ self.parent.removeChild(self);
+ } else {
+ self.style = WebInspector.CSSStyleDeclaration.parseStyle(newPayload);
+ for (var i = 0; i < changedProperties.length; ++i)
+ elementsPanel.addStyleChange(section.identifier, self.style, changedProperties[i]);
+ self._styleRule.style = self.style;
+ }
+
+ if (section && section.pane)
+ section.pane.dispatchEventToListeners("style edited");
+
+ if (updateInterface)
+ self.updateAll(true);
+
+ if (!self.rule)
+ WebInspector.panels.elements.treeOutline.update();
+ }
+
+ InjectedScriptAccess.applyStyleText(this.style.id, styleText.trimWhitespace(), this.name, callback);
+ }
+}
+
+WebInspector.StylePropertyTreeElement.prototype.__proto__ = TreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SummaryBar.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SummaryBar.js
new file mode 100644
index 0000000..bbf2b1a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/SummaryBar.js
@@ -0,0 +1,364 @@
+/*
+ * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2008, 2009 Anthony Ricaud <rik@webkit.org>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.SummaryBar = function(categories)
+{
+ this.categories = categories;
+
+ this.element = document.createElement("div");
+ this.element.className = "summary-bar";
+
+ this.graphElement = document.createElement("canvas");
+ this.graphElement.setAttribute("width", "450");
+ this.graphElement.setAttribute("height", "38");
+ this.graphElement.className = "summary-graph";
+ this.element.appendChild(this.graphElement);
+
+ this.legendElement = document.createElement("div");
+ this.legendElement.className = "summary-graph-legend";
+ this.element.appendChild(this.legendElement);
+}
+
+WebInspector.SummaryBar.prototype = {
+
+ get calculator() {
+ return this._calculator;
+ },
+
+ set calculator(x) {
+ this._calculator = x;
+ },
+
+ reset: function()
+ {
+ this.legendElement.removeChildren();
+ this._drawSummaryGraph();
+ },
+
+ update: function(data)
+ {
+ var graphInfo = this.calculator.computeSummaryValues(data);
+
+ var fillSegments = [];
+
+ this.legendElement.removeChildren();
+
+ for (var category in this.categories) {
+ var size = graphInfo.categoryValues[category];
+ if (!size)
+ continue;
+
+ var color = this.categories[category].color;
+ var colorString = "rgb(" + color.r + ", " + color.g + ", " + color.b + ")";
+
+ var fillSegment = {color: colorString, value: size};
+ fillSegments.push(fillSegment);
+
+ var legendLabel = this._makeLegendElement(this.categories[category].title, this.calculator.formatValue(size), colorString);
+ this.legendElement.appendChild(legendLabel);
+ }
+
+ if (graphInfo.total) {
+ var totalLegendLabel = this._makeLegendElement(WebInspector.UIString("Total"), this.calculator.formatValue(graphInfo.total));
+ totalLegendLabel.addStyleClass("total");
+ this.legendElement.appendChild(totalLegendLabel);
+ }
+
+ this._drawSummaryGraph(fillSegments);
+ },
+
+ _drawSwatch: function(canvas, color)
+ {
+ var ctx = canvas.getContext("2d");
+
+ function drawSwatchSquare() {
+ ctx.fillStyle = color;
+ ctx.fillRect(0, 0, 13, 13);
+
+ var gradient = ctx.createLinearGradient(0, 0, 13, 13);
+ gradient.addColorStop(0.0, "rgba(255, 255, 255, 0.2)");
+ gradient.addColorStop(1.0, "rgba(255, 255, 255, 0.0)");
+
+ ctx.fillStyle = gradient;
+ ctx.fillRect(0, 0, 13, 13);
+
+ gradient = ctx.createLinearGradient(13, 13, 0, 0);
+ gradient.addColorStop(0.0, "rgba(0, 0, 0, 0.2)");
+ gradient.addColorStop(1.0, "rgba(0, 0, 0, 0.0)");
+
+ ctx.fillStyle = gradient;
+ ctx.fillRect(0, 0, 13, 13);
+
+ ctx.strokeStyle = "rgba(0, 0, 0, 0.6)";
+ ctx.strokeRect(0.5, 0.5, 12, 12);
+ }
+
+ ctx.clearRect(0, 0, 13, 24);
+
+ drawSwatchSquare();
+
+ ctx.save();
+
+ ctx.translate(0, 25);
+ ctx.scale(1, -1);
+
+ drawSwatchSquare();
+
+ ctx.restore();
+
+ this._fadeOutRect(ctx, 0, 13, 13, 13, 0.5, 0.0);
+ },
+
+ _drawSummaryGraph: function(segments)
+ {
+ if (!segments || !segments.length) {
+ segments = [{color: "white", value: 1}];
+ this._showingEmptySummaryGraph = true;
+ } else
+ delete this._showingEmptySummaryGraph;
+
+ // Calculate the total of all segments.
+ var total = 0;
+ for (var i = 0; i < segments.length; ++i)
+ total += segments[i].value;
+
+ // Calculate the percentage of each segment, rounded to the nearest percent.
+ var percents = segments.map(function(s) { return Math.max(Math.round(100 * s.value / total), 1) });
+
+ // Calculate the total percentage.
+ var percentTotal = 0;
+ for (var i = 0; i < percents.length; ++i)
+ percentTotal += percents[i];
+
+ // Make sure our percentage total is not greater-than 100, it can be greater
+ // if we rounded up for a few segments.
+ while (percentTotal > 100) {
+ for (var i = 0; i < percents.length && percentTotal > 100; ++i) {
+ if (percents[i] > 1) {
+ --percents[i];
+ --percentTotal;
+ }
+ }
+ }
+
+ // Make sure our percentage total is not less-than 100, it can be less
+ // if we rounded down for a few segments.
+ while (percentTotal < 100) {
+ for (var i = 0; i < percents.length && percentTotal < 100; ++i) {
+ ++percents[i];
+ ++percentTotal;
+ }
+ }
+
+ var ctx = this.graphElement.getContext("2d");
+
+ var x = 0;
+ var y = 0;
+ var w = 450;
+ var h = 19;
+ var r = (h / 2);
+
+ function drawPillShadow()
+ {
+ // This draws a line with a shadow that is offset away from the line. The line is stroked
+ // twice with different X shadow offsets to give more feathered edges. Later we erase the
+ // line with destination-out 100% transparent black, leaving only the shadow. This only
+ // works if nothing has been drawn into the canvas yet.
+
+ ctx.beginPath();
+ ctx.moveTo(x + 4, y + h - 3 - 0.5);
+ ctx.lineTo(x + w - 4, y + h - 3 - 0.5);
+ ctx.closePath();
+
+ ctx.save();
+
+ ctx.shadowBlur = 2;
+ ctx.shadowColor = "rgba(0, 0, 0, 0.5)";
+ ctx.shadowOffsetX = 3;
+ ctx.shadowOffsetY = 5;
+
+ ctx.strokeStyle = "white";
+ ctx.lineWidth = 1;
+
+ ctx.stroke();
+
+ ctx.shadowOffsetX = -3;
+
+ ctx.stroke();
+
+ ctx.restore();
+
+ ctx.save();
+
+ ctx.globalCompositeOperation = "destination-out";
+ ctx.strokeStyle = "rgba(0, 0, 0, 1)";
+ ctx.lineWidth = 1;
+
+ ctx.stroke();
+
+ ctx.restore();
+ }
+
+ function drawPill()
+ {
+ // Make a rounded rect path.
+ ctx.beginPath();
+ ctx.moveTo(x, y + r);
+ ctx.lineTo(x, y + h - r);
+ ctx.quadraticCurveTo(x, y + h, x + r, y + h);
+ ctx.lineTo(x + w - r, y + h);
+ ctx.quadraticCurveTo(x + w, y + h, x + w, y + h - r);
+ ctx.lineTo(x + w, y + r);
+ ctx.quadraticCurveTo(x + w, y, x + w - r, y);
+ ctx.lineTo(x + r, y);
+ ctx.quadraticCurveTo(x, y, x, y + r);
+ ctx.closePath();
+
+ // Clip to the rounded rect path.
+ ctx.save();
+ ctx.clip();
+
+ // Fill the segments with the associated color.
+ var previousSegmentsWidth = 0;
+ for (var i = 0; i < segments.length; ++i) {
+ var segmentWidth = Math.round(w * percents[i] / 100);
+ ctx.fillStyle = segments[i].color;
+ ctx.fillRect(x + previousSegmentsWidth, y, segmentWidth, h);
+ previousSegmentsWidth += segmentWidth;
+ }
+
+ // Draw the segment divider lines.
+ ctx.lineWidth = 1;
+ for (var i = 1; i < 20; ++i) {
+ ctx.beginPath();
+ ctx.moveTo(x + (i * Math.round(w / 20)) + 0.5, y);
+ ctx.lineTo(x + (i * Math.round(w / 20)) + 0.5, y + h);
+ ctx.closePath();
+
+ ctx.strokeStyle = "rgba(0, 0, 0, 0.2)";
+ ctx.stroke();
+
+ ctx.beginPath();
+ ctx.moveTo(x + (i * Math.round(w / 20)) + 1.5, y);
+ ctx.lineTo(x + (i * Math.round(w / 20)) + 1.5, y + h);
+ ctx.closePath();
+
+ ctx.strokeStyle = "rgba(255, 255, 255, 0.2)";
+ ctx.stroke();
+ }
+
+ // Draw the pill shading.
+ var lightGradient = ctx.createLinearGradient(x, y, x, y + (h / 1.5));
+ lightGradient.addColorStop(0.0, "rgba(220, 220, 220, 0.6)");
+ lightGradient.addColorStop(0.4, "rgba(220, 220, 220, 0.2)");
+ lightGradient.addColorStop(1.0, "rgba(255, 255, 255, 0.0)");
+
+ var darkGradient = ctx.createLinearGradient(x, y + (h / 3), x, y + h);
+ darkGradient.addColorStop(0.0, "rgba(0, 0, 0, 0.0)");
+ darkGradient.addColorStop(0.8, "rgba(0, 0, 0, 0.2)");
+ darkGradient.addColorStop(1.0, "rgba(0, 0, 0, 0.5)");
+
+ ctx.fillStyle = darkGradient;
+ ctx.fillRect(x, y, w, h);
+
+ ctx.fillStyle = lightGradient;
+ ctx.fillRect(x, y, w, h);
+
+ ctx.restore();
+ }
+
+ ctx.clearRect(x, y, w, (h * 2));
+
+ drawPillShadow();
+ drawPill();
+
+ ctx.save();
+
+ ctx.translate(0, (h * 2) + 1);
+ ctx.scale(1, -1);
+
+ drawPill();
+
+ ctx.restore();
+
+ this._fadeOutRect(ctx, x, y + h + 1, w, h, 0.5, 0.0);
+ },
+
+ _fadeOutRect: function(ctx, x, y, w, h, a1, a2)
+ {
+ ctx.save();
+
+ var gradient = ctx.createLinearGradient(x, y, x, y + h);
+ gradient.addColorStop(0.0, "rgba(0, 0, 0, " + (1.0 - a1) + ")");
+ gradient.addColorStop(0.8, "rgba(0, 0, 0, " + (1.0 - a2) + ")");
+ gradient.addColorStop(1.0, "rgba(0, 0, 0, 1.0)");
+
+ ctx.globalCompositeOperation = "destination-out";
+
+ ctx.fillStyle = gradient;
+ ctx.fillRect(x, y, w, h);
+
+ ctx.restore();
+ },
+
+ _makeLegendElement: function(label, value, color)
+ {
+ var legendElement = document.createElement("label");
+ legendElement.className = "summary-graph-legend-item";
+
+ if (color) {
+ var swatch = document.createElement("canvas");
+ swatch.className = "summary-graph-legend-swatch";
+ swatch.setAttribute("width", "13");
+ swatch.setAttribute("height", "24");
+
+ legendElement.appendChild(swatch);
+
+ this._drawSwatch(swatch, color);
+ }
+
+ var labelElement = document.createElement("div");
+ labelElement.className = "summary-graph-legend-label";
+ legendElement.appendChild(labelElement);
+
+ var headerElement = document.createElement("div");
+ headerElement.className = "summary-graph-legend-header";
+ headerElement.textContent = label;
+ labelElement.appendChild(headerElement);
+
+ var valueElement = document.createElement("div");
+ valueElement.className = "summary-graph-legend-value";
+ valueElement.textContent = value;
+ labelElement.appendChild(valueElement);
+
+ return legendElement;
+ }
+}
+
+WebInspector.SummaryBar.prototype.__proto__ = WebInspector.Object.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TextPrompt.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TextPrompt.js
new file mode 100644
index 0000000..5ff774f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TextPrompt.js
@@ -0,0 +1,319 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.TextPrompt = function(element, completions, stopCharacters)
+{
+ this.element = element;
+ this.completions = completions;
+ this.completionStopCharacters = stopCharacters;
+ this.history = [];
+ this.historyOffset = 0;
+}
+
+WebInspector.TextPrompt.prototype = {
+ get text()
+ {
+ return this.element.textContent;
+ },
+
+ set text(x)
+ {
+ if (!x) {
+ // Append a break element instead of setting textContent to make sure the selection is inside the prompt.
+ this.element.removeChildren();
+ this.element.appendChild(document.createElement("br"));
+ } else
+ this.element.textContent = x;
+
+ this.moveCaretToEndOfPrompt();
+ },
+
+ handleKeyEvent: function(event)
+ {
+ switch (event.keyIdentifier) {
+ case "Up":
+ this._upKeyPressed(event);
+ break;
+ case "Down":
+ this._downKeyPressed(event);
+ break;
+ case "U+0009": // Tab
+ this._tabKeyPressed(event);
+ break;
+ case "Right":
+ case "End":
+ if (!this.acceptAutoComplete())
+ this.autoCompleteSoon();
+ break;
+ default:
+ this.clearAutoComplete();
+ this.autoCompleteSoon();
+ break;
+ }
+ },
+
+ acceptAutoComplete: function()
+ {
+ if (!this.autoCompleteElement || !this.autoCompleteElement.parentNode)
+ return false;
+
+ var text = this.autoCompleteElement.textContent;
+ var textNode = document.createTextNode(text);
+ this.autoCompleteElement.parentNode.replaceChild(textNode, this.autoCompleteElement);
+ delete this.autoCompleteElement;
+
+ var finalSelectionRange = document.createRange();
+ finalSelectionRange.setStart(textNode, text.length);
+ finalSelectionRange.setEnd(textNode, text.length);
+
+ var selection = window.getSelection();
+ selection.removeAllRanges();
+ selection.addRange(finalSelectionRange);
+
+ return true;
+ },
+
+ clearAutoComplete: function(includeTimeout)
+ {
+ if (includeTimeout && "_completeTimeout" in this) {
+ clearTimeout(this._completeTimeout);
+ delete this._completeTimeout;
+ }
+
+ if (!this.autoCompleteElement)
+ return;
+
+ if (this.autoCompleteElement.parentNode)
+ this.autoCompleteElement.parentNode.removeChild(this.autoCompleteElement);
+ delete this.autoCompleteElement;
+
+ if (!this._userEnteredRange || !this._userEnteredText)
+ return;
+
+ this._userEnteredRange.deleteContents();
+
+ var userTextNode = document.createTextNode(this._userEnteredText);
+ this._userEnteredRange.insertNode(userTextNode);
+
+ var selectionRange = document.createRange();
+ selectionRange.setStart(userTextNode, this._userEnteredText.length);
+ selectionRange.setEnd(userTextNode, this._userEnteredText.length);
+
+ var selection = window.getSelection();
+ selection.removeAllRanges();
+ selection.addRange(selectionRange);
+
+ delete this._userEnteredRange;
+ delete this._userEnteredText;
+ },
+
+ autoCompleteSoon: function()
+ {
+ if (!("_completeTimeout" in this))
+ this._completeTimeout = setTimeout(this.complete.bind(this, true), 250);
+ },
+
+ complete: function(auto)
+ {
+ this.clearAutoComplete(true);
+ var selection = window.getSelection();
+ if (!selection.rangeCount)
+ return;
+
+ var selectionRange = selection.getRangeAt(0);
+ if (!selectionRange.commonAncestorContainer.isDescendant(this.element))
+ return;
+ if (auto && !this.isCaretAtEndOfPrompt())
+ return;
+ var wordPrefixRange = selectionRange.startContainer.rangeOfWord(selectionRange.startOffset, this.completionStopCharacters, this.element, "backward");
+ this.completions(wordPrefixRange, auto, this._completionsReady.bind(this, selection, auto, wordPrefixRange));
+ },
+
+ _completionsReady: function(selection, auto, originalWordPrefixRange, completions)
+ {
+ if (!completions || !completions.length)
+ return;
+
+ var selectionRange = selection.getRangeAt(0);
+
+ var fullWordRange = document.createRange();
+ fullWordRange.setStart(originalWordPrefixRange.startContainer, originalWordPrefixRange.startOffset);
+ fullWordRange.setEnd(selectionRange.endContainer, selectionRange.endOffset);
+
+ if (originalWordPrefixRange.toString() + selectionRange.toString() != fullWordRange.toString())
+ return;
+
+ if (completions.length === 1 || selection.isCollapsed || auto) {
+ var completionText = completions[0];
+ } else {
+ var currentText = fullWordRange.toString();
+
+ var foundIndex = null;
+ for (var i = 0; i < completions.length; ++i) {
+ if (completions[i] === currentText)
+ foundIndex = i;
+ }
+
+ if (foundIndex === null || (foundIndex + 1) >= completions.length)
+ var completionText = completions[0];
+ else
+ var completionText = completions[foundIndex + 1];
+ }
+
+ var wordPrefixLength = originalWordPrefixRange.toString().length;
+
+ this._userEnteredRange = fullWordRange;
+ this._userEnteredText = fullWordRange.toString();
+
+ fullWordRange.deleteContents();
+
+ var finalSelectionRange = document.createRange();
+
+ if (auto) {
+ var prefixText = completionText.substring(0, wordPrefixLength);
+ var suffixText = completionText.substring(wordPrefixLength);
+
+ var prefixTextNode = document.createTextNode(prefixText);
+ fullWordRange.insertNode(prefixTextNode);
+
+ this.autoCompleteElement = document.createElement("span");
+ this.autoCompleteElement.className = "auto-complete-text";
+ this.autoCompleteElement.textContent = suffixText;
+
+ prefixTextNode.parentNode.insertBefore(this.autoCompleteElement, prefixTextNode.nextSibling);
+
+ finalSelectionRange.setStart(prefixTextNode, wordPrefixLength);
+ finalSelectionRange.setEnd(prefixTextNode, wordPrefixLength);
+ } else {
+ var completionTextNode = document.createTextNode(completionText);
+ fullWordRange.insertNode(completionTextNode);
+
+ if (completions.length > 1)
+ finalSelectionRange.setStart(completionTextNode, wordPrefixLength);
+ else
+ finalSelectionRange.setStart(completionTextNode, completionText.length);
+
+ finalSelectionRange.setEnd(completionTextNode, completionText.length);
+ }
+
+ selection.removeAllRanges();
+ selection.addRange(finalSelectionRange);
+ },
+
+ isCaretInsidePrompt: function()
+ {
+ return this.element.isInsertionCaretInside();
+ },
+
+ isCaretAtEndOfPrompt: function()
+ {
+ var selection = window.getSelection();
+ if (!selection.rangeCount || !selection.isCollapsed)
+ return false;
+
+ var selectionRange = selection.getRangeAt(0);
+ var node = selectionRange.startContainer;
+ if (node !== this.element && !node.isDescendant(this.element))
+ return false;
+
+ if (node.nodeType === Node.TEXT_NODE && selectionRange.startOffset < node.nodeValue.length)
+ return false;
+
+ var foundNextText = false;
+ while (node) {
+ if (node.nodeType === Node.TEXT_NODE && node.nodeValue.length) {
+ if (foundNextText)
+ return false;
+ foundNextText = true;
+ }
+
+ node = node.traverseNextNode(false, this.element);
+ }
+
+ return true;
+ },
+
+ moveCaretToEndOfPrompt: function()
+ {
+ var selection = window.getSelection();
+ var selectionRange = document.createRange();
+
+ var offset = this.element.childNodes.length;
+ selectionRange.setStart(this.element, offset);
+ selectionRange.setEnd(this.element, offset);
+
+ selection.removeAllRanges();
+ selection.addRange(selectionRange);
+ },
+
+ _tabKeyPressed: function(event)
+ {
+ event.preventDefault();
+ event.stopPropagation();
+
+ this.complete();
+ },
+
+ _upKeyPressed: function(event)
+ {
+ event.preventDefault();
+ event.stopPropagation();
+
+ if (this.historyOffset == this.history.length)
+ return;
+
+ this.clearAutoComplete(true);
+
+ if (this.historyOffset == 0)
+ this.tempSavedCommand = this.text;
+
+ ++this.historyOffset;
+ this.text = this.history[this.history.length - this.historyOffset];
+ },
+
+ _downKeyPressed: function(event)
+ {
+ event.preventDefault();
+ event.stopPropagation();
+
+ if (this.historyOffset == 0)
+ return;
+
+ this.clearAutoComplete(true);
+
+ --this.historyOffset;
+
+ if (this.historyOffset == 0) {
+ this.text = this.tempSavedCommand;
+ delete this.tempSavedCommand;
+ return;
+ }
+
+ this.text = this.history[this.history.length - this.historyOffset];
+ }
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TimelineAgent.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TimelineAgent.js
new file mode 100644
index 0000000..6d18732
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TimelineAgent.js
@@ -0,0 +1,54 @@
+/*
+ * Copyright (C) 2009 Google Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.TimelineAgent = function() {
+ // Not implemented.
+}
+
+// Must be kept in sync with TimelineItem.h
+WebInspector.TimelineAgent.ItemType = {
+ DOMDispatch : 0,
+ Layout : 1,
+ RecalculateStyles : 2,
+ Paint : 3,
+ ParseHTML : 4
+};
+
+WebInspector.addItemToTimeline = function(record) {
+ // Not implemented.
+}
+
+WebInspector.timelineWasEnabled = function() {
+ // Not implemented.
+}
+
+WebInspector.timelineWasDisabled = function() {
+ // Not implemented.
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TopDownProfileDataGridTree.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TopDownProfileDataGridTree.js
new file mode 100644
index 0000000..b9d8b94
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/TopDownProfileDataGridTree.js
@@ -0,0 +1,111 @@
+/*
+ * Copyright (C) 2009 280 North Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.TopDownProfileDataGridNode = function(/*ProfileView*/ profileView, /*ProfileNode*/ profileNode, /*TopDownProfileDataGridTree*/ owningTree)
+{
+ var hasChildren = (profileNode.children && profileNode.children.length);
+
+ WebInspector.ProfileDataGridNode.call(this, profileView, profileNode, owningTree, hasChildren);
+
+ this._remainingChildren = profileNode.children;
+}
+
+WebInspector.TopDownProfileDataGridNode.prototype = {
+ _populate: function(event)
+ {
+ var children = this._remainingChildren;
+ var childrenLength = children.length;
+
+ for (var i = 0; i < childrenLength; ++i)
+ this.appendChild(new WebInspector.TopDownProfileDataGridNode(this.profileView, children[i], this.tree));
+
+ if (this.removeEventListener)
+ this.removeEventListener("populate", this._populate, this);
+
+ this._remainingChildren = null;
+ },
+
+ _exclude: function(aCallUID)
+ {
+ if (this._remainingChildren)
+ this._populate();
+
+ this._save();
+
+ var children = this.children;
+ var index = this.children.length;
+
+ while (index--)
+ children[index]._exclude(aCallUID);
+
+ var child = this.childrenByCallUID[aCallUID];
+
+ if (child)
+ this._merge(child, true);
+ }
+}
+
+WebInspector.TopDownProfileDataGridNode.prototype.__proto__ = WebInspector.ProfileDataGridNode.prototype;
+
+WebInspector.TopDownProfileDataGridTree = function(/*ProfileView*/ profileView, /*ProfileNode*/ profileNode)
+{
+ WebInspector.ProfileDataGridTree.call(this, profileView, profileNode);
+
+ this._remainingChildren = profileNode.children;
+
+ WebInspector.TopDownProfileDataGridNode.prototype._populate.call(this);
+}
+
+WebInspector.TopDownProfileDataGridTree.prototype = {
+ focus: function(/*ProfileDataGridNode*/ profileDataGrideNode)
+ {
+ if (!profileDataGrideNode)
+ return;
+
+ this._save();
+
+ this.children = [profileDataGrideNode];
+ this.totalTime = profileDataGrideNode.totalTime;
+ },
+
+ exclude: function(/*ProfileDataGridNode*/ profileDataGrideNode)
+ {
+ if (!profileDataGrideNode)
+ return;
+
+ this._save();
+
+ var excludedCallUID = profileDataGrideNode.callUID;
+
+ WebInspector.TopDownProfileDataGridNode.prototype._exclude.call(this, excludedCallUID);
+
+ if (this.lastComparator)
+ this.sort(this.lastComparator, true);
+ },
+
+ _merge: WebInspector.TopDownProfileDataGridNode.prototype._merge
+}
+
+WebInspector.TopDownProfileDataGridTree.prototype.__proto__ = WebInspector.ProfileDataGridTree.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/View.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/View.js
new file mode 100644
index 0000000..632a61ae
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/View.js
@@ -0,0 +1,74 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+ * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+ * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+ * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+ * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
+ * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.View = function(element)
+{
+ this.element = element || document.createElement("div");
+ this._visible = false;
+}
+
+WebInspector.View.prototype = {
+ get visible()
+ {
+ return this._visible;
+ },
+
+ set visible(x)
+ {
+ if (this._visible === x)
+ return;
+
+ if (x)
+ this.show();
+ else
+ this.hide();
+ },
+
+ show: function(parentElement)
+ {
+ this._visible = true;
+ if (parentElement && parentElement !== this.element.parentNode) {
+ this.detach();
+ parentElement.appendChild(this.element);
+ }
+ if (!this.element.parentNode && this.attach)
+ this.attach();
+ this.element.addStyleClass("visible");
+ },
+
+ hide: function()
+ {
+ this.element.removeStyleClass("visible");
+ this._visible = false;
+ },
+
+ detach: function()
+ {
+ if (this.element.parentNode)
+ this.element.parentNode.removeChild(this.element);
+ }
+}
+
+WebInspector.View.prototype.__proto__ = WebInspector.Object.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/WatchExpressionsSidebarPane.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/WatchExpressionsSidebarPane.js
new file mode 100644
index 0000000..b568939
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/WatchExpressionsSidebarPane.js
@@ -0,0 +1,274 @@
+/*
+ * Copyright (C) IBM Corp. 2009 All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of IBM Corp. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+WebInspector.WatchExpressionsSidebarPane = function()
+{
+ WebInspector.SidebarPane.call(this, WebInspector.UIString("Watch Expressions"));
+
+ this.section = new WebInspector.WatchExpressionsSection();
+
+ this.bodyElement.appendChild(this.section.element);
+
+ var addElement = document.createElement("button");
+ addElement.setAttribute("type", "button");
+ addElement.textContent = WebInspector.UIString("Add");
+ addElement.addEventListener("click", this.section.addExpression.bind(this.section), false);
+
+ var refreshElement = document.createElement("button");
+ refreshElement.setAttribute("type", "button");
+ refreshElement.textContent = WebInspector.UIString("Refresh");
+ refreshElement.addEventListener("click", this.section.update.bind(this.section), false);
+
+ var centerElement = document.createElement("div");
+ centerElement.addStyleClass("watch-expressions-buttons-container");
+ centerElement.appendChild(addElement);
+ centerElement.appendChild(refreshElement);
+ this.bodyElement.appendChild(centerElement);
+
+ this.expanded = this.section.loadSavedExpressions().length > 0;
+ this.onexpand = this.refreshExpressions.bind(this);
+}
+
+WebInspector.WatchExpressionsSidebarPane.prototype = {
+ refreshExpressions: function()
+ {
+ this.section.update();
+ }
+}
+
+WebInspector.WatchExpressionsSidebarPane.prototype.__proto__ = WebInspector.SidebarPane.prototype;
+
+WebInspector.WatchExpressionsSection = function()
+{
+ WebInspector.ObjectPropertiesSection.call(this);
+
+ this.watchExpressions = this.loadSavedExpressions();
+
+ this.headerElement.className = "hidden";
+ this.editable = true;
+ this.expanded = true;
+ this.propertiesElement.addStyleClass("watch-expressions");
+}
+
+WebInspector.WatchExpressionsSection.NewWatchExpression = "\xA0";
+
+WebInspector.WatchExpressionsSection.prototype = {
+ update: function()
+ {
+ function appendResult(expression, watchIndex, result, exception)
+ {
+ // The null check catches some other cases, like null itself, and NaN
+ if ((typeof result !== "object") || (result == null))
+ result = new WebInspector.ObjectProxy(null, [], 0, String(result), false);
+
+ var property = new WebInspector.ObjectPropertyProxy(expression, result);
+ property.watchIndex = watchIndex;
+
+ // For newly added, empty expressions, set description to "",
+ // since otherwise you get DOMWindow
+ if (property.name === WebInspector.WatchExpressionsSection.NewWatchExpression)
+ property.value.description = "";
+
+ // To clarify what's going on here:
+ // In the outer function, we calculate the number of properties
+ // that we're going to be updating, and set that in the
+ // propertyCount variable.
+ // In this function, we test to see when we are processing the
+ // last property, and then call the superclass's updateProperties()
+ // method to get all the properties refreshed at once.
+ properties.push(property);
+
+ if (properties.length == propertyCount)
+ this.updateProperties(properties, WebInspector.WatchExpressionTreeElement, WebInspector.WatchExpressionsSection.CompareProperties);
+ }
+
+ var properties = [];
+
+ // Count the properties, so we known when to call this.updateProperties()
+ // in appendResult()
+ var propertyCount = 0;
+ for (var i = 0; i < this.watchExpressions.length; ++i) {
+ if (!this.watchExpressions[i])
+ continue;
+ ++propertyCount;
+ }
+
+ // Now process all the expressions, since we have the actual count,
+ // which is checked in the appendResult inner function.
+ for (var i = 0; i < this.watchExpressions.length; ++i) {
+ var expression = this.watchExpressions[i];
+ if (!expression)
+ continue;
+
+ WebInspector.console.evalInInspectedWindow("(" + expression + ")", appendResult.bind(this, expression, i));
+ }
+
+ // note this is setting the expansion of the tree, not the section;
+ // with no expressions, and expanded tree, we get some extra vertical
+ // white space
+ // FIXME: should change to use header buttons instead of the buttons
+ // at the bottom of the section, then we can add a "No Watch Expressions
+ // element when there are no watch expressions, and this issue should
+ // go away.
+ this.expanded = (propertyCount != 0);
+ },
+
+ addExpression: function()
+ {
+ this.watchExpressions.push(WebInspector.WatchExpressionsSection.NewWatchExpression);
+ this.update();
+
+ // After update(), the new empty expression to be edited
+ // will be in the tree, but we have to find it.
+ treeElement = this.findAddedTreeElement();
+ if (treeElement)
+ treeElement.startEditing();
+ },
+
+ updateExpression: function(element, value)
+ {
+ this.watchExpressions[element.property.watchIndex] = value;
+ this.saveExpressions();
+ this.update();
+ },
+
+ findAddedTreeElement: function()
+ {
+ var children = this.propertiesTreeOutline.children;
+ for (var i = 0; i < children.length; ++i)
+ if (children[i].property.name === WebInspector.WatchExpressionsSection.NewWatchExpression)
+ return children[i];
+ },
+
+ loadSavedExpressions: function()
+ {
+ var json = InspectorController.setting("watchExpressions");
+ if (!json)
+ return [];
+
+ try {
+ json = JSON.parse(json);
+ } catch(e) {
+ return [];
+ }
+
+ return json.expressions || [];
+ },
+
+ saveExpressions: function()
+ {
+ var toSave = [];
+ for (var i = 0; i < this.watchExpressions.length; i++)
+ if (this.watchExpressions[i])
+ toSave.push(this.watchExpressions[i]);
+
+ var json = JSON.stringify({expressions: toSave});
+ InspectorController.setSetting("watchExpressions", json);
+
+ return toSave.length;
+ }
+}
+
+WebInspector.WatchExpressionsSection.prototype.__proto__ = WebInspector.ObjectPropertiesSection.prototype;
+
+WebInspector.WatchExpressionsSection.CompareProperties = function(propertyA, propertyB)
+{
+ if (propertyA.watchIndex == propertyB.watchIndex)
+ return 0;
+ else if (propertyA.watchIndex < propertyB.watchIndex)
+ return -1;
+ else
+ return 1;
+}
+
+WebInspector.WatchExpressionTreeElement = function(property)
+{
+ WebInspector.ObjectPropertyTreeElement.call(this, property);
+}
+
+WebInspector.WatchExpressionTreeElement.prototype = {
+ update: function()
+ {
+ WebInspector.ObjectPropertyTreeElement.prototype.update.call(this);
+
+ var deleteButton = document.createElement("input");
+ deleteButton.type = "button";
+ deleteButton.title = WebInspector.UIString("Delete watch expression.");
+ deleteButton.addStyleClass("enabled-button");
+ deleteButton.addStyleClass("delete-button");
+ deleteButton.addEventListener("click", this._deleteButtonClicked.bind(this), false);
+
+ this.listItemElement.insertBefore(deleteButton, this.listItemElement.firstChild);
+ },
+
+ _deleteButtonClicked: function()
+ {
+ this.treeOutline.section.updateExpression(this, null);
+ },
+
+ startEditing: function()
+ {
+ if (WebInspector.isBeingEdited(this.nameElement) || !this.treeOutline.section.editable)
+ return;
+
+ this.nameElement.textContent = this.property.name.trimWhitespace();
+
+ var context = { expanded: this.expanded };
+
+ // collapse temporarily, if required
+ this.hasChildren = false;
+
+ this.listItemElement.addStyleClass("editing-sub-part");
+
+ WebInspector.startEditing(this.nameElement, this.editingCommitted.bind(this), this.editingCancelled.bind(this), context);
+ },
+
+ editingCancelled: function(element, context)
+ {
+ if (!this.nameElement.textContent)
+ this.treeOutline.section.updateExpression(this, null);
+
+ this.update();
+ this.editingEnded(context);
+ },
+
+ applyExpression: function(expression, updateInterface)
+ {
+ expression = expression.trimWhitespace();
+
+ if (!expression)
+ expression = null;
+
+ this.property.name = expression;
+ this.treeOutline.section.updateExpression(this, expression);
+ }
+}
+
+WebInspector.WatchExpressionTreeElement.prototype.__proto__ = WebInspector.ObjectPropertyTreeElement.prototype;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/base.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/base.js
new file mode 100644
index 0000000..1a76aee
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/base.js
@@ -0,0 +1,1015 @@
+// Copyright 2006 Google Inc.
+// All Rights Reserved.
+//
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions
+// are met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above copyright
+// notice, this list of conditions and the following disclaimer in
+// the documentation and/or other materials provided with the
+// distribution.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
+// FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
+// COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
+// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
+// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+// LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
+// ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+// POSSIBILITY OF SUCH DAMAGE.
+
+// NOTE: This file has been changed from the one on doctype. The following
+// changes were made:
+// - Removed goog.globalEval because it calls eval() which is not allowed from
+// inside v8 extensions. If we ever need to use globalEval, we will need to
+// find a way to work around this problem.
+// - Remove Function.prototype.apply() emulation for the same reason. This one
+// is useless anyway because V8 supports apply() natively.
+
+/**
+ * @fileoverview Bootstrap for the Google JS Library
+ */
+
+/**
+ * @define {boolean} Overridden to true by the compiler when
+ * --mark_as_compiled is specified.
+ */
+var COMPILED = true;
+
+
+/**
+ * Base namespace for the Google JS library. Checks to see goog is
+ * already defined in the current scope before assigning to prevent
+ * clobbering if base.js is loaded more than once.
+ */
+var goog = {}; // Check to see if already defined in current scope
+
+
+/**
+ * Reference to the global context. In most cases this will be 'window'.
+ */
+goog.global = this;
+
+
+/**
+ * Indicates whether or not we can call 'eval' directly to eval code in the
+ * global scope. Set to a Boolean by the first call to goog.globalEval (which
+ * empirically tests whether eval works for globals). @see goog.globalEval
+ * @type {boolean?}
+ * @private
+ */
+goog.evalWorksForGlobals_ = null;
+
+
+/**
+ * Creates object stubs for a namespace. When present in a file, goog.provide
+ * also indicates that the file defines the indicated object.
+ * @param {string} name name of the object that this file defines.
+ */
+goog.provide = function(name) {
+ if (!COMPILED) {
+ // Ensure that the same namespace isn't provided twice. This is intended
+ // to teach new developers that 'goog.provide' is effectively a variable
+ // declaration. And when JSCompiler transforms goog.provide into a real
+ // variable declaration, the compiled JS should work the same as the raw
+ // JS--even when the raw JS uses goog.provide incorrectly.
+ if (goog.getObjectByName(name) && !goog.implicitNamespaces_[name]) {
+ throw 'Namespace "' + name + '" already declared.';
+ }
+
+ var namespace = name;
+ while ((namespace = namespace.substr(0, namespace.lastIndexOf('.')))) {
+ goog.implicitNamespaces_[namespace] = true;
+ }
+ }
+
+ goog.exportPath_(name);
+};
+
+
+if (!COMPILED) {
+ /**
+ * Namespaces implicitly defined by goog.provide. For example,
+ * goog.provide('goog.events.Event') implicitly declares
+ * that 'goog' and 'goog.events' must be namespaces.
+ *
+ * @type {Object}
+ * @private
+ */
+ goog.implicitNamespaces_ = {};
+}
+
+
+/**
+ * Builds an object structure for the provided namespace path,
+ * ensuring that names that already exist are not overwritten. For
+ * example:
+ * "a.b.c" -> a = {};a.b={};a.b.c={};
+ * Used by goog.provide and goog.exportSymbol.
+ * @param {string} name name of the object that this file defines.
+ * @param {Object} opt_object the object to expose at the end of the path.
+ * @private
+ */
+goog.exportPath_ = function(name, opt_object) {
+ var parts = name.split('.');
+ var cur = goog.global;
+ var part;
+
+ // Internet Explorer exhibits strange behavior when throwing errors from
+ // methods externed in this manner. See the testExportSymbolExceptions in
+ // base_test.html for an example.
+ if (!(parts[0] in cur) && cur.execScript) {
+ cur.execScript('var ' + parts[0]);
+ }
+
+ // Parentheses added to eliminate strict JS warning in Firefox.
+ while ((part = parts.shift())) {
+ if (!parts.length && goog.isDef(opt_object)) {
+ // last part and we have an object; use it
+ cur[part] = opt_object;
+ } else if (cur[part]) {
+ cur = cur[part];
+ } else {
+ cur = cur[part] = {};
+ }
+ }
+};
+
+
+/**
+ * Returns an object based on its fully qualified name
+ * @param {string} name The fully qualified name.
+ * @return {Object?} The object or, if not found, null.
+ */
+goog.getObjectByName = function(name) {
+ var parts = name.split('.');
+ var cur = goog.global;
+ for (var part; part = parts.shift(); ) {
+ if (cur[part]) {
+ cur = cur[part];
+ } else {
+ return null;
+ }
+ }
+ return cur;
+};
+
+
+/**
+ * Globalizes a whole namespace, such as goog or goog.lang.
+ *
+ * @param {Object} obj The namespace to globalize.
+ * @param {Object} opt_global The object to add the properties to.
+ * @deprecated Properties may be explicitly exported to the global scope, but
+ * this should no longer be done in bulk.
+ */
+goog.globalize = function(obj, opt_global) {
+ var global = opt_global || goog.global;
+ for (var x in obj) {
+ global[x] = obj[x];
+ }
+};
+
+
+/**
+ * Adds a dependency from a file to the files it requires.
+ * @param {string} relPath The path to the js file.
+ * @param {Array} provides An array of strings with the names of the objects
+ * this file provides.
+ * @param {Array} requires An array of strings with the names of the objects
+ * this file requires.
+ */
+goog.addDependency = function(relPath, provides, requires) {
+ if (!COMPILED) {
+ var provide, require;
+ var path = relPath.replace(/\\/g, '/');
+ var deps = goog.dependencies_;
+ for (var i = 0; provide = provides[i]; i++) {
+ deps.nameToPath[provide] = path;
+ if (!(path in deps.pathToNames)) {
+ deps.pathToNames[path] = {};
+ }
+ deps.pathToNames[path][provide] = true;
+ }
+ for (var j = 0; require = requires[j]; j++) {
+ if (!(path in deps.requires)) {
+ deps.requires[path] = {};
+ }
+ deps.requires[path][require] = true;
+ }
+ }
+};
+
+
+/**
+ * Implements a system for the dynamic resolution of dependencies
+ * that works in parallel with the BUILD system.
+ * @param {string} rule Rule to include, in the form goog.package.part.
+ */
+goog.require = function(rule) {
+
+ // if the object already exists we do not need do do anything
+ if (!COMPILED) {
+ if (goog.getObjectByName(rule)) {
+ return;
+ }
+ var path = goog.getPathFromDeps_(rule);
+ if (path) {
+ goog.included_[path] = true;
+ goog.writeScripts_();
+ } else {
+ // NOTE(nicksantos): We could throw an error, but this would break
+ // legacy users that depended on this failing silently. Instead, the
+ // compiler should warn us when there are invalid goog.require calls.
+ }
+ }
+};
+
+
+/**
+ * Path for included scripts
+ * @type {string}
+ */
+goog.basePath = '';
+
+
+/**
+ * Null function used for default values of callbacks, etc.
+ * @type {Function}
+ */
+goog.nullFunction = function() {};
+
+
+/**
+ * When defining a class Foo with an abstract method bar(), you can do:
+ *
+ * Foo.prototype.bar = goog.abstractMethod
+ *
+ * Now if a subclass of Foo fails to override bar(), an error
+ * will be thrown when bar() is invoked.
+ *
+ * Note: This does not take the name of the function to override as
+ * an argument because that would make it more difficult to obfuscate
+ * our JavaScript code.
+ *
+ * @throws {Error} when invoked to indicate the method should be
+ * overridden.
+ */
+goog.abstractMethod = function() {
+ throw Error('unimplemented abstract method');
+};
+
+
+if (!COMPILED) {
+ /**
+ * Object used to keep track of urls that have already been added. This
+ * record allows the prevention of circular dependencies.
+ * @type {Object}
+ * @private
+ */
+ goog.included_ = {};
+
+
+ /**
+ * This object is used to keep track of dependencies and other data that is
+ * used for loading scripts
+ * @private
+ * @type {Object}
+ */
+ goog.dependencies_ = {
+ pathToNames: {}, // 1 to many
+ nameToPath: {}, // 1 to 1
+ requires: {}, // 1 to many
+ visited: {}, // used when resolving dependencies to prevent us from
+ // visiting the file twice
+ written: {} // used to keep track of script files we have written
+ };
+
+
+ /**
+ * Tries to detect the base path of the base.js script that bootstraps
+ * Google JS Library
+ * @private
+ */
+ goog.findBasePath_ = function() {
+ var doc = goog.global.document;
+ if (typeof doc == 'undefined') {
+ return;
+ }
+ if (goog.global.GOOG_BASE_PATH) {
+ goog.basePath = goog.global.GOOG_BASE_PATH;
+ return;
+ } else {
+ goog.global.GOOG_BASE_PATH = null;
+ }
+ var scripts = doc.getElementsByTagName('script');
+ for (var script, i = 0; script = scripts[i]; i++) {
+ var src = script.src;
+ var l = src.length;
+ if (src.substr(l - 7) == 'base.js') {
+ goog.basePath = src.substr(0, l - 7);
+ return;
+ }
+ }
+ };
+
+
+ /**
+ * Writes a script tag if, and only if, that script hasn't already been added
+ * to the document. (Must be called at execution time)
+ * @param {string} src Script source.
+ * @private
+ */
+ goog.writeScriptTag_ = function(src) {
+ var doc = goog.global.document;
+ if (typeof doc != 'undefined' &&
+ !goog.dependencies_.written[src]) {
+ goog.dependencies_.written[src] = true;
+ doc.write('<script type="text/javascript" src="' +
+ src + '"></' + 'script>');
+ }
+ };
+
+
+ /**
+ * Resolves dependencies based on the dependencies added using addDependency
+ * and calls writeScriptTag_ in the correct order.
+ * @private
+ */
+ goog.writeScripts_ = function() {
+ // the scripts we need to write this time
+ var scripts = [];
+ var seenScript = {};
+ var deps = goog.dependencies_;
+
+ function visitNode(path) {
+ if (path in deps.written) {
+ return;
+ }
+
+ // we have already visited this one. We can get here if we have cyclic
+ // dependencies
+ if (path in deps.visited) {
+ if (!(path in seenScript)) {
+ seenScript[path] = true;
+ scripts.push(path);
+ }
+ return;
+ }
+
+ deps.visited[path] = true;
+
+ if (path in deps.requires) {
+ for (var requireName in deps.requires[path]) {
+ visitNode(deps.nameToPath[requireName]);
+ }
+ }
+
+ if (!(path in seenScript)) {
+ seenScript[path] = true;
+ scripts.push(path);
+ }
+ }
+
+ for (var path in goog.included_) {
+ if (!deps.written[path]) {
+ visitNode(path);
+ }
+ }
+
+ for (var i = 0; i < scripts.length; i++) {
+ if (scripts[i]) {
+ goog.writeScriptTag_(goog.basePath + scripts[i]);
+ } else {
+ throw Error('Undefined script input');
+ }
+ }
+ };
+
+
+ /**
+ * Looks at the dependency rules and tries to determine the script file that
+ * fulfills a particular rule.
+ * @param {string} rule In the form goog.namespace.Class or project.script.
+ * @return {string?} Url corresponding to the rule, or null.
+ * @private
+ */
+ goog.getPathFromDeps_ = function(rule) {
+ if (rule in goog.dependencies_.nameToPath) {
+ return goog.dependencies_.nameToPath[rule];
+ } else {
+ return null;
+ }
+ };
+
+ goog.findBasePath_();
+ goog.writeScriptTag_(goog.basePath + 'deps.js');
+}
+
+
+
+//==============================================================================
+// Language Enhancements
+//==============================================================================
+
+
+/**
+ * This is a "fixed" version of the typeof operator. It differs from the typeof
+ * operator in such a way that null returns 'null' and arrays return 'array'.
+ * @param {*} value The value to get the type of.
+ * @return {string} The name of the type.
+ */
+goog.typeOf = function(value) {
+ var s = typeof value;
+ if (s == 'object') {
+ if (value) {
+ // We cannot use constructor == Array or instanceof Array because
+ // different frames have different Array objects. In IE6, if the iframe
+ // where the array was created is destroyed, the array loses its
+ // prototype. Then dereferencing val.splice here throws an exception, so
+ // we can't use goog.isFunction. Calling typeof directly returns 'unknown'
+ // so that will work. In this case, this function will return false and
+ // most array functions will still work because the array is still
+ // array-like (supports length and []) even though it has lost its
+ // prototype. Custom object cannot have non enumerable length and
+ // NodeLists don't have a slice method.
+ if (typeof value.length == 'number' &&
+ typeof value.splice != 'undefined' &&
+ !goog.propertyIsEnumerable_(value, 'length')) {
+ return 'array';
+ }
+
+ // IE in cross-window calls does not correctly marshal the function type
+ // (it appears just as an object) so we cannot use just typeof val ==
+ // 'function'. However, if the object has a call property, it is a
+ // function.
+ if (typeof value.call != 'undefined') {
+ return 'function';
+ }
+ } else {
+ return 'null';
+ }
+
+ // In Safari typeof nodeList returns function. We would like to return
+ // object for those and we can detect an invalid function by making sure that
+ // the function object has a call method
+ } else if (s == 'function' && typeof value.call == 'undefined') {
+ return 'object';
+ }
+ return s;
+};
+
+if (Object.prototype.propertyIsEnumerable) {
+ /**
+ * Safe way to test whether a property is enumarable. It allows testing
+ * for enumarable on objects where 'propertyIsEnumerable' is overridden or
+ * does not exist (like DOM nodes in IE).
+ * @param {Object} object The object to test if the property is enumerable.
+ * @param {string} propName The property name to check for.
+ * @return {boolean} True if the property is enumarable.
+ * @private
+ */
+ goog.propertyIsEnumerable_ = function(object, propName) {
+ return Object.prototype.propertyIsEnumerable.call(object, propName);
+ };
+} else {
+ /**
+ * Safe way to test whether a property is enumarable. It allows testing
+ * for enumarable on objects where 'propertyIsEnumerable' is overridden or
+ * does not exist (like DOM nodes in IE).
+ * @param {Object} object The object to test if the property is enumerable.
+ * @param {string} propName The property name to check for.
+ * @return {boolean} True if the property is enumarable.
+ * @private
+ */
+ goog.propertyIsEnumerable_ = function(object, propName) {
+ // KJS in Safari 2 is not ECMAScript compatible and lacks crucial methods
+ // such as propertyIsEnumerable. We therefore use a workaround.
+ // Does anyone know a more efficient work around?
+ if (propName in object) {
+ for (var key in object) {
+ if (key == propName) {
+ return true;
+ }
+ }
+ }
+ return false;
+ };
+}
+
+/**
+ * Returns true if the specified value is not |undefined|.
+ * WARNING: Do not use this to test if an object has a property. Use the in
+ * operator instead.
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is defined.
+ */
+goog.isDef = function(val) {
+ return typeof val != 'undefined';
+};
+
+
+/**
+ * Returns true if the specified value is |null|
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is null.
+ */
+goog.isNull = function(val) {
+ return val === null;
+};
+
+
+/**
+ * Returns true if the specified value is defined and not null
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is defined and not null.
+ */
+goog.isDefAndNotNull = function(val) {
+ return goog.isDef(val) && !goog.isNull(val);
+};
+
+
+/**
+ * Returns true if the specified value is an array
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is an array.
+ */
+goog.isArray = function(val) {
+ return goog.typeOf(val) == 'array';
+};
+
+
+/**
+ * Returns true if the object looks like an array. To qualify as array like
+ * the value needs to be either a NodeList or an object with a Number length
+ * property.
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is an array.
+ */
+goog.isArrayLike = function(val) {
+ var type = goog.typeOf(val);
+ return type == 'array' || type == 'object' && typeof val.length == 'number';
+};
+
+
+/**
+ * Returns true if the object looks like a Date. To qualify as Date-like
+ * the value needs to be an object and have a getFullYear() function.
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is a like a Date.
+ */
+goog.isDateLike = function(val) {
+ return goog.isObject(val) && typeof val.getFullYear == 'function';
+};
+
+
+/**
+ * Returns true if the specified value is a string
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is a string.
+ */
+goog.isString = function(val) {
+ return typeof val == 'string';
+};
+
+
+/**
+ * Returns true if the specified value is a boolean
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is boolean.
+ */
+goog.isBoolean = function(val) {
+ return typeof val == 'boolean';
+};
+
+
+/**
+ * Returns true if the specified value is a number
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is a number.
+ */
+goog.isNumber = function(val) {
+ return typeof val == 'number';
+};
+
+
+/**
+ * Returns true if the specified value is a function
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is a function.
+ */
+goog.isFunction = function(val) {
+ return goog.typeOf(val) == 'function';
+};
+
+
+/**
+ * Returns true if the specified value is an object. This includes arrays
+ * and functions.
+ * @param {*} val Variable to test.
+ * @return {boolean} Whether variable is an object.
+ */
+goog.isObject = function(val) {
+ var type = goog.typeOf(val);
+ return type == 'object' || type == 'array' || type == 'function';
+};
+
+
+/**
+ * Adds a hash code field to an object. The hash code is unique for the
+ * given object.
+ * @param {Object} obj The object to get the hash code for.
+ * @return {number} The hash code for the object.
+ */
+goog.getHashCode = function(obj) {
+ // In IE, DOM nodes do not extend Object so they do not have this method.
+ // we need to check hasOwnProperty because the proto might have this set.
+
+ if (obj.hasOwnProperty && obj.hasOwnProperty(goog.HASH_CODE_PROPERTY_)) {
+ return obj[goog.HASH_CODE_PROPERTY_];
+ }
+ if (!obj[goog.HASH_CODE_PROPERTY_]) {
+ obj[goog.HASH_CODE_PROPERTY_] = ++goog.hashCodeCounter_;
+ }
+ return obj[goog.HASH_CODE_PROPERTY_];
+};
+
+
+/**
+ * Removes the hash code field from an object.
+ * @param {Object} obj The object to remove the field from.
+ */
+goog.removeHashCode = function(obj) {
+ // DOM nodes in IE are not instance of Object and throws exception
+ // for delete. Instead we try to use removeAttribute
+ if ('removeAttribute' in obj) {
+ obj.removeAttribute(goog.HASH_CODE_PROPERTY_);
+ }
+ /** @preserveTry */
+ try {
+ delete obj[goog.HASH_CODE_PROPERTY_];
+ } catch (ex) {
+ }
+};
+
+
+/**
+ * {String} Name for hash code property
+ * @private
+ */
+goog.HASH_CODE_PROPERTY_ = 'goog_hashCode_';
+
+
+/**
+ * @type {number} Counter for hash codes.
+ * @private
+ */
+goog.hashCodeCounter_ = 0;
+
+
+/**
+ * Clone an object/array (recursively)
+ * @param {Object} proto Object to clone.
+ * @return {Object} Clone of x;.
+ */
+goog.cloneObject = function(proto) {
+ var type = goog.typeOf(proto);
+ if (type == 'object' || type == 'array') {
+ if (proto.clone) {
+ return proto.clone();
+ }
+ var clone = type == 'array' ? [] : {};
+ for (var key in proto) {
+ clone[key] = goog.cloneObject(proto[key]);
+ }
+ return clone;
+ }
+
+ return proto;
+};
+
+
+/**
+ * Partially applies this function to a particular 'this object' and zero or
+ * more arguments. The result is a new function with some arguments of the first
+ * function pre-filled and the value of |this| 'pre-specified'.<br><br>
+ *
+ * Remaining arguments specified at call-time are appended to the pre-
+ * specified ones.<br><br>
+ *
+ * Also see: {@link #partial}.<br><br>
+ *
+ * Note that bind and partial are optimized such that repeated calls to it do
+ * not create more than one function object, so there is no additional cost for
+ * something like:<br>
+ *
+ * <pre>var g = bind(f, obj);
+ * var h = partial(g, 1, 2, 3);
+ * var k = partial(h, a, b, c);</pre>
+ *
+ * Usage:
+ * <pre>var barMethBound = bind(myFunction, myObj, 'arg1', 'arg2');
+ * barMethBound('arg3', 'arg4');</pre>
+ *
+ * @param {Function} fn A function to partially apply.
+ * @param {Object} self Specifies the object which |this| should point to
+ * when the function is run. If the value is null or undefined, it will
+ * default to the global object.
+ * @param {Object} var_args Additional arguments that are partially
+ * applied to the function.
+ *
+ * @return {Function} A partially-applied form of the function bind() was
+ * invoked as a method of.
+ */
+goog.bind = function(fn, self, var_args) {
+ var boundArgs = fn.boundArgs_;
+
+ if (arguments.length > 2) {
+ var args = Array.prototype.slice.call(arguments, 2);
+ if (boundArgs) {
+ args.unshift.apply(args, boundArgs);
+ }
+ boundArgs = args;
+ }
+
+ self = fn.boundSelf_ || self;
+ fn = fn.boundFn_ || fn;
+
+ var newfn;
+ var context = self || goog.global;
+
+ if (boundArgs) {
+ newfn = function() {
+ // Combine the static args and the new args into one big array
+ var args = Array.prototype.slice.call(arguments);
+ args.unshift.apply(args, boundArgs);
+ return fn.apply(context, args);
+ }
+ } else {
+ newfn = function() {
+ return fn.apply(context, arguments);
+ }
+ }
+
+ newfn.boundArgs_ = boundArgs;
+ newfn.boundSelf_ = self;
+ newfn.boundFn_ = fn;
+
+ return newfn;
+};
+
+
+/**
+ * Like bind(), except that a 'this object' is not required. Useful when the
+ * target function is already bound.
+ *
+ * Usage:
+ * var g = partial(f, arg1, arg2);
+ * g(arg3, arg4);
+ *
+ * @param {Function} fn A function to partially apply.
+ * @param {Object} var_args Additional arguments that are partially
+ * applied to fn.
+ * @return {Function} A partially-applied form of the function bind() was
+ * invoked as a method of.
+ */
+goog.partial = function(fn, var_args) {
+ var args = Array.prototype.slice.call(arguments, 1);
+ args.unshift(fn, null);
+ return goog.bind.apply(null, args);
+};
+
+
+/**
+ * Copies all the members of a source object to a target object.
+ * This is deprecated. Use goog.object.extend instead.
+ * @param {Object} target Target.
+ * @param {Object} source Source.
+ * @deprecated
+ */
+goog.mixin = function(target, source) {
+ for (var x in source) {
+ target[x] = source[x];
+ }
+
+ // For IE the for-in-loop does not contain any properties that are not
+ // enumerable on the prototype object (for example, isPrototypeOf from
+ // Object.prototype) but also it will not include 'replace' on objects that
+ // extend String and change 'replace' (not that it is common for anyone to
+ // extend anything except Object).
+};
+
+
+/**
+ * A simple wrapper for new Date().getTime().
+ *
+ * @return {number} An integer value representing the number of milliseconds
+ * between midnight, January 1, 1970 and the current time.
+ */
+goog.now = Date.now || (function() {
+ return new Date().getTime();
+});
+
+
+/**
+ * Abstract implementation of goog.getMsg for use with localized messages
+ * @param {string} str Translatable string, places holders in the form.{$foo}
+ * @param {Object} opt_values Map of place holder name to value.
+ */
+goog.getMsg = function(str, opt_values) {
+ var values = opt_values || {};
+ for (var key in values) {
+ str = str.replace(new RegExp('\\{\\$' + key + '\\}', 'gi'), values[key]);
+ }
+ return str;
+};
+
+
+/**
+ * Exposes an unobfuscated global namespace path for the given object.
+ * Note that fields of the exported object *will* be obfuscated,
+ * unless they are exported in turn via this function or
+ * goog.exportProperty
+ *
+ * <p>Also handy for making public items that are defined in anonymous
+ * closures.
+ *
+ * ex. goog.exportSymbol('Foo', Foo);
+ *
+ * ex. goog.exportSymbol('public.path.Foo.staticFunction',
+ * Foo.staticFunction);
+ * public.path.Foo.staticFunction();
+ *
+ * ex. goog.exportSymbol('public.path.Foo.prototype.myMethod',
+ * Foo.prototype.myMethod);
+ * new public.path.Foo().myMethod();
+ *
+ * @param {string} publicPath Unobfuscated name to export.
+ * @param {Object} object Object the name should point to.
+ */
+goog.exportSymbol = function(publicPath, object) {
+ goog.exportPath_(publicPath, object);
+};
+
+
+/**
+ * Exports a property unobfuscated into the object's namespace.
+ * ex. goog.exportProperty(Foo, 'staticFunction', Foo.staticFunction);
+ * ex. goog.exportProperty(Foo.prototype, 'myMethod', Foo.prototype.myMethod);
+ * @param {Object} object Object whose static property is being exported.
+ * @param {string} publicName Unobfuscated name to export.
+ * @param {Object} symbol Object the name should point to.
+ */
+goog.exportProperty = function(object, publicName, symbol) {
+ object[publicName] = symbol;
+};
+
+
+
+//==============================================================================
+// Extending Function
+//==============================================================================
+
+
+/**
+ * An alias to the {@link goog.bind()} global function.
+ *
+ * Usage:
+ * var g = f.bind(obj, arg1, arg2);
+ * g(arg3, arg4);
+ *
+ * @param {Object} self Specifies the object to which |this| should point
+ * when the function is run. If the value is null or undefined, it will
+ * default to the global object.
+ * @param {Object} var_args Additional arguments that are partially
+ * applied to fn.
+ * @return {Function} A partially-applied form of the Function on which bind()
+ * was invoked as a method.
+ * @deprecated
+ */
+Function.prototype.bind = function(self, var_args) {
+ if (arguments.length > 1) {
+ var args = Array.prototype.slice.call(arguments, 1);
+ args.unshift(this, self);
+ return goog.bind.apply(null, args);
+ } else {
+ return goog.bind(this, self);
+ }
+};
+
+
+/**
+ * An alias to the {@link goog.partial()} global function.
+ *
+ * Usage:
+ * var g = f.partial(arg1, arg2);
+ * g(arg3, arg4);
+ *
+ * @param {Object} var_args Additional arguments that are partially
+ * applied to fn.
+ * @return {Function} A partially-applied form of the function partial() was
+ * invoked as a method of.
+ * @deprecated
+ */
+Function.prototype.partial = function(var_args) {
+ var args = Array.prototype.slice.call(arguments);
+ args.unshift(this, null);
+ return goog.bind.apply(null, args);
+};
+
+
+/**
+ * Inherit the prototype methods from one constructor into another.
+ *
+ * Usage:
+ * <pre>
+ * function ParentClass(a, b) { }
+ * ParentClass.prototype.foo = function(a) { }
+ *
+ * function ChildClass(a, b, c) {
+ * ParentClass.call(this, a, b);
+ * }
+ *
+ * ChildClass.inherits(ParentClass);
+ *
+ * var child = new ChildClass('a', 'b', 'see');
+ * child.foo(); // works
+ * </pre>
+ *
+ * In addition, a superclass' implementation of a method can be invoked
+ * as follows:
+ *
+ * <pre>
+ * ChildClass.prototype.foo = function(a) {
+ * ChildClass.superClass_.foo.call(this, a);
+ * // other code
+ * };
+ * </pre>
+ *
+ * @param {Function} parentCtor Parent class.
+ */
+Function.prototype.inherits = function(parentCtor) {
+ goog.inherits(this, parentCtor);
+};
+
+
+/**
+ * Static variant of Function.prototype.inherits.
+ * @param {Function} childCtor Child class.
+ * @param {Function} parentCtor Parent class.
+ */
+goog.inherits = function(childCtor, parentCtor) {
+ /** @constructor */
+ function tempCtor() {};
+ tempCtor.prototype = parentCtor.prototype;
+ childCtor.superClass_ = parentCtor.prototype;
+ childCtor.prototype = new tempCtor();
+ childCtor.prototype.constructor = childCtor;
+};
+
+
+/**
+ * Mixes in an object's properties and methods into the callee's prototype.
+ * Basically mixin based inheritance, thus providing an alternative method for
+ * adding properties and methods to a class' prototype.
+ *
+ * <pre>
+ * function X() {}
+ * X.mixin({
+ * one: 1,
+ * two: 2,
+ * three: 3,
+ * doit: function() { return this.one + this.two + this.three; }
+ * });
+ *
+ * function Y() { }
+ * Y.mixin(X.prototype);
+ * Y.prototype.four = 15;
+ * Y.prototype.doit2 = function() { return this.doit() + this.four; }
+ * });
+ *
+ * // or
+ *
+ * function Y() { }
+ * Y.inherits(X);
+ * Y.mixin({
+ * one: 10,
+ * four: 15,
+ * doit2: function() { return this.doit() + this.four; }
+ * });
+ * </pre>
+ *
+ * @param {Object} source from which to copy properties.
+ * @see goog.mixin
+ * @deprecated
+ */
+Function.prototype.mixin = function(source) {
+ goog.mixin(this.prototype, source);
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/codemap.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/codemap.js
new file mode 100644
index 0000000..404127f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/codemap.js
@@ -0,0 +1,258 @@
+// Copyright 2009 the V8 project authors. All rights reserved.
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following
+// disclaimer in the documentation and/or other materials provided
+// with the distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived
+// from this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+// Initlialize namespaces
+var devtools = devtools || {};
+devtools.profiler = devtools.profiler || {};
+
+
+/**
+ * Constructs a mapper that maps addresses into code entries.
+ *
+ * @constructor
+ */
+devtools.profiler.CodeMap = function() {
+ /**
+ * Dynamic code entries. Used for JIT compiled code.
+ */
+ this.dynamics_ = new goog.structs.SplayTree();
+
+ /**
+ * Name generator for entries having duplicate names.
+ */
+ this.dynamicsNameGen_ = new devtools.profiler.CodeMap.NameGenerator();
+
+ /**
+ * Static code entries. Used for statically compiled code.
+ */
+ this.statics_ = new goog.structs.SplayTree();
+
+ /**
+ * Libraries entries. Used for the whole static code libraries.
+ */
+ this.libraries_ = new goog.structs.SplayTree();
+
+ /**
+ * Map of memory pages occupied with static code.
+ */
+ this.pages_ = [];
+};
+
+
+/**
+ * The number of alignment bits in a page address.
+ */
+devtools.profiler.CodeMap.PAGE_ALIGNMENT = 12;
+
+
+/**
+ * Page size in bytes.
+ */
+devtools.profiler.CodeMap.PAGE_SIZE =
+ 1 << devtools.profiler.CodeMap.PAGE_ALIGNMENT;
+
+
+/**
+ * Adds a dynamic (i.e. moveable and discardable) code entry.
+ *
+ * @param {number} start The starting address.
+ * @param {devtools.profiler.CodeMap.CodeEntry} codeEntry Code entry object.
+ */
+devtools.profiler.CodeMap.prototype.addCode = function(start, codeEntry) {
+ this.dynamics_.insert(start, codeEntry);
+};
+
+
+/**
+ * Moves a dynamic code entry. Throws an exception if there is no dynamic
+ * code entry with the specified starting address.
+ *
+ * @param {number} from The starting address of the entry being moved.
+ * @param {number} to The destination address.
+ */
+devtools.profiler.CodeMap.prototype.moveCode = function(from, to) {
+ var removedNode = this.dynamics_.remove(from);
+ this.dynamics_.insert(to, removedNode.value);
+};
+
+
+/**
+ * Discards a dynamic code entry. Throws an exception if there is no dynamic
+ * code entry with the specified starting address.
+ *
+ * @param {number} start The starting address of the entry being deleted.
+ */
+devtools.profiler.CodeMap.prototype.deleteCode = function(start) {
+ var removedNode = this.dynamics_.remove(start);
+};
+
+
+/**
+ * Adds a library entry.
+ *
+ * @param {number} start The starting address.
+ * @param {devtools.profiler.CodeMap.CodeEntry} codeEntry Code entry object.
+ */
+devtools.profiler.CodeMap.prototype.addLibrary = function(
+ start, codeEntry) {
+ this.markPages_(start, start + codeEntry.size);
+ this.libraries_.insert(start, codeEntry);
+};
+
+
+/**
+ * Adds a static code entry.
+ *
+ * @param {number} start The starting address.
+ * @param {devtools.profiler.CodeMap.CodeEntry} codeEntry Code entry object.
+ */
+devtools.profiler.CodeMap.prototype.addStaticCode = function(
+ start, codeEntry) {
+ this.statics_.insert(start, codeEntry);
+};
+
+
+/**
+ * @private
+ */
+devtools.profiler.CodeMap.prototype.markPages_ = function(start, end) {
+ for (var addr = start; addr <= end;
+ addr += devtools.profiler.CodeMap.PAGE_SIZE) {
+ this.pages_[addr >>> devtools.profiler.CodeMap.PAGE_ALIGNMENT] = 1;
+ }
+};
+
+
+/**
+ * @private
+ */
+devtools.profiler.CodeMap.prototype.isAddressBelongsTo_ = function(addr, node) {
+ return addr >= node.key && addr < (node.key + node.value.size);
+};
+
+
+/**
+ * @private
+ */
+devtools.profiler.CodeMap.prototype.findInTree_ = function(tree, addr) {
+ var node = tree.findGreatestLessThan(addr);
+ return node && this.isAddressBelongsTo_(addr, node) ? node.value : null;
+};
+
+
+/**
+ * Finds a code entry that contains the specified address. Both static and
+ * dynamic code entries are considered.
+ *
+ * @param {number} addr Address.
+ */
+devtools.profiler.CodeMap.prototype.findEntry = function(addr) {
+ var pageAddr = addr >>> devtools.profiler.CodeMap.PAGE_ALIGNMENT;
+ if (pageAddr in this.pages_) {
+ // Static code entries can contain "holes" of unnamed code.
+ // In this case, the whole library is assigned to this address.
+ return this.findInTree_(this.statics_, addr) ||
+ this.findInTree_(this.libraries_, addr);
+ }
+ var min = this.dynamics_.findMin();
+ var max = this.dynamics_.findMax();
+ if (max != null && addr < (max.key + max.value.size) && addr >= min.key) {
+ var dynaEntry = this.findInTree_(this.dynamics_, addr);
+ if (dynaEntry == null) return null;
+ // Dedupe entry name.
+ if (!dynaEntry.nameUpdated_) {
+ dynaEntry.name = this.dynamicsNameGen_.getName(dynaEntry.name);
+ dynaEntry.nameUpdated_ = true;
+ }
+ return dynaEntry;
+ }
+ return null;
+};
+
+
+/**
+ * Returns an array of all dynamic code entries.
+ */
+devtools.profiler.CodeMap.prototype.getAllDynamicEntries = function() {
+ return this.dynamics_.exportValues();
+};
+
+
+/**
+ * Returns an array of all static code entries.
+ */
+devtools.profiler.CodeMap.prototype.getAllStaticEntries = function() {
+ return this.statics_.exportValues();
+};
+
+
+/**
+ * Returns an array of all libraries entries.
+ */
+devtools.profiler.CodeMap.prototype.getAllLibrariesEntries = function() {
+ return this.libraries_.exportValues();
+};
+
+
+/**
+ * Creates a code entry object.
+ *
+ * @param {number} size Code entry size in bytes.
+ * @param {string} opt_name Code entry name.
+ * @constructor
+ */
+devtools.profiler.CodeMap.CodeEntry = function(size, opt_name) {
+ this.size = size;
+ this.name = opt_name || '';
+ this.nameUpdated_ = false;
+};
+
+
+devtools.profiler.CodeMap.CodeEntry.prototype.getName = function() {
+ return this.name;
+};
+
+
+devtools.profiler.CodeMap.CodeEntry.prototype.toString = function() {
+ return this.name + ': ' + this.size.toString(16);
+};
+
+
+devtools.profiler.CodeMap.NameGenerator = function() {
+ this.knownNames_ = [];
+};
+
+
+devtools.profiler.CodeMap.NameGenerator.prototype.getName = function(name) {
+ if (!(name in this.knownNames_)) {
+ this.knownNames_[name] = 0;
+ return name;
+ }
+ var count = ++this.knownNames_[name];
+ return name + ' {' + count + '}';
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/consarray.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/consarray.js
new file mode 100644
index 0000000..c67abb7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/consarray.js
@@ -0,0 +1,93 @@
+// Copyright 2009 the V8 project authors. All rights reserved.
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following
+// disclaimer in the documentation and/or other materials provided
+// with the distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived
+// from this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+/**
+ * Constructs a ConsArray object. It is used mainly for tree traversal.
+ * In this use case we have lots of arrays that we need to iterate
+ * sequentally. The internal Array implementation is horribly slow
+ * when concatenating on large (10K items) arrays due to memory copying.
+ * That's why we avoid copying memory and insead build a linked list
+ * of arrays to iterate through.
+ *
+ * @constructor
+ */
+function ConsArray() {
+ this.tail_ = new ConsArray.Cell(null, null);
+ this.currCell_ = this.tail_;
+ this.currCellPos_ = 0;
+};
+
+
+/**
+ * Concatenates another array for iterating. Empty arrays are ignored.
+ * This operation can be safely performed during ongoing ConsArray
+ * iteration.
+ *
+ * @param {Array} arr Array to concatenate.
+ */
+ConsArray.prototype.concat = function(arr) {
+ if (arr.length > 0) {
+ this.tail_.data = arr;
+ this.tail_ = this.tail_.next = new ConsArray.Cell(null, null);
+ }
+};
+
+
+/**
+ * Whether the end of iteration is reached.
+ */
+ConsArray.prototype.atEnd = function() {
+ return this.currCell_ === null ||
+ this.currCell_.data === null ||
+ this.currCellPos_ >= this.currCell_.data.length;
+};
+
+
+/**
+ * Returns the current item, moves to the next one.
+ */
+ConsArray.prototype.next = function() {
+ var result = this.currCell_.data[this.currCellPos_++];
+ if (this.currCellPos_ >= this.currCell_.data.length) {
+ this.currCell_ = this.currCell_.next;
+ this.currCellPos_ = 0;
+ }
+ return result;
+};
+
+
+/**
+ * A cell object used for constructing a list in ConsArray.
+ *
+ * @constructor
+ */
+ConsArray.Cell = function(data, next) {
+ this.data = data;
+ this.next = next;
+};
+
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/csvparser.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/csvparser.js
new file mode 100644
index 0000000..9e58dea
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/csvparser.js
@@ -0,0 +1,98 @@
+// Copyright 2009 the V8 project authors. All rights reserved.
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following
+// disclaimer in the documentation and/or other materials provided
+// with the distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived
+// from this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+// Initlialize namespaces.
+var devtools = devtools || {};
+devtools.profiler = devtools.profiler || {};
+
+
+/**
+ * Creates a CSV lines parser.
+ */
+devtools.profiler.CsvParser = function() {
+};
+
+
+/**
+ * A regex for matching a trailing quote.
+ * @private
+ */
+devtools.profiler.CsvParser.TRAILING_QUOTE_RE_ = /\"$/;
+
+
+/**
+ * A regex for matching a double quote.
+ * @private
+ */
+devtools.profiler.CsvParser.DOUBLE_QUOTE_RE_ = /\"\"/g;
+
+
+/**
+ * Parses a line of CSV-encoded values. Returns an array of fields.
+ *
+ * @param {string} line Input line.
+ */
+devtools.profiler.CsvParser.prototype.parseLine = function(line) {
+ var insideQuotes = false;
+ var fields = [];
+ var prevPos = 0;
+ for (var i = 0, n = line.length; i < n; ++i) {
+ switch (line.charAt(i)) {
+ case ',':
+ if (!insideQuotes) {
+ fields.push(line.substring(prevPos, i));
+ prevPos = i + 1;
+ }
+ break;
+ case '"':
+ if (!insideQuotes) {
+ insideQuotes = true;
+ // Skip the leading quote.
+ prevPos++;
+ } else {
+ if (i + 1 < n && line.charAt(i + 1) != '"') {
+ insideQuotes = false;
+ } else {
+ i++;
+ }
+ }
+ break;
+ }
+ }
+ if (n > 0) {
+ fields.push(line.substring(prevPos));
+ }
+
+ for (i = 0; i < fields.length; ++i) {
+ // Eliminate trailing quotes.
+ fields[i] = fields[i].replace(devtools.profiler.CsvParser.TRAILING_QUOTE_RE_, '');
+ // Convert quoted quotes into single ones.
+ fields[i] = fields[i].replace(devtools.profiler.CsvParser.DOUBLE_QUOTE_RE_, '"');
+ }
+ return fields;
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/debugger_agent.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/debugger_agent.js
new file mode 100644
index 0000000..1d566eb
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/debugger_agent.js
@@ -0,0 +1,1490 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview Provides communication interface to remote v8 debugger. See
+ * protocol decription at http://code.google.com/p/v8/wiki/DebuggerProtocol
+ */
+goog.provide('devtools.DebuggerAgent');
+
+
+/**
+ * @constructor
+ */
+devtools.DebuggerAgent = function() {
+ RemoteDebuggerAgent.DebuggerOutput =
+ goog.bind(this.handleDebuggerOutput_, this);
+ RemoteDebuggerAgent.SetContextId =
+ goog.bind(this.setContextId_, this);
+ RemoteDebuggerAgent.DidGetActiveProfilerModules =
+ goog.bind(this.didGetActiveProfilerModules_, this);
+ RemoteDebuggerAgent.DidGetNextLogLines =
+ goog.bind(this.didGetNextLogLines_, this);
+
+ /**
+ * Id of the inspected page global context. It is used for filtering scripts.
+ * @type {number}
+ */
+ this.contextId_ = null;
+
+ /**
+ * Mapping from script id to script info.
+ * @type {Object}
+ */
+ this.parsedScripts_ = null;
+
+ /**
+ * Mapping from the request id to the devtools.BreakpointInfo for the
+ * breakpoints whose v8 ids are not set yet. These breakpoints are waiting for
+ * 'setbreakpoint' responses to learn their ids in the v8 debugger.
+ * @see #handleSetBreakpointResponse_
+ * @type {Object}
+ */
+ this.requestNumberToBreakpointInfo_ = null;
+
+ /**
+ * Information on current stack frames.
+ * @type {Array.<devtools.CallFrame>}
+ */
+ this.callFrames_ = [];
+
+ /**
+ * Whether to stop in the debugger on the exceptions.
+ * @type {boolean}
+ */
+ this.pauseOnExceptions_ = true;
+
+ /**
+ * Mapping: request sequence number->callback.
+ * @type {Object}
+ */
+ this.requestSeqToCallback_ = null;
+
+ /**
+ * Whether the scripts list has been requested.
+ * @type {boolean}
+ */
+ this.scriptsCacheInitialized_ = false;
+
+ /**
+ * Whether the scripts list should be requested next time when context id is
+ * set.
+ * @type {boolean}
+ */
+ this.requestScriptsWhenContextIdSet_ = false;
+
+ /**
+ * Active profiler modules flags.
+ * @type {number}
+ */
+ this.activeProfilerModules_ =
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_NONE;
+
+ /**
+ * Profiler processor instance.
+ * @type {devtools.profiler.Processor}
+ */
+ this.profilerProcessor_ = new devtools.profiler.Processor();
+
+ /**
+ * Container of all breakpoints set using resource URL. These breakpoints
+ * survive page reload. Breakpoints set by script id(for scripts that don't
+ * have URLs) are stored in ScriptInfo objects.
+ * @type {Object}
+ */
+ this.urlToBreakpoints_ = {};
+
+
+ /**
+ * Exception message that is shown to user while on exception break.
+ * @type {WebInspector.ConsoleMessage}
+ */
+ this.currentExceptionMessage_ = null;
+};
+
+
+/**
+ * A copy of the scope types from v8/src/mirror-delay.js
+ * @enum {number}
+ */
+devtools.DebuggerAgent.ScopeType = {
+ Global: 0,
+ Local: 1,
+ With: 2,
+ Closure: 3,
+ Catch: 4
+};
+
+
+/**
+ * A copy of enum from include/v8.h
+ * @enum {number}
+ */
+devtools.DebuggerAgent.ProfilerModules = {
+ PROFILER_MODULE_NONE: 0,
+ PROFILER_MODULE_CPU: 1,
+ PROFILER_MODULE_HEAP_STATS: 1 << 1,
+ PROFILER_MODULE_JS_CONSTRUCTORS: 1 << 2,
+ PROFILER_MODULE_HEAP_SNAPSHOT: 1 << 16
+};
+
+
+/**
+ * Resets debugger agent to its initial state.
+ */
+devtools.DebuggerAgent.prototype.reset = function() {
+ this.contextId_ = null;
+ // No need to request scripts since they all will be pushed in AfterCompile
+ // events.
+ this.requestScriptsWhenContextIdSet_ = false;
+ this.parsedScripts_ = {};
+ this.requestNumberToBreakpointInfo_ = {};
+ this.callFrames_ = [];
+ this.requestSeqToCallback_ = {};
+
+ // Profiler isn't reset because it contains no data that is
+ // specific for a particular V8 instance. All such data is
+ // managed by an agent on the Render's side.
+};
+
+
+/**
+ * Initializes scripts UI. This method is called every time Scripts panel
+ * is shown. It will send request for context id if it's not set yet.
+ */
+devtools.DebuggerAgent.prototype.initUI = function() {
+ // There can be a number of scripts from after-compile events that are
+ // pending addition into the UI.
+ for (var scriptId in this.parsedScripts_) {
+ var script = this.parsedScripts_[scriptId];
+ WebInspector.parsedScriptSource(scriptId, script.getUrl(),
+ undefined /* script source */, script.getLineOffset());
+ }
+
+ // Initialize scripts cache when Scripts panel is shown first time.
+ if (this.scriptsCacheInitialized_) {
+ return;
+ }
+ this.scriptsCacheInitialized_ = true;
+ if (this.contextId_) {
+ // We already have context id. This means that we are here from the
+ // very beginning of the page load cycle and hence will get all scripts
+ // via after-compile events. No need to request scripts for this session.
+ return;
+ }
+ // Script list should be requested only when current context id is known.
+ RemoteDebuggerAgent.GetContextId();
+ this.requestScriptsWhenContextIdSet_ = true;
+};
+
+
+/**
+ * Asynchronously requests the debugger for the script source.
+ * @param {number} scriptId Id of the script whose source should be resolved.
+ * @param {function(source:?string):void} callback Function that will be called
+ * when the source resolution is completed. 'source' parameter will be null
+ * if the resolution fails.
+ */
+devtools.DebuggerAgent.prototype.resolveScriptSource = function(
+ scriptId, callback) {
+ var script = this.parsedScripts_[scriptId];
+ if (!script || script.isUnresolved()) {
+ callback(null);
+ return;
+ }
+
+ var cmd = new devtools.DebugCommand('scripts', {
+ 'ids': [scriptId],
+ 'includeSource': true
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+ // Force v8 execution so that it gets to processing the requested command.
+ RemoteToolsAgent.ExecuteVoidJavaScript();
+
+ this.requestSeqToCallback_[cmd.getSequenceNumber()] = function(msg) {
+ if (msg.isSuccess()) {
+ var scriptJson = msg.getBody()[0];
+ callback(scriptJson.source);
+ } else {
+ callback(null);
+ }
+ };
+};
+
+
+/**
+ * Tells the v8 debugger to stop on as soon as possible.
+ */
+devtools.DebuggerAgent.prototype.pauseExecution = function() {
+ RemoteDebuggerAgent.DebugBreak();
+};
+
+
+/**
+ * @param {number} sourceId Id of the script fot the breakpoint.
+ * @param {number} line Number of the line for the breakpoint.
+ * @param {?string} condition The breakpoint condition.
+ */
+devtools.DebuggerAgent.prototype.addBreakpoint = function(
+ sourceId, line, condition) {
+ var script = this.parsedScripts_[sourceId];
+ if (!script) {
+ return;
+ }
+
+ line = devtools.DebuggerAgent.webkitToV8LineNumber_(line);
+
+ var commandArguments;
+ if (script.getUrl()) {
+ var breakpoints = this.urlToBreakpoints_[script.getUrl()];
+ if (breakpoints && breakpoints[line]) {
+ return;
+ }
+ if (!breakpoints) {
+ breakpoints = {};
+ this.urlToBreakpoints_[script.getUrl()] = breakpoints;
+ }
+
+ var breakpointInfo = new devtools.BreakpointInfo(line);
+ breakpoints[line] = breakpointInfo;
+
+ commandArguments = {
+ 'groupId': this.contextId_,
+ 'type': 'script',
+ 'target': script.getUrl(),
+ 'line': line,
+ 'condition': condition
+ };
+ } else {
+ var breakpointInfo = script.getBreakpointInfo(line);
+ if (breakpointInfo) {
+ return;
+ }
+
+ breakpointInfo = new devtools.BreakpointInfo(line);
+ script.addBreakpointInfo(breakpointInfo);
+
+ commandArguments = {
+ 'groupId': this.contextId_,
+ 'type': 'scriptId',
+ 'target': sourceId,
+ 'line': line,
+ 'condition': condition
+ };
+ }
+
+ var cmd = new devtools.DebugCommand('setbreakpoint', commandArguments);
+
+ this.requestNumberToBreakpointInfo_[cmd.getSequenceNumber()] = breakpointInfo;
+
+ devtools.DebuggerAgent.sendCommand_(cmd);
+ // Force v8 execution so that it gets to processing the requested command.
+ // It is necessary for being able to change a breakpoint just after it
+ // has been created (since we need an existing breakpoint id for that).
+ RemoteToolsAgent.ExecuteVoidJavaScript();
+};
+
+
+/**
+ * @param {number} sourceId Id of the script for the breakpoint.
+ * @param {number} line Number of the line for the breakpoint.
+ */
+devtools.DebuggerAgent.prototype.removeBreakpoint = function(sourceId, line) {
+ var script = this.parsedScripts_[sourceId];
+ if (!script) {
+ return;
+ }
+
+ line = devtools.DebuggerAgent.webkitToV8LineNumber_(line);
+
+ var breakpointInfo;
+ if (script.getUrl()) {
+ var breakpoints = this.urlToBreakpoints_[script.getUrl()];
+ breakpointInfo = breakpoints[line];
+ delete breakpoints[line];
+ } else {
+ breakpointInfo = script.getBreakpointInfo(line);
+ if (breakpointInfo) {
+ script.removeBreakpointInfo(breakpointInfo);
+ }
+ }
+
+ if (!breakpointInfo) {
+ return;
+ }
+
+ breakpointInfo.markAsRemoved();
+
+ var id = breakpointInfo.getV8Id();
+
+ // If we don't know id of this breakpoint in the v8 debugger we cannot send
+ // 'clearbreakpoint' request. In that case it will be removed in
+ // 'setbreakpoint' response handler when we learn the id.
+ if (id != -1) {
+ this.requestClearBreakpoint_(id);
+ }
+};
+
+
+/**
+ * @param {number} sourceId Id of the script for the breakpoint.
+ * @param {number} line Number of the line for the breakpoint.
+ * @param {?string} condition New breakpoint condition.
+ */
+devtools.DebuggerAgent.prototype.updateBreakpoint = function(
+ sourceId, line, condition) {
+ var script = this.parsedScripts_[sourceId];
+ if (!script) {
+ return;
+ }
+
+ line = devtools.DebuggerAgent.webkitToV8LineNumber_(line);
+
+ var breakpointInfo;
+ if (script.getUrl()) {
+ var breakpoints = this.urlToBreakpoints_[script.getUrl()];
+ breakpointInfo = breakpoints[line];
+ } else {
+ breakpointInfo = script.getBreakpointInfo(line);
+ }
+
+ var id = breakpointInfo.getV8Id();
+
+ // If we don't know id of this breakpoint in the v8 debugger we cannot send
+ // the 'changebreakpoint' request.
+ if (id != -1) {
+ // TODO(apavlov): make use of the real values for 'enabled' and
+ // 'ignoreCount' when appropriate.
+ this.requestChangeBreakpoint_(id, true, condition, null);
+ }
+};
+
+
+/**
+ * Tells the v8 debugger to step into the next statement.
+ */
+devtools.DebuggerAgent.prototype.stepIntoStatement = function() {
+ this.stepCommand_('in');
+};
+
+
+/**
+ * Tells the v8 debugger to step out of current function.
+ */
+devtools.DebuggerAgent.prototype.stepOutOfFunction = function() {
+ this.stepCommand_('out');
+};
+
+
+/**
+ * Tells the v8 debugger to step over the next statement.
+ */
+devtools.DebuggerAgent.prototype.stepOverStatement = function() {
+ this.stepCommand_('next');
+};
+
+
+/**
+ * Tells the v8 debugger to continue execution after it has been stopped on a
+ * breakpoint or an exception.
+ */
+devtools.DebuggerAgent.prototype.resumeExecution = function() {
+ this.clearExceptionMessage_();
+ var cmd = new devtools.DebugCommand('continue');
+ devtools.DebuggerAgent.sendCommand_(cmd);
+};
+
+
+/**
+ * Creates exception message and schedules it for addition to the resource upon
+ * backtrace availability.
+ * @param {string} url Resource url.
+ * @param {number} line Resource line number.
+ * @param {string} message Exception text.
+ */
+devtools.DebuggerAgent.prototype.createExceptionMessage_ = function(
+ url, line, message) {
+ this.currentExceptionMessage_ = new WebInspector.ConsoleMessage(
+ WebInspector.ConsoleMessage.MessageSource.JS,
+ WebInspector.ConsoleMessage.MessageType.Log,
+ WebInspector.ConsoleMessage.MessageLevel.Error,
+ line,
+ url,
+ 0 /* group level */,
+ 1 /* repeat count */,
+ '[Exception] ' + message);
+};
+
+
+/**
+ * Shows pending exception message that is created with createExceptionMessage_
+ * earlier.
+ */
+devtools.DebuggerAgent.prototype.showPendingExceptionMessage_ = function() {
+ if (!this.currentExceptionMessage_) {
+ return;
+ }
+ var msg = this.currentExceptionMessage_;
+ var resource = WebInspector.resourceURLMap[msg.url];
+ if (resource) {
+ msg.resource = resource;
+ WebInspector.panels.resources.addMessageToResource(resource, msg);
+ } else {
+ this.currentExceptionMessage_ = null;
+ }
+};
+
+
+/**
+ * Clears exception message from the resource.
+ */
+devtools.DebuggerAgent.prototype.clearExceptionMessage_ = function() {
+ if (this.currentExceptionMessage_) {
+ var messageElement =
+ this.currentExceptionMessage_._resourceMessageLineElement;
+ var bubble = messageElement.parentElement;
+ bubble.removeChild(messageElement);
+ if (!bubble.firstChild) {
+ // Last message in bubble removed.
+ bubble.parentElement.removeChild(bubble);
+ }
+ this.currentExceptionMessage_ = null;
+ }
+};
+
+
+/**
+ * @return {boolean} True iff the debugger will pause execution on the
+ * exceptions.
+ */
+devtools.DebuggerAgent.prototype.pauseOnExceptions = function() {
+ return this.pauseOnExceptions_;
+};
+
+
+/**
+ * Tells whether to pause in the debugger on the exceptions or not.
+ * @param {boolean} value True iff execution should be stopped in the debugger
+ * on the exceptions.
+ */
+devtools.DebuggerAgent.prototype.setPauseOnExceptions = function(value) {
+ this.pauseOnExceptions_ = value;
+};
+
+
+/**
+ * Sends 'evaluate' request to the debugger.
+ * @param {Object} arguments Request arguments map.
+ * @param {function(devtools.DebuggerMessage)} callback Callback to be called
+ * when response is received.
+ */
+devtools.DebuggerAgent.prototype.requestEvaluate = function(
+ arguments, callback) {
+ var cmd = new devtools.DebugCommand('evaluate', arguments);
+ devtools.DebuggerAgent.sendCommand_(cmd);
+ this.requestSeqToCallback_[cmd.getSequenceNumber()] = callback;
+};
+
+
+/**
+ * Sends 'lookup' request for each unresolved property of the object. When
+ * response is received the properties will be changed with their resolved
+ * values.
+ * @param {Object} object Object whose properties should be resolved.
+ * @param {function(devtools.DebuggerMessage)} Callback to be called when all
+ * children are resolved.
+ * @param {boolean} noIntrinsic Whether intrinsic properties should be included.
+ */
+devtools.DebuggerAgent.prototype.resolveChildren = function(object, callback,
+ noIntrinsic) {
+ if ('handle' in object) {
+ var result = [];
+ devtools.DebuggerAgent.formatObjectProperties_(object, result,
+ noIntrinsic);
+ callback(result);
+ } else {
+ this.requestLookup_([object.ref], function(msg) {
+ var result = [];
+ if (msg.isSuccess()) {
+ var handleToObject = msg.getBody();
+ var resolved = handleToObject[object.ref];
+ devtools.DebuggerAgent.formatObjectProperties_(resolved, result,
+ noIntrinsic);
+ callback(result);
+ } else {
+ callback([]);
+ }
+ });
+ }
+};
+
+
+/**
+ * Sends 'scope' request for the scope object to resolve its variables.
+ * @param {Object} scope Scope to be resolved.
+ * @param {function(Array.<WebInspector.ObjectPropertyProxy>)} callback
+ * Callback to be called when all scope variables are resolved.
+ */
+devtools.DebuggerAgent.prototype.resolveScope = function(scope, callback) {
+ var cmd = new devtools.DebugCommand('scope', {
+ 'frameNumber': scope.frameNumber,
+ 'number': scope.index,
+ 'compactFormat': true
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+ this.requestSeqToCallback_[cmd.getSequenceNumber()] = function(msg) {
+ var result = [];
+ if (msg.isSuccess()) {
+ var scopeObjectJson = msg.getBody().object;
+ devtools.DebuggerAgent.formatObjectProperties_(scopeObjectJson, result,
+ true /* no intrinsic */);
+ }
+ callback(result);
+ };
+};
+
+
+/**
+ * Sets up callbacks that deal with profiles processing.
+ */
+devtools.DebuggerAgent.prototype.setupProfilerProcessorCallbacks = function() {
+ // A temporary icon indicating that the profile is being processed.
+ var processingIcon = new WebInspector.SidebarTreeElement(
+ 'profile-sidebar-tree-item',
+ WebInspector.UIString('Processing...'),
+ '', null, false);
+ var profilesSidebar = WebInspector.panels.profiles.sidebarTree;
+
+ this.profilerProcessor_.setCallbacks(
+ function onProfileProcessingStarted() {
+ // Set visually empty string. Subtitle hiding is done via styles
+ // manipulation which doesn't play well with dynamic append / removal.
+ processingIcon.subtitle = ' ';
+ profilesSidebar.appendChild(processingIcon);
+ },
+ function onProfileProcessingStatus(ticksCount) {
+ processingIcon.subtitle =
+ WebInspector.UIString('%d ticks processed', ticksCount);
+ },
+ function onProfileProcessingFinished(profile) {
+ profilesSidebar.removeChild(processingIcon);
+ WebInspector.addProfile(profile);
+ // If no profile is currently shown, show the new one.
+ var profilesPanel = WebInspector.panels.profiles;
+ if (!profilesPanel.visibleView) {
+ profilesPanel.showProfile(profile);
+ }
+ }
+ );
+};
+
+
+/**
+ * Initializes profiling state.
+ */
+devtools.DebuggerAgent.prototype.initializeProfiling = function() {
+ this.setupProfilerProcessorCallbacks();
+ RemoteDebuggerAgent.GetActiveProfilerModules();
+};
+
+
+/**
+ * Starts profiling.
+ * @param {number} modules List of modules to enable.
+ */
+devtools.DebuggerAgent.prototype.startProfiling = function(modules) {
+ RemoteDebuggerAgent.StartProfiling(modules);
+ if (modules &
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_HEAP_SNAPSHOT) {
+ // Active modules will not change, instead, a snapshot will be logged.
+ RemoteDebuggerAgent.GetNextLogLines();
+ } else {
+ RemoteDebuggerAgent.GetActiveProfilerModules();
+ }
+};
+
+
+/**
+ * Stops profiling.
+ */
+devtools.DebuggerAgent.prototype.stopProfiling = function(modules) {
+ RemoteDebuggerAgent.StopProfiling(modules);
+};
+
+
+/**
+ * @param{number} scriptId
+ * @return {string} Type of the context of the script with specified id.
+ */
+devtools.DebuggerAgent.prototype.getScriptContextType = function(scriptId) {
+ return this.parsedScripts_[scriptId].getContextType();
+};
+
+
+/**
+ * Removes specified breakpoint from the v8 debugger.
+ * @param {number} breakpointId Id of the breakpoint in the v8 debugger.
+ */
+devtools.DebuggerAgent.prototype.requestClearBreakpoint_ = function(
+ breakpointId) {
+ var cmd = new devtools.DebugCommand('clearbreakpoint', {
+ 'breakpoint': breakpointId
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+};
+
+
+/**
+ * Changes breakpoint parameters in the v8 debugger.
+ * @param {number} breakpointId Id of the breakpoint in the v8 debugger.
+ * @param {boolean} enabled Whether to enable the breakpoint.
+ * @param {?string} condition New breakpoint condition.
+ * @param {number} ignoreCount New ignore count for the breakpoint.
+ */
+devtools.DebuggerAgent.prototype.requestChangeBreakpoint_ = function(
+ breakpointId, enabled, condition, ignoreCount) {
+ var cmd = new devtools.DebugCommand('changebreakpoint', {
+ 'breakpoint': breakpointId,
+ 'enabled': enabled,
+ 'condition': condition,
+ 'ignoreCount': ignoreCount
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+};
+
+
+/**
+ * Sends 'backtrace' request to v8.
+ */
+devtools.DebuggerAgent.prototype.requestBacktrace_ = function() {
+ var cmd = new devtools.DebugCommand('backtrace', {
+ 'compactFormat':true
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+};
+
+
+/**
+ * Sends command to v8 debugger.
+ * @param {devtools.DebugCommand} cmd Command to execute.
+ */
+devtools.DebuggerAgent.sendCommand_ = function(cmd) {
+ RemoteDebuggerCommandExecutor.DebuggerCommand(cmd.toJSONProtocol());
+};
+
+
+/**
+ * Tells the v8 debugger to make the next execution step.
+ * @param {string} action 'in', 'out' or 'next' action.
+ */
+devtools.DebuggerAgent.prototype.stepCommand_ = function(action) {
+ this.clearExceptionMessage_();
+ var cmd = new devtools.DebugCommand('continue', {
+ 'stepaction': action,
+ 'stepcount': 1
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+};
+
+
+/**
+ * Sends 'lookup' request to v8.
+ * @param {number} handle Handle to the object to lookup.
+ */
+devtools.DebuggerAgent.prototype.requestLookup_ = function(handles, callback) {
+ var cmd = new devtools.DebugCommand('lookup', {
+ 'compactFormat':true,
+ 'handles': handles
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+ this.requestSeqToCallback_[cmd.getSequenceNumber()] = callback;
+};
+
+
+/**
+ * Sets debugger context id for scripts filtering.
+ * @param {number} contextId Id of the inspected page global context.
+ */
+devtools.DebuggerAgent.prototype.setContextId_ = function(contextId) {
+ this.contextId_ = contextId;
+
+ // If it's the first time context id is set request scripts list.
+ if (this.requestScriptsWhenContextIdSet_) {
+ this.requestScriptsWhenContextIdSet_ = false;
+ var cmd = new devtools.DebugCommand('scripts', {
+ 'includeSource': false
+ });
+ devtools.DebuggerAgent.sendCommand_(cmd);
+ // Force v8 execution so that it gets to processing the requested command.
+ RemoteToolsAgent.ExecuteVoidJavaScript();
+
+ var debuggerAgent = this;
+ this.requestSeqToCallback_[cmd.getSequenceNumber()] = function(msg) {
+ // Handle the response iff the context id hasn't changed since the request
+ // was issued. Otherwise if the context id did change all up-to-date
+ // scripts will be pushed in after compile events and there is no need to
+ // handle the response.
+ if (contextId == debuggerAgent.contextId_) {
+ debuggerAgent.handleScriptsResponse_(msg);
+ }
+ };
+ }
+};
+
+
+/**
+ * Handles output sent by v8 debugger. The output is either asynchronous event
+ * or response to a previously sent request. See protocol definitioun for more
+ * details on the output format.
+ * @param {string} output
+ */
+devtools.DebuggerAgent.prototype.handleDebuggerOutput_ = function(output) {
+ var msg;
+ try {
+ msg = new devtools.DebuggerMessage(output);
+ } catch(e) {
+ debugPrint('Failed to handle debugger reponse:\n' + e);
+ throw e;
+ }
+
+ if (msg.getType() == 'event') {
+ if (msg.getEvent() == 'break') {
+ this.handleBreakEvent_(msg);
+ } else if (msg.getEvent() == 'exception') {
+ this.handleExceptionEvent_(msg);
+ } else if (msg.getEvent() == 'afterCompile') {
+ this.handleAfterCompileEvent_(msg);
+ }
+ } else if (msg.getType() == 'response') {
+ if (msg.getCommand() == 'scripts') {
+ this.invokeCallbackForResponse_(msg);
+ } else if (msg.getCommand() == 'setbreakpoint') {
+ this.handleSetBreakpointResponse_(msg);
+ } else if (msg.getCommand() == 'clearbreakpoint') {
+ this.handleClearBreakpointResponse_(msg);
+ } else if (msg.getCommand() == 'backtrace') {
+ this.handleBacktraceResponse_(msg);
+ } else if (msg.getCommand() == 'lookup') {
+ this.invokeCallbackForResponse_(msg);
+ } else if (msg.getCommand() == 'evaluate') {
+ this.invokeCallbackForResponse_(msg);
+ } else if (msg.getCommand() == 'scope') {
+ this.invokeCallbackForResponse_(msg);
+ }
+ }
+};
+
+
+/**
+ * @param {devtools.DebuggerMessage} msg
+ */
+devtools.DebuggerAgent.prototype.handleBreakEvent_ = function(msg) {
+ // Force scrips panel to be shown first.
+ WebInspector.currentPanel = WebInspector.panels.scripts;
+
+ var body = msg.getBody();
+
+ var line = devtools.DebuggerAgent.v8ToWwebkitLineNumber_(body.sourceLine);
+ this.requestBacktrace_();
+};
+
+
+/**
+ * @param {devtools.DebuggerMessage} msg
+ */
+devtools.DebuggerAgent.prototype.handleExceptionEvent_ = function(msg) {
+ // Force scrips panel to be shown first.
+ WebInspector.currentPanel = WebInspector.panels.scripts;
+
+ var body = msg.getBody();
+ if (this.pauseOnExceptions_) {
+ var body = msg.getBody();
+ var line = devtools.DebuggerAgent.v8ToWwebkitLineNumber_(body.sourceLine);
+ this.createExceptionMessage_(body.script.name, line, body.exception.text);
+ this.requestBacktrace_();
+ } else {
+ this.resumeExecution();
+ }
+};
+
+
+/**
+ * @param {devtools.DebuggerMessage} msg
+ */
+devtools.DebuggerAgent.prototype.handleScriptsResponse_ = function(msg) {
+ var scripts = msg.getBody();
+ for (var i = 0; i < scripts.length; i++) {
+ var script = scripts[i];
+
+ // Skip scripts from other tabs.
+ if (!this.isScriptFromInspectedContext_(script, msg)) {
+ continue;
+ }
+
+ // We may already have received the info in an afterCompile event.
+ if (script.id in this.parsedScripts_) {
+ continue;
+ }
+ this.addScriptInfo_(script, msg);
+ }
+};
+
+
+/**
+ * @param {Object} script Json object representing script.
+ * @param {devtools.DebuggerMessage} msg Debugger response.
+ */
+devtools.DebuggerAgent.prototype.isScriptFromInspectedContext_ = function(
+ script, msg) {
+ if (!script.context) {
+ // Always ignore scripts from the utility context.
+ return false;
+ }
+ var context = msg.lookup(script.context.ref);
+ var scriptContextId = context.data;
+ if (!goog.isDef(scriptContextId)) {
+ return false; // Always ignore scripts from the utility context.
+ }
+ if (this.contextId_ === null) {
+ return true;
+ }
+ return (scriptContextId.value == this.contextId_);
+};
+
+
+/**
+ * @param {devtools.DebuggerMessage} msg
+ */
+devtools.DebuggerAgent.prototype.handleSetBreakpointResponse_ = function(msg) {
+ var requestSeq = msg.getRequestSeq();
+ var breakpointInfo = this.requestNumberToBreakpointInfo_[requestSeq];
+ if (!breakpointInfo) {
+ // TODO(yurys): handle this case
+ return;
+ }
+ delete this.requestNumberToBreakpointInfo_[requestSeq];
+ if (!msg.isSuccess()) {
+ // TODO(yurys): handle this case
+ return;
+ }
+ var idInV8 = msg.getBody().breakpoint;
+ breakpointInfo.setV8Id(idInV8);
+
+ if (breakpointInfo.isRemoved()) {
+ this.requestClearBreakpoint_(idInV8);
+ }
+};
+
+
+/**
+ * @param {devtools.DebuggerMessage} msg
+ */
+devtools.DebuggerAgent.prototype.handleAfterCompileEvent_ = function(msg) {
+ if (!this.contextId_) {
+ // Ignore scripts delta if main request has not been issued yet.
+ return;
+ }
+ var script = msg.getBody().script;
+
+ // Ignore scripts from other tabs.
+ if (!this.isScriptFromInspectedContext_(script, msg)) {
+ return;
+ }
+ this.addScriptInfo_(script, msg);
+};
+
+
+/**
+ * Handles current profiler status.
+ * @param {number} modules List of active (started) modules.
+ */
+devtools.DebuggerAgent.prototype.didGetActiveProfilerModules_ = function(
+ modules) {
+ var profModules = devtools.DebuggerAgent.ProfilerModules;
+ var profModuleNone = profModules.PROFILER_MODULE_NONE;
+ if (modules != profModuleNone &&
+ this.activeProfilerModules_ == profModuleNone) {
+ // Start to query log data.
+ RemoteDebuggerAgent.GetNextLogLines();
+ }
+ this.activeProfilerModules_ = modules;
+ // Update buttons.
+ WebInspector.setRecordingProfile(modules & profModules.PROFILER_MODULE_CPU);
+ if (modules != profModuleNone) {
+ // Monitor profiler state. It can stop itself on buffer fill-up.
+ setTimeout(
+ function() { RemoteDebuggerAgent.GetActiveProfilerModules(); }, 1000);
+ }
+};
+
+
+/**
+ * Handles a portion of a profiler log retrieved by GetNextLogLines call.
+ * @param {string} log A portion of profiler log.
+ */
+devtools.DebuggerAgent.prototype.didGetNextLogLines_ = function(log) {
+ if (log.length > 0) {
+ this.profilerProcessor_.processLogChunk(log);
+ } else if (this.activeProfilerModules_ ==
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_NONE) {
+ // No new data and profiling is stopped---suspend log reading.
+ return;
+ }
+ setTimeout(function() { RemoteDebuggerAgent.GetNextLogLines(); }, 500);
+};
+
+
+/**
+ * Adds the script info to the local cache. This method assumes that the script
+ * is not in the cache yet.
+ * @param {Object} script Script json object from the debugger message.
+ * @param {devtools.DebuggerMessage} msg Debugger message containing the script
+ * data.
+ */
+devtools.DebuggerAgent.prototype.addScriptInfo_ = function(script, msg) {
+ var context = msg.lookup(script.context.ref);
+ var contextType = context.data.type;
+ this.parsedScripts_[script.id] = new devtools.ScriptInfo(
+ script.id, script.name, script.lineOffset, contextType);
+ if (WebInspector.panels.scripts.element.parentElement) {
+ // Only report script as parsed after scripts panel has been shown.
+ WebInspector.parsedScriptSource(
+ script.id, script.name, script.source, script.lineOffset);
+ }
+};
+
+
+/**
+ * @param {devtools.DebuggerMessage} msg
+ */
+devtools.DebuggerAgent.prototype.handleClearBreakpointResponse_ = function(
+ msg) {
+ // Do nothing.
+};
+
+
+/**
+ * Handles response to 'backtrace' command.
+ * @param {devtools.DebuggerMessage} msg
+ */
+devtools.DebuggerAgent.prototype.handleBacktraceResponse_ = function(msg) {
+ var frames = msg.getBody().frames;
+ this.callFrames_ = [];
+ for (var i = 0; i < frames.length; ++i) {
+ this.callFrames_.push(this.formatCallFrame_(frames[i]));
+ }
+ WebInspector.pausedScript(this.callFrames_);
+ this.showPendingExceptionMessage_();
+ DevToolsHost.activateWindow();
+};
+
+
+/**
+ * Evaluates code on given callframe.
+ */
+devtools.DebuggerAgent.prototype.evaluateInCallFrame = function(
+ callFrameId, code, callback) {
+ var callFrame = this.callFrames_[callFrameId];
+ callFrame.evaluate_(code, callback);
+};
+
+
+/**
+ * Handles response to a command by invoking its callback (if any).
+ * @param {devtools.DebuggerMessage} msg
+ * @return {boolean} Whether a callback for the given message was found and
+ * excuted.
+ */
+devtools.DebuggerAgent.prototype.invokeCallbackForResponse_ = function(msg) {
+ var callback = this.requestSeqToCallback_[msg.getRequestSeq()];
+ if (!callback) {
+ // It may happend if reset was called.
+ return false;
+ }
+ delete this.requestSeqToCallback_[msg.getRequestSeq()];
+ callback(msg);
+ return true;
+};
+
+
+/**
+ * @param {Object} stackFrame Frame json object from 'backtrace' response.
+ * @return {!devtools.CallFrame} Object containing information related to the
+ * call frame in the format expected by ScriptsPanel and its panes.
+ */
+devtools.DebuggerAgent.prototype.formatCallFrame_ = function(stackFrame) {
+ var func = stackFrame.func;
+ var sourceId = func.scriptId;
+
+ // Add service script if it does not exist.
+ var existingScript = this.parsedScripts_[sourceId];
+ if (!existingScript) {
+ this.parsedScripts_[sourceId] = new devtools.ScriptInfo(
+ sourceId, null /* name */, 0 /* line */, 'unknown' /* type */,
+ true /* unresolved */);
+ WebInspector.parsedScriptSource(sourceId, null, null, 0);
+ }
+
+ var funcName = func.name || func.inferredName || '(anonymous function)';
+ var line = devtools.DebuggerAgent.v8ToWwebkitLineNumber_(stackFrame.line);
+
+ // Add basic scope chain info with scope variables.
+ var scopeChain = [];
+ var ScopeType = devtools.DebuggerAgent.ScopeType;
+ for (var i = 0; i < stackFrame.scopes.length; i++) {
+ var scope = stackFrame.scopes[i];
+ scope.frameNumber = stackFrame.index;
+ var scopeObjectProxy = new WebInspector.ObjectProxy(scope, [], 0, '', true);
+ scopeObjectProxy.isScope = true;
+ scopeObjectProxy.properties = {}; // TODO(pfeldman): Fix autocomplete.
+ switch(scope.type) {
+ case ScopeType.Global:
+ scopeObjectProxy.isDocument = true;
+ break;
+ case ScopeType.Local:
+ scopeObjectProxy.isLocal = true;
+ scopeObjectProxy.thisObject =
+ devtools.DebuggerAgent.formatObjectProxy_(stackFrame.receiver);
+ break;
+ case ScopeType.With:
+ // Catch scope is treated as a regular with scope by WebKit so we
+ // also treat it this way.
+ case ScopeType.Catch:
+ scopeObjectProxy.isWithBlock = true;
+ break;
+ case ScopeType.Closure:
+ scopeObjectProxy.isClosure = true;
+ break;
+ }
+ scopeChain.push(scopeObjectProxy);
+ }
+ return new devtools.CallFrame(stackFrame.index, 'function', funcName,
+ sourceId, line, scopeChain);
+};
+
+
+/**
+ * Collects properties for an object from the debugger response.
+ * @param {Object} object An object from the debugger protocol response.
+ * @param {Array.<WebInspector.ObjectPropertyProxy>} result An array to put the
+ * properties into.
+ * @param {boolean} noIntrinsic Whether intrinsic properties should be
+ * included.
+ */
+devtools.DebuggerAgent.formatObjectProperties_ = function(object, result,
+ noIntrinsic) {
+ devtools.DebuggerAgent.propertiesToProxies_(object.properties, result);
+ if (noIntrinsic) {
+ return;
+ }
+
+ result.push(new WebInspector.ObjectPropertyProxy('__proto__',
+ devtools.DebuggerAgent.formatObjectProxy_(object.protoObject)));
+ result.push(new WebInspector.ObjectPropertyProxy('prototype',
+ devtools.DebuggerAgent.formatObjectProxy_(object.prototypeObject)));
+ result.push(new WebInspector.ObjectPropertyProxy('constructor',
+ devtools.DebuggerAgent.formatObjectProxy_(object.constructorFunction)));
+};
+
+
+/**
+ * For each property in 'properties' creates its proxy representative.
+ * @param {Array.<Object>} properties Receiver properties or locals array from
+ * 'backtrace' response.
+ * @param {Array.<WebInspector.ObjectPropertyProxy>} Results holder.
+ */
+devtools.DebuggerAgent.propertiesToProxies_ = function(properties, result) {
+ var map = {};
+ for (var i = 0; i < properties.length; ++i) {
+ var property = properties[i];
+ var name = String(property.name);
+ if (name in map) {
+ continue;
+ }
+ map[name] = true;
+ var value = devtools.DebuggerAgent.formatObjectProxy_(property.value);
+ var propertyProxy = new WebInspector.ObjectPropertyProxy(name, value);
+ result.push(propertyProxy);
+ }
+};
+
+
+/**
+ * @param {Object} v An object reference from the debugger response.
+ * @return {*} The value representation expected by ScriptsPanel.
+ */
+devtools.DebuggerAgent.formatObjectProxy_ = function(v) {
+ var description;
+ var hasChildren = false;
+ if (v.type == 'object') {
+ description = v.className;
+ hasChildren = true;
+ } else if (v.type == 'function') {
+ if (v.source) {
+ description = v.source;
+ } else {
+ description = 'function ' + v.name + '()';
+ }
+ hasChildren = true;
+ } else if (goog.isDef(v.value)) {
+ description = v.value;
+ } else if (v.type == 'undefined') {
+ description = 'undefined';
+ } else if (v.type == 'null') {
+ description = 'null';
+ } else {
+ description = '<unresolved ref: ' + v.ref + ', type: ' + v.type + '>';
+ }
+ var proxy = new WebInspector.ObjectProxy(v, [], 0, description, hasChildren);
+ proxy.type = v.type;
+ proxy.isV8Ref = true;
+ return proxy;
+};
+
+
+/**
+ * Converts line number from Web Inspector UI(1-based) to v8(0-based).
+ * @param {number} line Resource line number in Web Inspector UI.
+ * @return {number} The line number in v8.
+ */
+devtools.DebuggerAgent.webkitToV8LineNumber_ = function(line) {
+ return line - 1;
+};
+
+
+/**
+ * Converts line number from v8(0-based) to Web Inspector UI(1-based).
+ * @param {number} line Resource line number in v8.
+ * @return {number} The line number in Web Inspector.
+ */
+devtools.DebuggerAgent.v8ToWwebkitLineNumber_ = function(line) {
+ return line + 1;
+};
+
+
+/**
+ * @param {number} scriptId Id of the script.
+ * @param {?string} url Script resource URL if any.
+ * @param {number} lineOffset First line 0-based offset in the containing
+ * document.
+ * @param {string} contextType Type of the script's context:
+ * "page" - regular script from html page
+ * "injected" - extension content script
+ * @param {bool} opt_isUnresolved If true, script will not be resolved.
+ * @constructor
+ */
+devtools.ScriptInfo = function(
+ scriptId, url, lineOffset, contextType, opt_isUnresolved) {
+ this.scriptId_ = scriptId;
+ this.lineOffset_ = lineOffset;
+ this.contextType_ = contextType;
+ this.url_ = url;
+ this.isUnresolved_ = opt_isUnresolved;
+
+ this.lineToBreakpointInfo_ = {};
+};
+
+
+/**
+ * @return {number}
+ */
+devtools.ScriptInfo.prototype.getLineOffset = function() {
+ return this.lineOffset_;
+};
+
+
+/**
+ * @return {string}
+ */
+devtools.ScriptInfo.prototype.getContextType = function() {
+ return this.contextType_;
+};
+
+
+/**
+ * @return {?string}
+ */
+devtools.ScriptInfo.prototype.getUrl = function() {
+ return this.url_;
+};
+
+
+/**
+ * @return {?bool}
+ */
+devtools.ScriptInfo.prototype.isUnresolved = function() {
+ return this.isUnresolved_;
+};
+
+
+/**
+ * @param {number} line 0-based line number in the script.
+ * @return {?devtools.BreakpointInfo} Information on a breakpoint at the
+ * specified line in the script or undefined if there is no breakpoint at
+ * that line.
+ */
+devtools.ScriptInfo.prototype.getBreakpointInfo = function(line) {
+ return this.lineToBreakpointInfo_[line];
+};
+
+
+/**
+ * Adds breakpoint info to the script.
+ * @param {devtools.BreakpointInfo} breakpoint
+ */
+devtools.ScriptInfo.prototype.addBreakpointInfo = function(breakpoint) {
+ this.lineToBreakpointInfo_[breakpoint.getLine()] = breakpoint;
+};
+
+
+/**
+ * @param {devtools.BreakpointInfo} breakpoint Breakpoint info to be removed.
+ */
+devtools.ScriptInfo.prototype.removeBreakpointInfo = function(breakpoint) {
+ var line = breakpoint.getLine();
+ delete this.lineToBreakpointInfo_[line];
+};
+
+
+
+/**
+ * @param {number} line Breakpoint 0-based line number in the containing script.
+ * @constructor
+ */
+devtools.BreakpointInfo = function(line) {
+ this.line_ = line;
+ this.v8id_ = -1;
+ this.removed_ = false;
+};
+
+
+/**
+ * @return {number}
+ */
+devtools.BreakpointInfo.prototype.getLine = function(n) {
+ return this.line_;
+};
+
+
+/**
+ * @return {number} Unique identifier of this breakpoint in the v8 debugger.
+ */
+devtools.BreakpointInfo.prototype.getV8Id = function(n) {
+ return this.v8id_;
+};
+
+
+/**
+ * Sets id of this breakpoint in the v8 debugger.
+ * @param {number} id
+ */
+devtools.BreakpointInfo.prototype.setV8Id = function(id) {
+ this.v8id_ = id;
+};
+
+
+/**
+ * Marks this breakpoint as removed from the front-end.
+ */
+devtools.BreakpointInfo.prototype.markAsRemoved = function() {
+ this.removed_ = true;
+};
+
+
+/**
+ * @return {boolean} Whether this breakpoint has been removed from the
+ * front-end.
+ */
+devtools.BreakpointInfo.prototype.isRemoved = function() {
+ return this.removed_;
+};
+
+
+/**
+ * Call stack frame data.
+ * @param {string} id CallFrame id.
+ * @param {string} type CallFrame type.
+ * @param {string} functionName CallFrame type.
+ * @param {string} sourceID Source id.
+ * @param {number} line Source line.
+ * @param {Array.<Object>} scopeChain Array of scoped objects.
+ * @construnctor
+ */
+devtools.CallFrame = function(id, type, functionName, sourceID, line,
+ scopeChain) {
+ this.id = id;
+ this.type = type;
+ this.functionName = functionName;
+ this.sourceID = sourceID;
+ this.line = line;
+ this.scopeChain = scopeChain;
+};
+
+
+/**
+ * This method issues asynchronous evaluate request, reports result to the
+ * callback.
+ * @param {string} expression An expression to be evaluated in the context of
+ * this call frame.
+ * @param {function(Object):undefined} callback Callback to report result to.
+ */
+devtools.CallFrame.prototype.evaluate_ = function(expression, callback) {
+ devtools.tools.getDebuggerAgent().requestEvaluate({
+ 'expression': expression,
+ 'frame': this.id,
+ 'global': false,
+ 'disable_break': false,
+ 'compactFormat': true
+ },
+ function(response) {
+ var result = {};
+ if (response.isSuccess()) {
+ result.value = devtools.DebuggerAgent.formatObjectProxy_(
+ response.getBody());
+ } else {
+ result.value = response.getMessage();
+ result.isException = true;
+ }
+ callback(result);
+ });
+};
+
+
+/**
+ * JSON based commands sent to v8 debugger.
+ * @param {string} command Name of the command to execute.
+ * @param {Object} opt_arguments Command-specific arguments map.
+ * @constructor
+ */
+devtools.DebugCommand = function(command, opt_arguments) {
+ this.command_ = command;
+ this.type_ = 'request';
+ this.seq_ = ++devtools.DebugCommand.nextSeq_;
+ if (opt_arguments) {
+ this.arguments_ = opt_arguments;
+ }
+};
+
+
+/**
+ * Next unique number to be used as debugger request sequence number.
+ * @type {number}
+ */
+devtools.DebugCommand.nextSeq_ = 1;
+
+
+/**
+ * @return {number}
+ */
+devtools.DebugCommand.prototype.getSequenceNumber = function() {
+ return this.seq_;
+};
+
+
+/**
+ * @return {string}
+ */
+devtools.DebugCommand.prototype.toJSONProtocol = function() {
+ var json = {
+ 'seq': this.seq_,
+ 'type': this.type_,
+ 'command': this.command_
+ }
+ if (this.arguments_) {
+ json.arguments = this.arguments_;
+ }
+ return JSON.stringify(json);
+};
+
+
+/**
+ * JSON messages sent from v8 debugger. See protocol definition for more
+ * details: http://code.google.com/p/v8/wiki/DebuggerProtocol
+ * @param {string} msg Raw protocol packet as JSON string.
+ * @constructor
+ */
+devtools.DebuggerMessage = function(msg) {
+ this.packet_ = JSON.parse(msg);
+ this.refs_ = [];
+ if (this.packet_.refs) {
+ for (var i = 0; i < this.packet_.refs.length; i++) {
+ this.refs_[this.packet_.refs[i].handle] = this.packet_.refs[i];
+ }
+ }
+};
+
+
+/**
+ * @return {string} The packet type.
+ */
+devtools.DebuggerMessage.prototype.getType = function() {
+ return this.packet_.type;
+};
+
+
+/**
+ * @return {?string} The packet event if the message is an event.
+ */
+devtools.DebuggerMessage.prototype.getEvent = function() {
+ return this.packet_.event;
+};
+
+
+/**
+ * @return {?string} The packet command if the message is a response to a
+ * command.
+ */
+devtools.DebuggerMessage.prototype.getCommand = function() {
+ return this.packet_.command;
+};
+
+
+/**
+ * @return {number} The packet request sequence.
+ */
+devtools.DebuggerMessage.prototype.getRequestSeq = function() {
+ return this.packet_.request_seq;
+};
+
+
+/**
+ * @return {number} Whether the v8 is running after processing the request.
+ */
+devtools.DebuggerMessage.prototype.isRunning = function() {
+ return this.packet_.running ? true : false;
+};
+
+
+/**
+ * @return {boolean} Whether the request succeeded.
+ */
+devtools.DebuggerMessage.prototype.isSuccess = function() {
+ return this.packet_.success ? true : false;
+};
+
+
+/**
+ * @return {string}
+ */
+devtools.DebuggerMessage.prototype.getMessage = function() {
+ return this.packet_.message;
+};
+
+
+/**
+ * @return {Object} Parsed message body json.
+ */
+devtools.DebuggerMessage.prototype.getBody = function() {
+ return this.packet_.body;
+};
+
+
+/**
+ * @param {number} handle Object handle.
+ * @return {?Object} Returns the object with the handle if it was sent in this
+ * message(some objects referenced by handles may be missing in the message).
+ */
+devtools.DebuggerMessage.prototype.lookup = function(handle) {
+ return this.refs_[handle];
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.css b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.css
new file mode 100644
index 0000000..eb649c5
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.css
@@ -0,0 +1,99 @@
+#scripts-files option.injected {
+ color: rgb(70, 134, 240);
+}
+
+.data-grid table {
+ line-height: 120%;
+}
+
+body.attached #toolbar {
+ height: 34px;
+ border-top: 1px solid rgb(100, 100, 100);
+ cursor: default; /* overriden */
+ padding-left: 0;
+}
+
+
+/* Chrome theme overrides */
+#toolbar {
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(242, 247, 253)), to(rgb(223, 234, 248)));
+}
+
+
+/* Heap Profiler Styles */
+
+#heap-snapshot-views {
+ position: absolute;
+ top: 0;
+ right: 0;
+ left: 200px;
+ bottom: 0;
+}
+
+#heap-snapshot-status-bar-items {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ left: 200px;
+ overflow: hidden;
+ border-left: 1px solid rgb(184, 184, 184);
+ margin-left: -1px;
+}
+
+.heap-snapshot-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/focusButtonGlyph.png);
+}
+
+.heap-snapshot-sidebar-tree-item .icon {
+ content: url(Images/profileIcon.png);
+}
+
+.heap-snapshot-sidebar-tree-item.small .icon {
+ content: url(Images/profileSmallIcon.png);
+}
+
+.heap-snapshot-view {
+ display: none;
+ overflow: hidden;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+}
+
+.heap-snapshot-view.visible {
+ display: block;
+}
+
+.heap-snapshot-view .data-grid {
+ border: none;
+ max-height: 100%;
+ position: absolute;
+ left: 0;
+ right: 0;
+ top: 0;
+ bottom: 93px;
+}
+
+#heap-snapshot-summary {
+ position: absolute;
+ padding-top: 20px;
+ bottom: 0;
+ left: 0;
+ right: 0;
+ height: 93px;
+ margin-left: -1px;
+ border-left: 1px solid rgb(102, 102, 102);
+ background-color: rgb(101, 111, 130);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgba(0, 0, 0, 0.5)), to(rgba(0, 0, 0, 0)));
+ background-repeat: repeat-x;
+ background-position: top;
+ text-align: center;
+ text-shadow: black 0 1px 1px;
+ white-space: nowrap;
+ color: white;
+ -webkit-background-size: 1px 6px;
+ -webkit-background-origin: padding;
+ -webkit-background-clip: padding;
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.html b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.html
new file mode 100644
index 0000000..398bbed
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.html
@@ -0,0 +1,142 @@
+<!--
+Copyright (c) 2009 The Chromium Authors. All rights reserved.
+
+This is the Chromium version of the WebInspector. This html file serves
+as a deployment descriptor and specifies which js libraries to include into the
+app. Once the "main" frontend method that is building WebInspector
+from the js building blocks is extracted, we will be able have different
+implementations of it for Chromium and WebKit. That would allow us for
+example not to create WebKit Database tab and remove corresponding js files
+from here. Longer term we would like to employ closure + basic js compilation.
+That way js libraries would know their dependencies and js compiler would be
+able to compile them into a single file. After that this HTML file will
+include single <script src='fe-compiled.js'> and will become upstreamable.
+
+Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions
+are met:
+
+1. Redistributions of source code must retain the above copyright
+ notice, this list of conditions and the following disclaimer.
+2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in the
+ documentation and/or other materials provided with the distribution.
+3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ its contributors may be used to endorse or promote products derived
+ from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+-->
+<!DOCTYPE html>
+<html>
+<head>
+ <meta http-equiv="content-type" content="text/html; charset=utf-8">
+ <link rel="stylesheet" type="text/css" href="inspector.css">
+ <link rel="stylesheet" type="text/css" href="devtools.css">
+ <script type="text/javascript" src="base.js"></script>
+ <script type="text/javascript" src="utilities.js"></script>
+ <script type="text/javascript" src="treeoutline.js"></script>
+ <script type="text/javascript" src="devtools_callback.js"></script>
+ <script type="text/javascript" src="debugger_agent.js"></script>
+ <script type="text/javascript" src="inspector_controller.js"></script>
+ <script type="text/javascript" src="inspector.js"></script>
+ <script type="text/javascript" src="codemap.js"></script>
+ <script type="text/javascript" src="consarray.js"></script>
+ <script type="text/javascript" src="csvparser.js"></script>
+ <script type="text/javascript" src="logreader.js"></script>
+ <script type="text/javascript" src="profile.js"></script>
+ <script type="text/javascript" src="profile_view.js"></script>
+ <script type="text/javascript" src="profiler_processor.js"></script>
+ <script type="text/javascript" src="splaytree.js"></script>
+ <script type="text/javascript" src="Object.js"></script>
+ <script type="text/javascript" src="KeyboardShortcut.js"></script>
+ <script type="text/javascript" src="TextPrompt.js"></script>
+ <script type="text/javascript" src="Placard.js"></script>
+ <script type="text/javascript" src="View.js"></script>
+ <script type="text/javascript" src="ChangesView.js"></script>
+ <script type="text/javascript" src="ConsoleView.js"></script>
+ <script type="text/javascript" src="Drawer.js"></script>
+ <script type="text/javascript" src="Resource.js"></script>
+ <script type="text/javascript" src="ResourceCategory.js"></script>
+ <script type="text/javascript" src="Database.js"></script>
+ <script type="text/javascript" src="Callback.js"></script>
+ <script type="text/javascript" src="DOMAgent.js"></script>
+ <script type="text/javascript" src="TimelineAgent.js"></script>
+ <script type="text/javascript" src="InjectedScriptAccess.js"></script>
+ <script type="text/javascript" src="inspector_controller_impl.js"></script>
+ <script type="text/javascript" src="DOMStorage.js"></script>
+ <script type="text/javascript" src="DOMStorageItemsView.js"></script>
+ <script type="text/javascript" src="DataGrid.js"></script>
+ <script type="text/javascript" src="DOMStorageDataGrid.js"></script>
+ <script type="text/javascript" src="Script.js"></script>
+ <script type="text/javascript" src="Breakpoint.js"></script>
+ <script type="text/javascript" src="SidebarPane.js"></script>
+ <script type="text/javascript" src="ElementsTreeOutline.js"></script>
+ <script type="text/javascript" src="SidebarTreeElement.js"></script>
+ <script type="text/javascript" src="PropertiesSection.js"></script>
+ <script type="text/javascript" src="ObjectPropertiesSection.js"></script>
+ <script type="text/javascript" src="ObjectProxy.js"></script>
+ <script type="text/javascript" src="BreakpointsSidebarPane.js"></script>
+ <script type="text/javascript" src="CallStackSidebarPane.js"></script>
+ <script type="text/javascript" src="ScopeChainSidebarPane.js"></script>
+ <script type="text/javascript" src="WatchExpressionsSidebarPane.js"></script>
+ <script type="text/javascript" src="MetricsSidebarPane.js"></script>
+ <script type="text/javascript" src="PropertiesSidebarPane.js"></script>
+ <script type="text/javascript" src="Color.js"></script>
+ <script type="text/javascript" src="StylesSidebarPane.js"></script>
+ <script type="text/javascript" src="Panel.js"></script>
+ <script type="text/javascript" src="PanelEnablerView.js"></script>
+ <script type="text/javascript" src="StatusBarButton.js"></script>
+ <script type="text/javascript" src="SummaryBar.js"></script>
+ <script type="text/javascript" src="ElementsPanel.js"></script>
+ <script type="text/javascript" src="ResourcesPanel.js"></script>
+ <script type="text/javascript" src="ScriptsPanel.js"></script>
+ <script type="text/javascript" src="DatabasesPanel.js"></script>
+ <script type="text/javascript" src="ProfilesPanel.js"></script>
+ <script type="text/javascript" src="ResourceView.js"></script>
+ <script type="text/javascript" src="Popup.js"></script>
+ <script type="text/javascript" src="SourceFrame.js"></script>
+ <script type="text/javascript" src="SourceView.js"></script>
+ <script type="text/javascript" src="FontView.js"></script>
+ <script type="text/javascript" src="ImageView.js"></script>
+ <script type="text/javascript" src="DatabaseTableView.js"></script>
+ <script type="text/javascript" src="DatabaseQueryView.js"></script>
+ <script type="text/javascript" src="ScriptView.js"></script>
+ <script type="text/javascript" src="ProfileView.js"></script>
+ <script type="text/javascript" src="ProfileDataGridTree.js"></script>
+ <script type="text/javascript" src="BottomUpProfileDataGridTree.js"></script>
+ <script type="text/javascript" src="TopDownProfileDataGridTree.js"></script>
+ <script type="text/javascript" src="heap_profiler_panel.js"></script>
+ <script type="text/javascript" src="devtools.js"></script>
+ <script type="text/javascript" src="devtools_host_stub.js"></script>
+ <script type="text/javascript" src="tests.js"></script>
+</head>
+<body class="detached">
+ <div id="toolbar">
+ <div class="toolbar-item hidden"></div>
+ <div class="toolbar-item flexable-space"></div>
+ <div class="toolbar-item hidden" id="search-results-matches"></div>
+ <div class="toolbar-item"><input id="search" type="search" incremental results="0"><div id="search-toolbar-label" class="toolbar-label"></div></div>
+ <div class="toolbar-item close"><button id="close-button"></button></div>
+ </div>
+ <div id="main">
+ <div id="main-panels" tabindex="0" spellcheck="false"></div>
+ <div id="main-status-bar" class="status-bar"><div id="anchored-status-bar-items"><button id="dock-status-bar-item" class="status-bar-item toggled"><div class="glyph"></div><div class="glyph shadow"></div></button><button id="console-status-bar-item" class="status-bar-item"><div class="glyph"></div><div class="glyph shadow"></div></button><button id="changes-status-bar-item" class="status-bar-item hidden"></button><div id="count-items"><div id="changes-count" class="hidden"></div><div id="error-warning-count" class="hidden"></div></div></div></div>
+ </div>
+ <div id="drawer">
+ <div id="console-view"><div id="console-messages"><div id="console-prompt" spellcheck="false"><br></div></div></div>
+ <div id="drawer-status-bar" class="status-bar"><div id="other-drawer-status-bar-items"><button id="clear-console-status-bar-item" class="status-bar-item"><div class="glyph"></div><div class="glyph shadow"></div></button><div id="console-filter" class="status-bar-item"></div></div></div>
+ </div>
+</body>
+</html>
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.js
new file mode 100644
index 0000000..5086f80
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools.js
@@ -0,0 +1,392 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview Tools is a main class that wires all components of the
+ * DevTools frontend together. It is also responsible for overriding existing
+ * WebInspector functionality while it is getting upstreamed into WebCore.
+ */
+goog.provide('devtools.Tools');
+
+goog.require('devtools.DebuggerAgent');
+
+
+/**
+ * Dispatches raw message from the host.
+ * @param {string} remoteName
+ * @prama {string} methodName
+ * @param {string} param1, param2, param3 Arguments to dispatch.
+ */
+devtools$$dispatch = function(remoteName, methodName, param1, param2, param3) {
+ remoteName = 'Remote' + remoteName.substring(0, remoteName.length - 8);
+ var agent = window[remoteName];
+ if (!agent) {
+ debugPrint('No remote agent "' + remoteName + '" found.');
+ return;
+ }
+ var method = agent[methodName];
+ if (!method) {
+ debugPrint('No method "' + remoteName + '.' + methodName + '" found.');
+ return;
+ }
+ method.call(this, param1, param2, param3);
+};
+
+
+devtools.ToolsAgent = function() {
+ RemoteToolsAgent.DidExecuteUtilityFunction =
+ devtools.Callback.processCallback;
+ RemoteToolsAgent.FrameNavigate =
+ goog.bind(this.frameNavigate_, this);
+ RemoteToolsAgent.DispatchOnClient =
+ goog.bind(this.dispatchOnClient_, this);
+ this.debuggerAgent_ = new devtools.DebuggerAgent();
+};
+
+
+/**
+ * Resets tools agent to its initial state.
+ */
+devtools.ToolsAgent.prototype.reset = function() {
+ DevToolsHost.reset();
+ this.debuggerAgent_.reset();
+};
+
+
+/**
+ * @param {string} script Script exression to be evaluated in the context of the
+ * inspected page.
+ * @param {function(Object|string, boolean):undefined} opt_callback Function to
+ * call with the result.
+ */
+devtools.ToolsAgent.prototype.evaluateJavaScript = function(script,
+ opt_callback) {
+ InspectorController.evaluate(script, opt_callback || function() {});
+};
+
+
+/**
+ * @return {devtools.DebuggerAgent} Debugger agent instance.
+ */
+devtools.ToolsAgent.prototype.getDebuggerAgent = function() {
+ return this.debuggerAgent_;
+};
+
+
+/**
+ * @param {string} url Url frame navigated to.
+ * @see tools_agent.h
+ * @private
+ */
+devtools.ToolsAgent.prototype.frameNavigate_ = function(url) {
+ this.reset();
+ // Do not reset Profiles panel.
+ var profiles = null;
+ if ('profiles' in WebInspector.panels) {
+ profiles = WebInspector.panels['profiles'];
+ delete WebInspector.panels['profiles'];
+ }
+ WebInspector.reset();
+ if (profiles != null) {
+ WebInspector.panels['profiles'] = profiles;
+ }
+};
+
+
+/**
+ * @param {string} message Serialized call to be dispatched on WebInspector.
+ * @private
+ */
+devtools.ToolsAgent.prototype.dispatchOnClient_ = function(message) {
+ WebInspector.dispatch.apply(WebInspector, JSON.parse(message));
+};
+
+
+/**
+ * Evaluates js expression.
+ * @param {string} expr
+ */
+devtools.ToolsAgent.prototype.evaluate = function(expr) {
+ RemoteToolsAgent.evaluate(expr);
+};
+
+
+/**
+ * Enables / disables resources panel in the ui.
+ * @param {boolean} enabled New panel status.
+ */
+WebInspector.setResourcesPanelEnabled = function(enabled) {
+ InspectorController.resourceTrackingEnabled_ = enabled;
+ WebInspector.panels.resources.reset();
+};
+
+
+/**
+ * Prints string to the inspector console or shows alert if the console doesn't
+ * exist.
+ * @param {string} text
+ */
+function debugPrint(text) {
+ var console = WebInspector.console;
+ if (console) {
+ console.addMessage(new WebInspector.ConsoleMessage(
+ WebInspector.ConsoleMessage.MessageSource.JS,
+ WebInspector.ConsoleMessage.MessageType.Log,
+ WebInspector.ConsoleMessage.MessageLevel.Log,
+ 1, 'chrome://devtools/<internal>', undefined, -1, text));
+ } else {
+ alert(text);
+ }
+}
+
+
+/**
+ * Global instance of the tools agent.
+ * @type {devtools.ToolsAgent}
+ */
+devtools.tools = null;
+
+
+var context = {}; // Used by WebCore's inspector routines.
+
+///////////////////////////////////////////////////////////////////////////////
+// Here and below are overrides to existing WebInspector methods only.
+// TODO(pfeldman): Patch WebCore and upstream changes.
+var oldLoaded = WebInspector.loaded;
+WebInspector.loaded = function() {
+ devtools.tools = new devtools.ToolsAgent();
+ devtools.tools.reset();
+
+ Preferences.ignoreWhitespace = false;
+ Preferences.samplingCPUProfiler = true;
+ oldLoaded.call(this);
+
+ // Hide dock button on Mac OS.
+ // TODO(pfeldman): remove once Mac OS docking is implemented.
+ if (InspectorController.platform().indexOf('mac') == 0) {
+ document.getElementById('dock-status-bar-item').addStyleClass('hidden');
+ }
+
+ // Mute refresh action.
+ document.addEventListener("keydown", function(event) {
+ if (event.keyIdentifier == 'F5') {
+ event.preventDefault();
+ } else if (event.keyIdentifier == 'U+0052' /* 'R' */ &&
+ (event.ctrlKey || event.metaKey)) {
+ event.preventDefault();
+ }
+ }, true);
+
+ DevToolsHost.loaded();
+};
+
+
+/**
+ * This override is necessary for adding script source asynchronously.
+ * @override
+ */
+WebInspector.ScriptView.prototype.setupSourceFrameIfNeeded = function() {
+ if (!this._frameNeedsSetup) {
+ return;
+ }
+
+ this.attach();
+
+ if (this.script.source) {
+ this.didResolveScriptSource_();
+ } else {
+ var self = this;
+ devtools.tools.getDebuggerAgent().resolveScriptSource(
+ this.script.sourceID,
+ function(source) {
+ self.script.source = source ||
+ WebInspector.UIString('<source is not available>');
+ self.didResolveScriptSource_();
+ });
+ }
+};
+
+
+/**
+ * Performs source frame setup when script source is aready resolved.
+ */
+WebInspector.ScriptView.prototype.didResolveScriptSource_ = function() {
+ if (!InspectorController.addSourceToFrame(
+ "text/javascript", this.script.source, this.sourceFrame.element)) {
+ return;
+ }
+
+ delete this._frameNeedsSetup;
+
+ this.sourceFrame.addEventListener(
+ "syntax highlighting complete", this._syntaxHighlightingComplete, this);
+ this.sourceFrame.syntaxHighlightJavascript();
+};
+
+
+/**
+ * @param {string} type Type of the the property value('object' or 'function').
+ * @param {string} className Class name of the property value.
+ * @constructor
+ */
+WebInspector.UnresolvedPropertyValue = function(type, className) {
+ this.type = type;
+ this.className = className;
+};
+
+
+/**
+ * This function overrides standard searchableViews getters to perform search
+ * only in the current view (other views are loaded asynchronously, no way to
+ * search them yet).
+ */
+WebInspector.searchableViews_ = function() {
+ var views = [];
+ const visibleView = this.visibleView;
+ if (visibleView && visibleView.performSearch) {
+ views.push(visibleView);
+ }
+ return views;
+};
+
+
+/**
+ * @override
+ */
+WebInspector.ResourcesPanel.prototype.__defineGetter__(
+ 'searchableViews',
+ WebInspector.searchableViews_);
+
+
+/**
+ * @override
+ */
+WebInspector.ScriptsPanel.prototype.__defineGetter__(
+ 'searchableViews',
+ WebInspector.searchableViews_);
+
+
+(function() {
+ var oldShow = WebInspector.ScriptsPanel.prototype.show;
+ WebInspector.ScriptsPanel.prototype.show = function() {
+ devtools.tools.getDebuggerAgent().initUI();
+ this.enableToggleButton.visible = false;
+ oldShow.call(this);
+ };
+})();
+
+
+(function InterceptProfilesPanelEvents() {
+ var oldShow = WebInspector.ProfilesPanel.prototype.show;
+ WebInspector.ProfilesPanel.prototype.show = function() {
+ devtools.tools.getDebuggerAgent().initializeProfiling();
+ this.enableToggleButton.visible = false;
+ oldShow.call(this);
+ // Show is called on every show event of a panel, so
+ // we only need to intercept it once.
+ WebInspector.ProfilesPanel.prototype.show = oldShow;
+ };
+})();
+
+
+/*
+ * @override
+ * TODO(mnaganov): Restore l10n when it will be agreed that it is needed.
+ */
+WebInspector.UIString = function(string) {
+ return String.vsprintf(string, Array.prototype.slice.call(arguments, 1));
+};
+
+
+// There is no clear way of setting frame title yet. So sniffing main resource
+// load.
+(function OverrideUpdateResource() {
+ var originalUpdateResource = WebInspector.updateResource;
+ WebInspector.updateResource = function(identifier, payload) {
+ originalUpdateResource.call(this, identifier, payload);
+ var resource = this.resources[identifier];
+ if (resource && resource.mainResource && resource.finished) {
+ document.title =
+ WebInspector.UIString('Developer Tools - %s', resource.url);
+ }
+ };
+})();
+
+
+// Highlight extension content scripts in the scripts list.
+(function () {
+ var original = WebInspector.ScriptsPanel.prototype._addScriptToFilesMenu;
+ WebInspector.ScriptsPanel.prototype._addScriptToFilesMenu = function(script) {
+ var result = original.apply(this, arguments);
+ var debuggerAgent = devtools.tools.getDebuggerAgent();
+ var type = debuggerAgent.getScriptContextType(script.sourceID);
+ var option = script.filesSelectOption;
+ if (type == 'injected' && option) {
+ option.addStyleClass('injected');
+ }
+ return result;
+ };
+})();
+
+
+/** Pending WebKit upstream by apavlov). Fixes iframe vs drag problem. */
+(function() {
+ var originalDragStart = WebInspector.elementDragStart;
+ WebInspector.elementDragStart = function(element) {
+ var glassPane = document.createElement("div");
+ glassPane.style.cssText =
+ 'position:absolute;width:100%;height:100%;opacity:0;z-index:1';
+ glassPane.id = 'glass-pane-for-drag';
+ element.parentElement.appendChild(glassPane);
+
+ originalDragStart.apply(this, arguments);
+ };
+
+ var originalDragEnd = WebInspector.elementDragEnd;
+ WebInspector.elementDragEnd = function() {
+ originalDragEnd.apply(this, arguments);
+
+ var glassPane = document.getElementById('glass-pane-for-drag');
+ glassPane.parentElement.removeChild(glassPane);
+ };
+})();
+
+
+(function() {
+ var originalCreatePanels = WebInspector._createPanels;
+ WebInspector._createPanels = function() {
+ originalCreatePanels.apply(this, arguments);
+ this.panels.heap = new WebInspector.HeapProfilerPanel();
+ };
+})();
+
+
+(function () {
+var orig = InjectedScriptAccess.getProperties;
+InjectedScriptAccess.getProperties = function(
+ objectProxy, ignoreHasOwnProperty, callback) {
+ if (objectProxy.isScope) {
+ devtools.tools.getDebuggerAgent().resolveScope(objectProxy.objectId,
+ callback);
+ } else if (objectProxy.isV8Ref) {
+ devtools.tools.getDebuggerAgent().resolveChildren(objectProxy.objectId,
+ callback, true);
+ } else {
+ orig.apply(this, arguments);
+ }
+};
+})()
+
+
+WebInspector.resourceTrackingWasEnabled = function()
+{
+ InspectorController.resourceTrackingEnabled_ = true;
+ this.panels.resources.resourceTrackingWasEnabled();
+};
+
+WebInspector.resourceTrackingWasDisabled = function()
+{
+ InspectorController.resourceTrackingEnabled_ = false;
+ this.panels.resources.resourceTrackingWasDisabled();
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_callback.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_callback.js
new file mode 100644
index 0000000..f252861
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_callback.js
@@ -0,0 +1,57 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview Generic callback manager.
+ */
+goog.provide('devtools.Callback');
+
+
+/**
+ * Generic callback support as a singleton object.
+ * @constructor
+ */
+devtools.Callback = function() {
+ this.lastCallbackId_ = 1;
+ this.callbacks_ = {};
+};
+
+
+/**
+ * Assigns id to a callback.
+ * @param {Function} callback Callback to assign id to.
+ * @return {number} Callback id.
+ */
+devtools.Callback.prototype.wrap = function(callback) {
+ var callbackId = this.lastCallbackId_++;
+ this.callbacks_[callbackId] = callback || function() {};
+ return callbackId;
+};
+
+
+/**
+ * Executes callback with the given id.
+ * @param {callbackId} callbackId Id of a callback to call.
+ */
+devtools.Callback.prototype.processCallback = function(callbackId,
+ opt_vararg) {
+ var args = Array.prototype.slice.call(arguments, 1);
+ var callback = this.callbacks_[callbackId];
+ callback.apply(null, args);
+ delete this.callbacks_[callbackId];
+};
+
+
+/**
+ * @type {devtools.Callback} Callback support singleton.
+ * @private
+ */
+devtools.Callback.INSTANCE_ = new devtools.Callback();
+
+devtools.Callback.wrap = goog.bind(
+ devtools.Callback.INSTANCE_.wrap,
+ devtools.Callback.INSTANCE_);
+devtools.Callback.processCallback = goog.bind(
+ devtools.Callback.INSTANCE_.processCallback,
+ devtools.Callback.INSTANCE_);
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_host_stub.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_host_stub.js
new file mode 100644
index 0000000..2f3da60
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/devtools_host_stub.js
@@ -0,0 +1,335 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview These stubs emulate backend functionality and allows
+ * DevTools frontend to function as a standalone web app.
+ */
+
+/**
+ * @constructor
+ */
+RemoteDebuggerAgentStub = function() {
+ this.activeProfilerModules_ =
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_NONE;
+ this.profileLogPos_ = 0;
+ this.heapProfSample_ = 0;
+ this.heapProfLog_ = '';
+};
+
+
+RemoteDebuggerAgentStub.prototype.DebugBreak = function() {
+};
+
+
+RemoteDebuggerAgentStub.prototype.GetContextId = function() {
+ RemoteDebuggerAgent.SetContextId(3);
+};
+
+
+RemoteDebuggerAgentStub.prototype.StopProfiling = function(modules) {
+ this.activeProfilerModules_ &= ~modules;
+};
+
+
+RemoteDebuggerAgentStub.prototype.StartProfiling = function(modules) {
+ var profModules = devtools.DebuggerAgent.ProfilerModules;
+ if (modules & profModules.PROFILER_MODULE_HEAP_SNAPSHOT) {
+ if (modules & profModules.PROFILER_MODULE_HEAP_STATS) {
+ this.heapProfLog_ +=
+ 'heap-sample-begin,"Heap","allocated",' +
+ (new Date()).getTime() + '\n' +
+ 'heap-sample-stats,"Heap","allocated",10000,1000\n';
+ this.heapProfLog_ +=
+ 'heap-sample-item,STRING_TYPE,100,1000\n' +
+ 'heap-sample-item,CODE_TYPE,10,200\n' +
+ 'heap-sample-item,MAP_TYPE,20,350\n';
+ var sample = RemoteDebuggerAgentStub.HeapSamples[this.heapProfSample_];
+ if (++this.heapProfSample_ == RemoteDebuggerAgentStub.HeapSamples.length)
+ this.heapProfSample_ = 0;
+ for (var obj in sample) {
+ this.heapProfLog_ +=
+ 'heap-js-cons-item,"' + obj + '",' + sample[obj][0] +
+ ',' + sample[obj][1] + '\n';
+ }
+ this.heapProfLog_ +=
+ 'heap-sample-end,"Heap","allocated"\n';
+ }
+ } else {
+ this.activeProfilerModules_ |= modules;
+ }
+};
+
+
+RemoteDebuggerAgentStub.prototype.GetActiveProfilerModules = function() {
+ var self = this;
+ setTimeout(function() {
+ RemoteDebuggerAgent.DidGetActiveProfilerModules(
+ self.activeProfilerModules_);
+ }, 100);
+};
+
+
+RemoteDebuggerAgentStub.prototype.GetNextLogLines = function() {
+ var profModules = devtools.DebuggerAgent.ProfilerModules;
+ var logLines = '';
+ if (this.activeProfilerModules_ & profModules.PROFILER_MODULE_CPU) {
+ if (this.profileLogPos_ < RemoteDebuggerAgentStub.ProfilerLogBuffer.length) {
+ this.profileLogPos_ += RemoteDebuggerAgentStub.ProfilerLogBuffer.length;
+ logLines += RemoteDebuggerAgentStub.ProfilerLogBuffer;
+ }
+ }
+ if (this.heapProfLog_) {
+ logLines += this.heapProfLog_;
+ this.heapProfLog_ = '';
+ }
+ setTimeout(function() {
+ RemoteDebuggerAgent.DidGetNextLogLines(logLines);
+ }, 100);
+};
+
+
+/**
+ * @constructor
+ */
+RemoteToolsAgentStub = function() {
+};
+
+
+RemoteToolsAgentStub.prototype.HideDOMNodeHighlight = function() {
+};
+
+
+RemoteToolsAgentStub.prototype.HighlightDOMNode = function() {
+};
+
+
+RemoteToolsAgentStub.prototype.evaluate = function(expr) {
+ window.eval(expr);
+};
+
+RemoteToolsAgentStub.prototype.EvaluateJavaScript = function(callId, script) {
+ setTimeout(function() {
+ var result = eval(script);
+ RemoteToolsAgent.DidEvaluateJavaScript(callId, result);
+ }, 0);
+};
+
+
+RemoteToolsAgentStub.prototype.ExecuteUtilityFunction = function(callId,
+ functionName, args) {
+ setTimeout(function() {
+ var result = [];
+ if (functionName == 'getProperties') {
+ result = [
+ 'undefined', 'undefined_key', undefined,
+ 'string', 'string_key', 'value',
+ 'function', 'func', undefined,
+ 'array', 'array_key', [10],
+ 'object', 'object_key', undefined,
+ 'boolean', 'boolean_key', true,
+ 'number', 'num_key', 911,
+ 'date', 'date_key', new Date() ];
+ } else if (functionName == 'getPrototypes') {
+ result = ['Proto1', 'Proto2', 'Proto3'];
+ } else if (functionName == 'getStyles') {
+ result = {
+ 'computedStyle' : [0, '0px', '0px', null, null, null, ['display', false, false, '', 'none']],
+ 'inlineStyle' : [1, '0px', '0px', null, null, null, ['display', false, false, '', 'none']],
+ 'styleAttributes' : {
+ attr: [2, '0px', '0px', null, null, null, ['display', false, false, '', 'none']]
+ },
+ 'matchedCSSRules' : [
+ { 'selector' : 'S',
+ 'style' : [3, '0px', '0px', null, null, null, ['display', false, false, '', 'none']],
+ 'parentStyleSheet' : { 'href' : 'http://localhost',
+ 'ownerNodeName' : 'DIV' }
+ }
+ ]
+ };
+ } else if (functionName == 'toggleNodeStyle' ||
+ functionName == 'applyStyleText' ||
+ functionName == 'setStyleProperty') {
+ alert(functionName + '(' + args + ')');
+ } else if (functionName == 'evaluate') {
+ try {
+ result = [ window.eval(JSON.parse(args)[0]), false ];
+ } catch (e) {
+ result = [ e.toString(), true ];
+ }
+ } else if (functionName == 'InspectorController' ||
+ functionName == 'InjectedScript') {
+ // do nothing;
+ } else {
+ alert('Unexpected utility function:' + functionName);
+ }
+ RemoteToolsAgent.DidExecuteUtilityFunction(callId,
+ JSON.stringify(result), '');
+ }, 0);
+};
+
+
+RemoteToolsAgentStub.prototype.ExecuteVoidJavaScript = function() {
+};
+
+
+RemoteToolsAgentStub.prototype.SetResourceTrackingEnabled = function(enabled, always) {
+ RemoteToolsAgent.SetResourcesPanelEnabled(enabled);
+ if (enabled) {
+ WebInspector.resourceTrackingWasEnabled();
+ } else {
+ WebInspector.resourceTrackingWasDisabled();
+ }
+ addDummyResource();
+};
+
+
+RemoteDebuggerAgentStub.ProfilerLogBuffer =
+ 'profiler,begin,1\n' +
+ 'profiler,resume\n' +
+ 'code-creation,LazyCompile,0x1000,256,"test1 http://aaa.js:1"\n' +
+ 'code-creation,LazyCompile,0x2000,256,"test2 http://bbb.js:2"\n' +
+ 'code-creation,LazyCompile,0x3000,256,"test3 http://ccc.js:3"\n' +
+ 'tick,0x1010,0x0,3\n' +
+ 'tick,0x2020,0x0,3,0x1010\n' +
+ 'tick,0x2020,0x0,3,0x1010\n' +
+ 'tick,0x3010,0x0,3,0x2020, 0x1010\n' +
+ 'tick,0x2020,0x0,3,0x1010\n' +
+ 'tick,0x2030,0x0,3,0x2020, 0x1010\n' +
+ 'tick,0x2020,0x0,3,0x1010\n' +
+ 'tick,0x1010,0x0,3\n' +
+ 'profiler,pause\n';
+
+
+RemoteDebuggerAgentStub.HeapSamples = [
+ {foo: [1, 100], bar: [20, 2000]},
+ {foo: [2000, 200000], bar: [10, 1000]},
+ {foo: [15, 1500], bar: [15, 1500]},
+ {bar: [20, 2000]},
+ {foo: [15, 1500], bar: [15, 1500]},
+ {bar: [20, 2000], baz: [15, 1500]}
+];
+
+
+/**
+ * @constructor
+ */
+RemoteDebuggerCommandExecutorStub = function() {
+};
+
+
+RemoteDebuggerCommandExecutorStub.prototype.DebuggerCommand = function(cmd) {
+ if ('{"seq":2,"type":"request","command":"scripts","arguments":{"' +
+ 'includeSource":false}}' == cmd) {
+ var response1 =
+ '{"seq":5,"request_seq":2,"type":"response","command":"scripts","' +
+ 'success":true,"body":[{"handle":61,"type":"script","name":"' +
+ 'http://www/~test/t.js","id":59,"lineOffset":0,"columnOffset":0,' +
+ '"lineCount":1,"sourceStart":"function fib(n) {","sourceLength":300,' +
+ '"scriptType":2,"compilationType":0,"context":{"ref":60}}],"refs":[{' +
+ '"handle":60,"type":"context","data":{"type":"page","value":3}}],' +
+ '"running":false}';
+ this.sendResponse_(response1);
+ } else if ('{"seq":3,"type":"request","command":"scripts","arguments":{' +
+ '"ids":[59],"includeSource":true}}' == cmd) {
+ this.sendResponse_(
+ '{"seq":8,"request_seq":3,"type":"response","command":"scripts",' +
+ '"success":true,"body":[{"handle":1,"type":"script","name":' +
+ '"http://www/~test/t.js","id":59,"lineOffset":0,"columnOffset":0,' +
+ '"lineCount":1,"source":"function fib(n) {return n+1;}",' +
+ '"sourceLength":244,"scriptType":2,"compilationType":0,"context":{' +
+ '"ref":0}}],"refs":[{"handle":0,"type":"context","data":{"type":' +
+ '"page","value":3}}],"running":false}');
+ } else {
+ debugPrint('Unexpected command: ' + cmd);
+ }
+};
+
+
+RemoteDebuggerCommandExecutorStub.prototype.sendResponse_ = function(response) {
+ setTimeout(function() {
+ RemoteDebuggerAgent.DebuggerOutput(response);
+ }, 0);
+};
+
+
+/**
+ * @constructor
+ */
+DevToolsHostStub = function() {
+ this.isStub = true;
+ window.domAutomationController = {
+ send: function(text) {
+ debugPrint(text);
+ }
+ };
+};
+
+
+function addDummyResource() {
+ var payload = {
+ requestHeaders : {},
+ requestURL: 'http://google.com/simple_page.html',
+ host: 'google.com',
+ path: 'simple_page.html',
+ lastPathComponent: 'simple_page.html',
+ isMainResource: true,
+ cached: false,
+ mimeType: 'text/html',
+ suggestedFilename: 'simple_page.html',
+ expectedContentLength: 10000,
+ statusCode: 200,
+ contentLength: 10000,
+ responseHeaders: {},
+ type: WebInspector.Resource.Type.Document,
+ finished: true,
+ startTime: new Date(),
+
+ didResponseChange: true,
+ didCompletionChange: true,
+ didTypeChange: true
+ };
+
+ WebInspector.addResource(1, payload);
+ WebInspector.updateResource(1, payload);
+}
+
+
+DevToolsHostStub.prototype.loaded = function() {
+ addDummyResource();
+ uiTests.runAllTests();
+};
+
+
+DevToolsHostStub.prototype.reset = function() {
+};
+
+
+DevToolsHostStub.prototype.getPlatform = function() {
+ return "windows";
+};
+
+
+DevToolsHostStub.prototype.addResourceSourceToFrame = function(
+ identifier, mimeType, element) {
+};
+
+
+DevToolsHostStub.prototype.addSourceToFrame = function(mimeType, source,
+ element) {
+};
+
+
+DevToolsHostStub.prototype.getApplicationLocale = function() {
+ return "en-US";
+};
+
+
+if (!window['DevToolsHost']) {
+ window['RemoteDebuggerAgent'] = new RemoteDebuggerAgentStub();
+ window['RemoteDebuggerCommandExecutor'] =
+ new RemoteDebuggerCommandExecutorStub();
+ window['RemoteToolsAgent'] = new RemoteToolsAgentStub();
+ window['DevToolsHost'] = new DevToolsHostStub();
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/heap_profiler_panel.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/heap_profiler_panel.js
new file mode 100644
index 0000000..eb1dffa
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/heap_profiler_panel.js
@@ -0,0 +1,680 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview Heap profiler panel implementation.
+ */
+
+WebInspector.HeapProfilerPanel = function() {
+ WebInspector.Panel.call(this);
+
+ this.element.addStyleClass("heap-profiler");
+
+ this.sidebarElement = document.createElement("div");
+ this.sidebarElement.id = "heap-snapshot-sidebar";
+ this.sidebarElement.className = "sidebar";
+ this.element.appendChild(this.sidebarElement);
+
+ this.sidebarResizeElement = document.createElement("div");
+ this.sidebarResizeElement.className = "sidebar-resizer-vertical";
+ this.sidebarResizeElement.addEventListener("mousedown", this._startSidebarDragging.bind(this), false);
+ this.element.appendChild(this.sidebarResizeElement);
+
+ this.sidebarTreeElement = document.createElement("ol");
+ this.sidebarTreeElement.className = "sidebar-tree";
+ this.sidebarElement.appendChild(this.sidebarTreeElement);
+
+ this.sidebarTree = new TreeOutline(this.sidebarTreeElement);
+
+ this.snapshotViews = document.createElement("div");
+ this.snapshotViews.id = "heap-snapshot-views";
+ this.element.appendChild(this.snapshotViews);
+
+ this.snapshotButton = new WebInspector.StatusBarButton(WebInspector.UIString("Take heap snapshot."), "heap-snapshot-status-bar-item");
+ this.snapshotButton.addEventListener("click", this._snapshotClicked.bind(this), false);
+
+ this.snapshotViewStatusBarItemsContainer = document.createElement("div");
+ this.snapshotViewStatusBarItemsContainer.id = "heap-snapshot-status-bar-items";
+
+ this.reset();
+};
+
+WebInspector.HeapProfilerPanel.prototype = {
+ toolbarItemClass: "heap-profiler",
+
+ get toolbarItemLabel() {
+ return WebInspector.UIString("Heap");
+ },
+
+ get statusBarItems() {
+ return [this.snapshotButton.element, this.snapshotViewStatusBarItemsContainer];
+ },
+
+ show: function() {
+ WebInspector.Panel.prototype.show.call(this);
+ this._updateSidebarWidth();
+ },
+
+ reset: function() {
+ if (this._snapshots) {
+ var snapshotsLength = this._snapshots.length;
+ for (var i = 0; i < snapshotsLength; ++i) {
+ var snapshot = this._snapshots[i];
+ delete snapshot._snapshotView;
+ }
+ }
+
+ this._snapshots = [];
+
+ this.sidebarTreeElement.removeStyleClass("some-expandable");
+
+ this.sidebarTree.removeChildren();
+ this.snapshotViews.removeChildren();
+
+ this.snapshotViewStatusBarItemsContainer.removeChildren();
+ },
+
+ handleKeyEvent: function(event) {
+ this.sidebarTree.handleKeyEvent(event);
+ },
+
+ addSnapshot: function(snapshot) {
+ this._snapshots.push(snapshot);
+ snapshot.list = this._snapshots;
+ snapshot.listIndex = this._snapshots.length - 1;
+
+ var snapshotsTreeElement = new WebInspector.HeapSnapshotSidebarTreeElement(snapshot);
+ snapshotsTreeElement.small = false;
+ snapshot._snapshotsTreeElement = snapshotsTreeElement;
+
+ this.sidebarTree.appendChild(snapshotsTreeElement);
+
+ this.dispatchEventToListeners("snapshot added");
+ },
+
+ showSnapshot: function(snapshot) {
+ if (!snapshot)
+ return;
+
+ if (this.visibleView)
+ this.visibleView.hide();
+ var view = this.snapshotViewForSnapshot(snapshot);
+ view.show(this.snapshotViews);
+ this.visibleView = view;
+
+ this.snapshotViewStatusBarItemsContainer.removeChildren();
+ var statusBarItems = view.statusBarItems;
+ for (var i = 0; i < statusBarItems.length; ++i)
+ this.snapshotViewStatusBarItemsContainer.appendChild(statusBarItems[i]);
+ },
+
+ showView: function(view)
+ {
+ this.showSnapshot(view.snapshot);
+ },
+
+ snapshotViewForSnapshot: function(snapshot)
+ {
+ if (!snapshot)
+ return null;
+ if (!snapshot._snapshotView)
+ snapshot._snapshotView = new WebInspector.HeapSnapshotView(this, snapshot);
+ return snapshot._snapshotView;
+ },
+
+ closeVisibleView: function()
+ {
+ if (this.visibleView)
+ this.visibleView.hide();
+ delete this.visibleView;
+ },
+
+ _snapshotClicked: function() {
+ devtools.tools.getDebuggerAgent().startProfiling(
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_HEAP_SNAPSHOT |
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_HEAP_STATS |
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_JS_CONSTRUCTORS);
+ },
+
+ _startSidebarDragging: function(event)
+ {
+ WebInspector.elementDragStart(this.sidebarResizeElement, this._sidebarDragging.bind(this), this._endSidebarDragging.bind(this), event, "col-resize");
+ },
+
+ _sidebarDragging: function(event)
+ {
+ this._updateSidebarWidth(event.pageX);
+
+ event.preventDefault();
+ },
+
+ _endSidebarDragging: function(event)
+ {
+ WebInspector.elementDragEnd(event);
+ },
+
+ _updateSidebarWidth: function(width)
+ {
+ if (this.sidebarElement.offsetWidth <= 0) {
+ // The stylesheet hasn"t loaded yet or the window is closed,
+ // so we can"t calculate what is need. Return early.
+ return;
+ }
+
+ if (!("_currentSidebarWidth" in this))
+ this._currentSidebarWidth = this.sidebarElement.offsetWidth;
+ if (typeof width === "undefined")
+ width = this._currentSidebarWidth;
+
+ width = Number.constrain(width, Preferences.minSidebarWidth, window.innerWidth / 2);
+ this._currentSidebarWidth = width;
+ this.sidebarElement.style.width = width + "px";
+ this.snapshotViews.style.left = width + "px";
+ this.snapshotViewStatusBarItemsContainer.style.left = width + "px";
+ this.sidebarResizeElement.style.left = (width - 3) + "px";
+ }
+};
+
+WebInspector.HeapProfilerPanel.prototype.__proto__ = WebInspector.Panel.prototype;
+
+WebInspector.HeapSnapshotView = function(parent, snapshot)
+{
+ WebInspector.View.call(this);
+
+ this.element.addStyleClass("heap-snapshot-view");
+
+ this.parent = parent;
+ this.parent.addEventListener("snapshot added", this._updateBaseOptions, this);
+
+ this.showCountAsPercent = true;
+ this.showSizeAsPercent = true;
+ this.showCountDeltaAsPercent = true;
+ this.showSizeDeltaAsPercent = true;
+
+ this.summaryBar = new WebInspector.SummaryBar(this.categories);
+ this.summaryBar.element.id = "heap-snapshot-summary";
+ this.summaryBar.calculator = new WebInspector.HeapSummaryCalculator(snapshot.used);
+ this.element.appendChild(this.summaryBar.element);
+
+ var columns = { "cons": { title: WebInspector.UIString("Constructor"), disclosure: true, sortable: true },
+ "count": { title: WebInspector.UIString("Count"), width: "54px", sortable: true },
+ "size": { title: WebInspector.UIString("Size"), width: "72px", sort: "descending", sortable: true },
+ "countDelta": { title: WebInspector.UIString("\xb1 Count"), width: "72px", sortable: true },
+ "sizeDelta": { title: WebInspector.UIString("\xb1 Size"), width: "72px", sortable: true } };
+
+ this.dataGrid = new WebInspector.DataGrid(columns);
+ this.dataGrid.addEventListener("sorting changed", this._sortData, this);
+ this.dataGrid.element.addEventListener("mousedown", this._mouseDownInDataGrid.bind(this), true);
+ this.element.appendChild(this.dataGrid.element);
+
+ this.snapshot = snapshot;
+
+ this.baseSelectElement = document.createElement("select");
+ this.baseSelectElement.className = "status-bar-item";
+ this.baseSelectElement.addEventListener("change", this._changeBase.bind(this), false);
+ this._updateBaseOptions();
+ if (this.snapshot.listIndex > 0)
+ this.baseSelectElement.selectedIndex = this.snapshot.listIndex - 1;
+ else
+ this.baseSelectElement.selectedIndex = this.snapshot.listIndex;
+ this._resetDataGridList();
+
+ this.percentButton = new WebInspector.StatusBarButton("", "percent-time-status-bar-item status-bar-item");
+ this.percentButton.addEventListener("click", this._percentClicked.bind(this), false);
+
+ this.refresh();
+
+ this._updatePercentButton();
+};
+
+WebInspector.HeapSnapshotView.prototype = {
+
+ get categories()
+ {
+ return {code: {title: WebInspector.UIString("Code"), color: {r: 255, g: 121, b: 0}}, data: {title: WebInspector.UIString("Objects and Data"), color: {r: 47, g: 102, b: 236}}, other: {title: WebInspector.UIString("Other"), color: {r: 186, g: 186, b: 186}}};
+ },
+
+ get statusBarItems()
+ {
+ return [this.baseSelectElement, this.percentButton.element];
+ },
+
+ get snapshot()
+ {
+ return this._snapshot;
+ },
+
+ set snapshot(snapshot)
+ {
+ this._snapshot = snapshot;
+ },
+
+ show: function(parentElement)
+ {
+ WebInspector.View.prototype.show.call(this, parentElement);
+ this.dataGrid.updateWidths();
+ },
+
+ resize: function()
+ {
+ if (this.dataGrid)
+ this.dataGrid.updateWidths();
+ },
+
+ refresh: function()
+ {
+ this.dataGrid.removeChildren();
+
+ var children = this.snapshotDataGridList.children;
+ var count = children.length;
+ for (var index = 0; index < count; ++index)
+ this.dataGrid.appendChild(children[index]);
+
+ this._updateSummaryGraph();
+ },
+
+ refreshShowAsPercents: function()
+ {
+ this._updatePercentButton();
+ this.refreshVisibleData();
+ },
+
+ refreshVisibleData: function()
+ {
+ var child = this.dataGrid.children[0];
+ while (child) {
+ child.refresh();
+ child = child.traverseNextNode(false, null, true);
+ }
+ this._updateSummaryGraph();
+ },
+
+ _changeBase: function() {
+ if (this.baseSnapshot === this.snapshot.list[this.baseSelectElement.selectedIndex])
+ return;
+
+ this._resetDataGridList();
+ this.refresh();
+ },
+
+ _createSnapshotDataGridList: function()
+ {
+ if (this._snapshotDataGridList)
+ delete this._snapshotDataGridList;
+
+ this._snapshotDataGridList = new WebInspector.HeapSnapshotDataGridList(this, this.baseSnapshot.entries, this.snapshot.entries);
+ return this._snapshotDataGridList;
+ },
+
+ _mouseDownInDataGrid: function(event)
+ {
+ if (event.detail < 2)
+ return;
+
+ var cell = event.target.enclosingNodeOrSelfWithNodeName("td");
+ if (!cell || (!cell.hasStyleClass("count-column") && !cell.hasStyleClass("size-column") && !cell.hasStyleClass("countDelta-column") && !cell.hasStyleClass("sizeDelta-column")))
+ return;
+
+ if (cell.hasStyleClass("count-column"))
+ this.showCountAsPercent = !this.showCountAsPercent;
+ else if (cell.hasStyleClass("size-column"))
+ this.showSizeAsPercent = !this.showSizeAsPercent;
+ else if (cell.hasStyleClass("countDelta-column"))
+ this.showCountDeltaAsPercent = !this.showCountDeltaAsPercent;
+ else if (cell.hasStyleClass("sizeDelta-column"))
+ this.showSizeDeltaAsPercent = !this.showSizeDeltaAsPercent;
+
+ this.refreshShowAsPercents();
+
+ event.preventDefault();
+ event.stopPropagation();
+ },
+
+ get _isShowingAsPercent()
+ {
+ return this.showCountAsPercent && this.showSizeAsPercent && this.showCountDeltaAsPercent && this.showSizeDeltaAsPercent;
+ },
+
+ _percentClicked: function(event)
+ {
+ var currentState = this._isShowingAsPercent;
+ this.showCountAsPercent = !currentState;
+ this.showSizeAsPercent = !currentState;
+ this.showCountDeltaAsPercent = !currentState;
+ this.showSizeDeltaAsPercent = !currentState;
+ this.refreshShowAsPercents();
+ },
+
+ _resetDataGridList: function()
+ {
+ this.baseSnapshot = this.snapshot.list[this.baseSelectElement.selectedIndex];
+ var lastComparator = WebInspector.HeapSnapshotDataGridList.propertyComparator("objectsSize", false);
+ if (this.snapshotDataGridList) {
+ lastComparator = this.snapshotDataGridList.lastComparator;
+ }
+ this.snapshotDataGridList = this._createSnapshotDataGridList();
+ this.snapshotDataGridList.sort(lastComparator, true);
+ },
+
+ _sortData: function()
+ {
+ var sortAscending = this.dataGrid.sortOrder === "ascending";
+ var sortColumnIdentifier = this.dataGrid.sortColumnIdentifier;
+ var sortProperty = {
+ "cons": "constructorName",
+ "count": "objectsCount",
+ "size": "objectsSize",
+ "countDelta": this.showCountDeltaAsPercent ? "objectsCountDeltaPercent" : "objectsCountDelta",
+ "sizeDelta": this.showSizeDeltaAsPercent ? "objectsSizeDeltaPercent" : "objectsSizeDelta"
+ }[sortColumnIdentifier];
+
+ this.snapshotDataGridList.sort(WebInspector.HeapSnapshotDataGridList.propertyComparator(sortProperty, sortAscending));
+
+ this.refresh();
+ },
+
+ _updateBaseOptions: function()
+ {
+ // We're assuming that snapshots can only be added.
+ if (this.baseSelectElement.length == this.snapshot.list.length)
+ return;
+
+ for (var i = this.baseSelectElement.length, n = this.snapshot.list.length; i < n; ++i) {
+ var baseOption = document.createElement("option");
+ baseOption.label = WebInspector.UIString("Compared to %s", this.snapshot.list[i].title);
+ this.baseSelectElement.appendChild(baseOption);
+ }
+ },
+
+ _updatePercentButton: function()
+ {
+ if (this._isShowingAsPercent) {
+ this.percentButton.title = WebInspector.UIString("Show absolute counts and sizes.");
+ this.percentButton.toggled = true;
+ } else {
+ this.percentButton.title = WebInspector.UIString("Show counts and sizes as percentages.");
+ this.percentButton.toggled = false;
+ }
+ },
+
+ _updateSummaryGraph: function()
+ {
+ this.summaryBar.calculator.showAsPercent = this._isShowingAsPercent;
+ this.summaryBar.update(this.snapshot.lowlevels);
+ }
+};
+
+WebInspector.HeapSnapshotView.prototype.__proto__ = WebInspector.View.prototype;
+
+WebInspector.HeapSummaryCalculator = function(total)
+{
+ this.total = total;
+}
+
+WebInspector.HeapSummaryCalculator.prototype = {
+ computeSummaryValues: function(lowLevels)
+ {
+ function highFromLow(type) {
+ if (type == "CODE_TYPE" || type == "SHARED_FUNCTION_INFO_TYPE" || type == "SCRIPT_TYPE") return "code";
+ if (type == "STRING_TYPE" || type == "HEAP_NUMBER_TYPE" || type.match(/^JS_/) || type.match(/_ARRAY_TYPE$/)) return "data";
+ return "other";
+ }
+ var highLevels = {data: 0, code: 0, other: 0};
+ for (var item in lowLevels) {
+ var highItem = highFromLow(item);
+ highLevels[highItem] += lowLevels[item].size;
+ }
+ return {categoryValues: highLevels};
+ },
+
+ formatValue: function(value)
+ {
+ if (this.showAsPercent)
+ return WebInspector.UIString("%.2f%%", value / this.total * 100.0);
+ else
+ return Number.bytesToString(value);
+ },
+
+ get showAsPercent()
+ {
+ return this._showAsPercent;
+ },
+
+ set showAsPercent(x)
+ {
+ this._showAsPercent = x;
+ }
+}
+
+WebInspector.HeapSnapshotSidebarTreeElement = function(snapshot)
+{
+ this.snapshot = snapshot;
+ this.snapshot.title = WebInspector.UIString("Snapshot %d", this.snapshot.number);
+
+ WebInspector.SidebarTreeElement.call(this, "heap-snapshot-sidebar-tree-item", "", "", snapshot, false);
+
+ this.refreshTitles();
+};
+
+WebInspector.HeapSnapshotSidebarTreeElement.prototype = {
+ onselect: function()
+ {
+ WebInspector.panels.heap.showSnapshot(this.snapshot);
+ },
+
+ get mainTitle()
+ {
+ if (this._mainTitle)
+ return this._mainTitle;
+ return this.snapshot.title;
+ },
+
+ set mainTitle(x)
+ {
+ this._mainTitle = x;
+ this.refreshTitles();
+ },
+
+ get subtitle()
+ {
+ if (this._subTitle)
+ return this._subTitle;
+ return WebInspector.UIString("Used %s of %s (%.0f%%)", Number.bytesToString(this.snapshot.used, null, false), Number.bytesToString(this.snapshot.capacity, null, false), this.snapshot.used / this.snapshot.capacity * 100.0);
+ },
+
+ set subtitle(x)
+ {
+ this._subTitle = x;
+ this.refreshTitles();
+ }
+};
+
+WebInspector.HeapSnapshotSidebarTreeElement.prototype.__proto__ = WebInspector.SidebarTreeElement.prototype;
+
+WebInspector.HeapSnapshotDataGridNode = function(snapshotView, baseEntry, snapshotEntry, owningList)
+{
+ WebInspector.DataGridNode.call(this, null, false);
+
+ this.snapshotView = snapshotView;
+ this.list = owningList;
+
+ if (!snapshotEntry)
+ snapshotEntry = { cons: baseEntry.cons, count: 0, size: 0 };
+ this.constructorName = snapshotEntry.cons;
+ this.objectsCount = snapshotEntry.count;
+ this.objectsSize = snapshotEntry.size;
+
+ if (!baseEntry)
+ baseEntry = { count: 0, size: 0 };
+ this.baseObjectsCount = baseEntry.count;
+ this.objectsCountDelta = this.objectsCount - this.baseObjectsCount;
+ this.baseObjectsSize = baseEntry.size;
+ this.objectsSizeDelta = this.objectsSize - this.baseObjectsSize;
+};
+
+WebInspector.HeapSnapshotDataGridNode.prototype = {
+ get data()
+ {
+ var data = {};
+
+ data["cons"] = this.constructorName;
+
+ if (this.snapshotView.showCountAsPercent)
+ data["count"] = WebInspector.UIString("%.2f%%", this.objectsCountPercent);
+ else
+ data["count"] = this.objectsCount;
+
+ if (this.snapshotView.showSizeAsPercent)
+ data["size"] = WebInspector.UIString("%.2f%%", this.objectsSizePercent);
+ else
+ data["size"] = Number.bytesToString(this.objectsSize);
+
+ function signForDelta(delta) {
+ if (delta == 0)
+ return "";
+ if (delta > 0)
+ return "+";
+ else
+ // Math minus sign, same width as plus.
+ return "\u2212";
+ }
+
+ function showDeltaAsPercent(value) {
+ if (value === Number.POSITIVE_INFINITY)
+ return WebInspector.UIString("new");
+ else if (value === Number.NEGATIVE_INFINITY)
+ return WebInspector.UIString("deleted");
+ if (value > 1000.0)
+ return WebInspector.UIString("%s >1000%%", signForDelta(value));
+ return WebInspector.UIString("%s%.2f%%", signForDelta(value), Math.abs(value));
+ }
+
+ if (this.snapshotView.showCountDeltaAsPercent)
+ data["countDelta"] = showDeltaAsPercent(this.objectsCountDeltaPercent);
+ else
+ data["countDelta"] = WebInspector.UIString("%s%d", signForDelta(this.objectsCountDelta), Math.abs(this.objectsCountDelta));
+
+ if (this.snapshotView.showSizeDeltaAsPercent)
+ data["sizeDelta"] = showDeltaAsPercent(this.objectsSizeDeltaPercent);
+ else
+ data["sizeDelta"] = WebInspector.UIString("%s%s", signForDelta(this.objectsSizeDelta), Number.bytesToString(Math.abs(this.objectsSizeDelta)));
+
+ return data;
+ },
+
+ get objectsCountPercent()
+ {
+ return this.objectsCount / this.list.objectsCount * 100.0;
+ },
+
+ get objectsSizePercent()
+ {
+ return this.objectsSize / this.list.objectsSize * 100.0;
+ },
+
+ get objectsCountDeltaPercent()
+ {
+ if (this.baseObjectsCount > 0) {
+ if (this.objectsCount > 0)
+ return this.objectsCountDelta / this.baseObjectsCount * 100.0;
+ else
+ return Number.NEGATIVE_INFINITY;
+ } else
+ return Number.POSITIVE_INFINITY;
+ },
+
+ get objectsSizeDeltaPercent()
+ {
+ if (this.baseObjectsSize > 0) {
+ if (this.objectsSize > 0)
+ return this.objectsSizeDelta / this.baseObjectsSize * 100.0;
+ else
+ return Number.NEGATIVE_INFINITY;
+ } else
+ return Number.POSITIVE_INFINITY;
+ }
+};
+
+WebInspector.HeapSnapshotDataGridNode.prototype.__proto__ = WebInspector.DataGridNode.prototype;
+
+WebInspector.HeapSnapshotDataGridList = function(snapshotView, baseEntries, snapshotEntries)
+{
+ this.list = this;
+ this.snapshotView = snapshotView;
+ this.children = [];
+ this.lastComparator = null;
+ this.populateChildren(baseEntries, snapshotEntries);
+};
+
+WebInspector.HeapSnapshotDataGridList.prototype = {
+ appendChild: function(child)
+ {
+ this.insertChild(child, this.children.length);
+ },
+
+ insertChild: function(child, index)
+ {
+ this.children.splice(index, 0, child);
+ },
+
+ removeChildren: function()
+ {
+ this.children = [];
+ },
+
+ populateChildren: function(baseEntries, snapshotEntries)
+ {
+ for (var item in snapshotEntries)
+ this.appendChild(new WebInspector.HeapSnapshotDataGridNode(this.snapshotView, baseEntries[item], snapshotEntries[item], this));
+
+ for (item in baseEntries) {
+ if (!(item in snapshotEntries))
+ this.appendChild(new WebInspector.HeapSnapshotDataGridNode(this.snapshotView, baseEntries[item], null, this));
+ }
+ },
+
+ sort: function(comparator, force) {
+ if (!force && this.lastComparator === comparator)
+ return;
+
+ this.children.sort(comparator);
+ this.lastComparator = comparator;
+ },
+
+ get objectsCount() {
+ if (!this._objectsCount) {
+ this._objectsCount = 0;
+ for (var i = 0, n = this.children.length; i < n; ++i) {
+ this._objectsCount += this.children[i].objectsCount;
+ }
+ }
+ return this._objectsCount;
+ },
+
+ get objectsSize() {
+ if (!this._objectsSize) {
+ this._objectsSize = 0;
+ for (var i = 0, n = this.children.length; i < n; ++i) {
+ this._objectsSize += this.children[i].objectsSize;
+ }
+ }
+ return this._objectsSize;
+ }
+};
+
+WebInspector.HeapSnapshotDataGridList.propertyComparators = [{}, {}];
+
+WebInspector.HeapSnapshotDataGridList.propertyComparator = function(property, isAscending)
+{
+ var comparator = this.propertyComparators[(isAscending ? 1 : 0)][property];
+ if (!comparator) {
+ comparator = function(lhs, rhs) {
+ var l = lhs[property], r = rhs[property];
+ var result = l < r ? -1 : (l > r ? 1 : 0);
+ return isAscending ? result : -result;
+ };
+ this.propertyComparators[(isAscending ? 1 : 0)][property] = comparator;
+ }
+ return comparator;
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/back.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/back.png
new file mode 100644
index 0000000..9363960
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/back.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/checker.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/checker.png
new file mode 100644
index 0000000..8349908
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/checker.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/clearConsoleButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/clearConsoleButtonGlyph.png
new file mode 100644
index 0000000..b1f9465
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/clearConsoleButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/closeButtons.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/closeButtons.png
new file mode 100644
index 0000000..28158a4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/closeButtons.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/consoleButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/consoleButtonGlyph.png
new file mode 100644
index 0000000..d10d43c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/consoleButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/cookie.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/cookie.png
new file mode 100644
index 0000000..90c3c15
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/cookie.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/database.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/database.png
new file mode 100644
index 0000000..339efa6
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/database.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/databaseTable.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/databaseTable.png
new file mode 100644
index 0000000..3718708
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/databaseTable.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerContinue.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerContinue.png
new file mode 100644
index 0000000..d90a855
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerContinue.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerPause.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerPause.png
new file mode 100644
index 0000000..97f958a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerPause.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepInto.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepInto.png
new file mode 100644
index 0000000..277f126
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepInto.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOut.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOut.png
new file mode 100644
index 0000000..3032e32
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOut.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOver.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOver.png
new file mode 100644
index 0000000..7d47245
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/debuggerStepOver.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDown.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDown.png
new file mode 100644
index 0000000..cffc835
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDown.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownBlack.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownBlack.png
new file mode 100644
index 0000000..4b49c13
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownBlack.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownWhite.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownWhite.png
new file mode 100644
index 0000000..aebae12
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallDownWhite.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRight.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRight.png
new file mode 100644
index 0000000..a3102ea
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRight.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightBlack.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightBlack.png
new file mode 100644
index 0000000..2c45859
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightBlack.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDown.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDown.png
new file mode 100644
index 0000000..035c069
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDown.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownBlack.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownBlack.png
new file mode 100644
index 0000000..86f67bd
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownBlack.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownWhite.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownWhite.png
new file mode 100644
index 0000000..972d794
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightDownWhite.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightWhite.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightWhite.png
new file mode 100644
index 0000000..a10168f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/disclosureTriangleSmallRightWhite.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/dockButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/dockButtonGlyph.png
new file mode 100644
index 0000000..7052f4b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/dockButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/elementsIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/elementsIcon.png
new file mode 100644
index 0000000..fde3db9
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/elementsIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableOutlineButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableOutlineButtonGlyph.png
new file mode 100644
index 0000000..85e0bd6
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableOutlineButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableSolidButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableSolidButtonGlyph.png
new file mode 100644
index 0000000..25b2e96
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/enableSolidButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorIcon.png
new file mode 100644
index 0000000..c697263
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorMediumIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorMediumIcon.png
new file mode 100644
index 0000000..6ca32bb
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/errorMediumIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/excludeButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/excludeButtonGlyph.png
new file mode 100644
index 0000000..5128576
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/excludeButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/focusButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/focusButtonGlyph.png
new file mode 100644
index 0000000..b71807c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/focusButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/forward.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/forward.png
new file mode 100644
index 0000000..ad70f3e
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/forward.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeader.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeader.png
new file mode 100644
index 0000000..6cbefb7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeader.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderPressed.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderPressed.png
new file mode 100644
index 0000000..1153506
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderPressed.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelected.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelected.png
new file mode 100644
index 0000000..71d5af6
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelected.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelectedPressed.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelectedPressed.png
new file mode 100644
index 0000000..7047dbe
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/glossyHeaderSelectedPressed.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/goArrow.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/goArrow.png
new file mode 100644
index 0000000..f318a56
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/goArrow.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutLeft.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutLeft.png
new file mode 100644
index 0000000..6426dbd
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutLeft.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutRight.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutRight.png
new file mode 100644
index 0000000..8c87eae
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/graphLabelCalloutRight.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/largerResourcesButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/largerResourcesButtonGlyph.png
new file mode 100644
index 0000000..71256d6
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/largerResourcesButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/localStorage.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/localStorage.png
new file mode 100644
index 0000000..44a3019
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/localStorage.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/nodeSearchButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/nodeSearchButtonGlyph.png
new file mode 100644
index 0000000..faf5df2
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/nodeSearchButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrow.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrow.png
new file mode 100644
index 0000000..d55b865
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrow.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrowActive.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrowActive.png
new file mode 100644
index 0000000..ef3f259
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneBottomGrowActive.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneGrowHandleLine.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneGrowHandleLine.png
new file mode 100644
index 0000000..4eaf61b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/paneGrowHandleLine.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/pauseOnExceptionButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/pauseOnExceptionButtonGlyph.png
new file mode 100644
index 0000000..c3cec5f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/pauseOnExceptionButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/percentButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/percentButtonGlyph.png
new file mode 100644
index 0000000..0ace3b7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/percentButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileGroupIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileGroupIcon.png
new file mode 100644
index 0000000..44616d4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileGroupIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileIcon.png
new file mode 100644
index 0000000..8008f9b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileSmallIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileSmallIcon.png
new file mode 100644
index 0000000..7935520
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profileSmallIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesIcon.png
new file mode 100644
index 0000000..ecd5b04
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesSilhouette.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesSilhouette.png
new file mode 100644
index 0000000..42bb966
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/profilesSilhouette.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/radioDot.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/radioDot.png
new file mode 100644
index 0000000..609878f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/radioDot.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordButtonGlyph.png
new file mode 100644
index 0000000..bfdad1a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordToggledButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordToggledButtonGlyph.png
new file mode 100644
index 0000000..2c22f87
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/recordToggledButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/reloadButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/reloadButtonGlyph.png
new file mode 100644
index 0000000..28e047a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/reloadButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceCSSIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceCSSIcon.png
new file mode 100644
index 0000000..aead6a7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceCSSIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIcon.png
new file mode 100644
index 0000000..1683a09
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIconSmall.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIconSmall.png
new file mode 100644
index 0000000..468ced9
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceDocumentIconSmall.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceJSIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceJSIcon.png
new file mode 100644
index 0000000..9ef6ed0
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourceJSIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIcon.png
new file mode 100644
index 0000000..0ed37b6
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIconSmall.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIconSmall.png
new file mode 100644
index 0000000..0fa967d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcePlainIconSmall.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesIcon.png
new file mode 100644
index 0000000..982424d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSilhouette.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSilhouette.png
new file mode 100644
index 0000000..9c8bb53
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSilhouette.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSizeGraphIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSizeGraphIcon.png
new file mode 100644
index 0000000..e60dbe5
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesSizeGraphIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesTimeGraphIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesTimeGraphIcon.png
new file mode 100644
index 0000000..c6953e9
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/resourcesTimeGraphIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsIcon.png
new file mode 100644
index 0000000..213b31e
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsSilhouette.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsSilhouette.png
new file mode 100644
index 0000000..206396f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/scriptsSilhouette.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBlue.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBlue.png
new file mode 100644
index 0000000..9c990f4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBlue.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBrightBlue.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBrightBlue.png
new file mode 100644
index 0000000..b1d8055
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallBrightBlue.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallGray.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallGray.png
new file mode 100644
index 0000000..4f3c068
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallGray.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallWhite.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallWhite.png
new file mode 100644
index 0000000..85f430d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/searchSmallWhite.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segment.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segment.png
new file mode 100644
index 0000000..759266e
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segment.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentEnd.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentEnd.png
new file mode 100644
index 0000000..72672ff
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentEnd.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHover.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHover.png
new file mode 100644
index 0000000..c5017f4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHover.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHoverEnd.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHoverEnd.png
new file mode 100644
index 0000000..d51363d
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentHoverEnd.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelected.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelected.png
new file mode 100644
index 0000000..c92f584
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelected.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelectedEnd.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelectedEnd.png
new file mode 100644
index 0000000..be5e0852
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/segmentSelectedEnd.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/sessionStorage.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/sessionStorage.png
new file mode 100644
index 0000000..4d50e35
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/sessionStorage.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDimple.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDimple.png
new file mode 100644
index 0000000..584ffd4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDimple.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDividerBackground.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDividerBackground.png
new file mode 100644
index 0000000..1120a7f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/splitviewDividerBackground.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBackground.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBackground.png
new file mode 100644
index 0000000..b466a49
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBackground.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBottomBackground.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBottomBackground.png
new file mode 100644
index 0000000..fb5c9e4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarBottomBackground.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarButtons.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarButtons.png
new file mode 100644
index 0000000..e8090cb
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarButtons.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButton.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButton.png
new file mode 100644
index 0000000..9b3abdd
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButton.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButtonSelected.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButtonSelected.png
new file mode 100644
index 0000000..8189c43
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarMenuButtonSelected.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerHorizontal.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerHorizontal.png
new file mode 100644
index 0000000..56deeab
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerHorizontal.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerVertical.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerVertical.png
new file mode 100644
index 0000000..7fc145277
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/statusbarResizerVertical.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/storageIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/storageIcon.png
new file mode 100644
index 0000000..79c7bb3
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/storageIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillBlue.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillBlue.png
new file mode 100644
index 0000000..c7c273b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillBlue.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGray.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGray.png
new file mode 100644
index 0000000..9ff37ef
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGray.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGreen.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGreen.png
new file mode 100644
index 0000000..cc5a8f3
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillGreen.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillOrange.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillOrange.png
new file mode 100644
index 0000000..08a81e4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillOrange.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillPurple.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillPurple.png
new file mode 100644
index 0000000..565a05c
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillPurple.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillRed.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillRed.png
new file mode 100644
index 0000000..c3a1b9b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillRed.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillYellow.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillYellow.png
new file mode 100644
index 0000000..780045b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelineHollowPillYellow.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillBlue.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillBlue.png
new file mode 100644
index 0000000..c897faa
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillBlue.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGray.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGray.png
new file mode 100644
index 0000000..2128896
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGray.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGreen.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGreen.png
new file mode 100644
index 0000000..9b66125
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillGreen.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillOrange.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillOrange.png
new file mode 100644
index 0000000..dd944fb
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillOrange.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillPurple.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillPurple.png
new file mode 100644
index 0000000..21b96f7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillPurple.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillRed.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillRed.png
new file mode 100644
index 0000000..f5e213b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillRed.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillYellow.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillYellow.png
new file mode 100644
index 0000000..ae2a5a23
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/timelinePillYellow.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloon.png
new file mode 100644
index 0000000..4cdf738
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloonBottom.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloonBottom.png
new file mode 100644
index 0000000..3317a5a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipBalloonBottom.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIcon.png
new file mode 100644
index 0000000..8ca6124
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIconPressed.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIconPressed.png
new file mode 100644
index 0000000..443e410
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/tipIconPressed.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/toolbarItemSelected.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/toolbarItemSelected.png
new file mode 100644
index 0000000..bd681f18
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/toolbarItemSelected.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleBlack.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleBlack.png
new file mode 100644
index 0000000..0821112
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleBlack.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleWhite.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleWhite.png
new file mode 100644
index 0000000..1667b51
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeDownTriangleWhite.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleBlack.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleBlack.png
new file mode 100644
index 0000000..90de820
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleBlack.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleWhite.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleWhite.png
new file mode 100644
index 0000000..2b6a82f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeRightTriangleWhite.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleBlack.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleBlack.png
new file mode 100644
index 0000000..ef69dbc
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleBlack.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleWhite.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleWhite.png
new file mode 100644
index 0000000..43ce4be
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/treeUpTriangleWhite.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/undockButtonGlyph.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/undockButtonGlyph.png
new file mode 100644
index 0000000..eed2b65
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/undockButtonGlyph.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputIcon.png
new file mode 100644
index 0000000..325023f
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputPreviousIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputPreviousIcon.png
new file mode 100644
index 0000000..068d572
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputPreviousIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputResultIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputResultIcon.png
new file mode 100644
index 0000000..794a5ca
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/userInputResultIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningIcon.png
new file mode 100644
index 0000000..d5e4c82
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningMediumIcon.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningMediumIcon.png
new file mode 100644
index 0000000..291e111
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningMediumIcon.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningsErrors.png b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningsErrors.png
new file mode 100644
index 0000000..878b593
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/images/warningsErrors.png
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inject.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inject.js
new file mode 100644
index 0000000..8a9b199
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inject.js
@@ -0,0 +1,50 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview Javascript that is being injected into the inspectable page
+ * while debugging.
+ */
+goog.provide('devtools.Injected');
+
+
+/**
+ * Main injected object.
+ * @constructor.
+ */
+devtools.Injected = function() {
+};
+
+
+/**
+ * Dispatches given method with given args on the host object.
+ * @param {string} method Method name.
+ */
+devtools.Injected.prototype.InspectorController = function(method, var_args) {
+ var args = Array.prototype.slice.call(arguments, 1);
+ return InspectorController[method].apply(InspectorController, args);
+};
+
+
+/**
+ * Dispatches given method with given args on the InjectedScript.
+ * @param {string} method Method name.
+ */
+devtools.Injected.prototype.InjectedScript = function(method, var_args) {
+ var args = Array.prototype.slice.call(arguments, 1);
+ var result = InjectedScript[method].apply(InjectedScript, args);
+ return result;
+};
+
+
+// Plugging into upstreamed support.
+InjectedScript._window = function() {
+ return contentWindow;
+};
+
+
+// Plugging into upstreamed support.
+Object.className = function(obj) {
+ return (obj == null) ? "null" : obj.constructor.name;
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.css b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.css
new file mode 100644
index 0000000..ea6f661
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.css
@@ -0,0 +1,3302 @@
+/*
+ * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2009 Anthony Ricaud <rik@webkit.org>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+body {
+ cursor: default;
+ height: 100%;
+ width: 100%;
+ overflow: hidden;
+ font-family: Lucida Grande, sans-serif;
+ font-size: 10px;
+ margin: 0;
+ -webkit-text-size-adjust: none;
+ -webkit-user-select: none;
+}
+
+* {
+ -webkit-box-sizing: border-box;
+}
+
+:focus {
+ outline: none;
+}
+
+input[type="search"]:focus, input[type="text"]:focus {
+ outline: auto 5px -webkit-focus-ring-color;
+}
+
+iframe, a img {
+ border: none;
+}
+
+img {
+ -webkit-user-drag: none;
+}
+
+.hidden {
+ display: none !important;
+}
+
+#toolbar {
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ height: 56px;
+ display: -webkit-box;
+ padding: 0 5px;
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(191, 191, 191)), to(rgb(151, 151, 151)));
+ border-bottom: 1px solid rgb(80, 80, 80);
+ -webkit-box-orient: horizontal;
+ -webkit-background-origin: padding;
+ -webkit-background-clip: padding;
+}
+
+body.inactive #toolbar {
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(233, 233, 233)), to(rgb(207, 207, 207)));
+ border-bottom: 1px solid rgb(64%, 64%, 64%);
+}
+
+body.detached.platform-mac-leopard #toolbar {
+ background: transparent !important;
+}
+
+body.attached #toolbar {
+ height: 34px;
+ border-top: 1px solid rgb(100, 100, 100);
+ cursor: row-resize;
+ padding-left: 0;
+}
+
+body.attached.inactive #toolbar {
+ border-top: 1px solid rgb(64%, 64%, 64%);
+}
+
+.toolbar-item {
+ display: -webkit-box;
+ padding: 4px 6px;
+ margin: 0;
+ background-color: transparent;
+ border-style: none;
+ border-color: transparent;
+ -webkit-box-orient: vertical;
+ -webkit-box-align: center;
+ -webkit-box-pack: end;
+}
+
+.toolbar-item.toggleable.toggled-on {
+ border-width: 0 2px 0 2px;
+ padding: 4px 4px;
+ -webkit-border-image: url(Images/toolbarItemSelected.png) 0 2 0 2;
+}
+
+.toolbar-item.flexable-space {
+ -webkit-box-flex: 1;
+ visibility: hidden;
+}
+
+.toolbar-item input {
+ margin-bottom: 8px;
+}
+
+.toolbar-icon {
+ display: inline-block;
+ width: 32px;
+ height: 32px;
+ -webkit-background-size: 100% auto;
+}
+
+body.attached .toolbar-icon {
+ width: 24px;
+ height: 24px;
+ vertical-align: middle;
+}
+
+.toolbar-item:active .toolbar-icon {
+ background-position: 0 32px;
+}
+
+body.attached .toolbar-item:active .toolbar-icon {
+ background-position: 0 24px;
+}
+
+.toolbar-label {
+ font-size: 11px;
+ font-family: Lucida Grande, sans-serif;
+ text-shadow: rgba(255, 255, 255, 0.5) 0 1px 0;
+}
+
+.toolbar-item.toggleable:active .toolbar-label {
+ text-shadow: none;
+}
+
+body.attached .toolbar-label {
+ display: inline-block;
+ vertical-align: middle;
+ margin-left: 3px;
+}
+
+body.attached #search-toolbar-label {
+ display: none;
+}
+
+#search {
+ width: 205px;
+ font-size: 16px;
+ margin-bottom: 5px;
+}
+
+body.attached #search {
+ font-size: 11px;
+ margin-bottom: 8px;
+}
+
+#search-results-matches {
+ font-size: 11px;
+ text-shadow: rgba(255, 255, 255, 0.5) 0 1px 0;
+ margin-bottom: 22px;
+}
+
+body.attached #search-results-matches {
+ margin-bottom: 6px;
+}
+
+.toolbar-item.elements .toolbar-icon {
+ background-image: url(Images/elementsIcon.png);
+}
+
+.toolbar-item.resources .toolbar-icon {
+ background-image: url(Images/resourcesIcon.png);
+}
+
+.toolbar-item.scripts .toolbar-icon {
+ background-image: url(Images/scriptsIcon.png);
+}
+
+.toolbar-item.storage .toolbar-icon {
+ background-image: url(Images/storageIcon.png);
+}
+
+.toolbar-item.profiles .toolbar-icon {
+ background-image: url(Images/profilesIcon.png);
+}
+
+#close-button {
+ width: 14px;
+ height: 14px;
+ background-image: url(Images/closeButtons.png);
+ background-position: 0 0;
+ background-color: transparent;
+ border: 0 none transparent;
+ margin: 5px 0;
+}
+
+#close-button:hover {
+ background-position: 14px 0;
+}
+
+#close-button:active {
+ background-position: 28px 0;
+}
+
+body.detached .toolbar-item.close {
+ display: none;
+}
+
+#main {
+ position: absolute;
+ z-index: 1;
+ top: 56px;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ overflow: hidden;
+ background-color: white;
+}
+
+body.attached #main {
+ top: 34px;
+}
+
+#main-panels {
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 23px;
+ overflow: hidden;
+}
+
+#main-status-bar {
+ position: absolute;
+ bottom: 0;
+ left: 0;
+ right: 0;
+}
+
+body.drawer-visible #main-status-bar {
+ height: 24px;
+ background-image: url(Images/statusbarResizerVertical.png), url(Images/statusbarBackground.png);
+ background-repeat: no-repeat, repeat-x;
+ background-position: right center, center;
+ cursor: row-resize;
+}
+
+body.drawer-visible #main-status-bar * {
+ cursor: default;
+}
+
+body.drawer-visible #main-panels {
+ bottom: 24px;
+}
+
+.status-bar {
+ background-color: rgb(235, 235, 235);
+ background-image: url(Images/statusbarBackground.png);
+ background-repeat: repeat-x;
+ white-space: nowrap;
+ height: 23px;
+ overflow: hidden;
+ z-index: 12;
+}
+
+.status-bar > div {
+ display: inline-block;
+ vertical-align: top;
+}
+
+.status-bar-item {
+ display: inline-block;
+ height: 24px;
+ padding: 0;
+ margin-left: -1px;
+ margin-right: 0;
+ vertical-align: top;
+ border: 0 transparent none;
+ background-color: transparent;
+}
+
+.status-bar-item:active {
+ position: relative;
+ z-index: 200;
+}
+
+.glyph {
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ background-color: rgba(0, 0, 0, 0.75);
+ z-index: 1;
+}
+
+.glyph.shadow {
+ top: 1px;
+ background-color: white !important;
+ z-index: 0;
+}
+
+button.status-bar-item {
+ position: relative;
+ width: 32px;
+ background-image: url(Images/statusbarButtons.png);
+ background-position: 0 0;
+}
+
+button.status-bar-item:active {
+ background-position: 32px 0 !important;
+}
+
+button.status-bar-item .glyph.shadow {
+ background-color: rgba(255, 255, 255, 0.33) !important;
+}
+
+button.status-bar-item.toggled-on .glyph {
+ background-color: rgb(66, 129, 235);
+}
+
+button.status-bar-item:disabled {
+ opacity: 0.5;
+ background-position: 0 0 !important;
+}
+
+select.status-bar-item {
+ min-width: 48px;
+ border-width: 0 17px 0 2px;
+ padding: 0 2px 0 6px;
+ font-weight: bold;
+ color: rgb(48, 48, 48);
+ text-shadow: rgba(255, 255, 255, 0.75) 0 1px 0;
+ -webkit-border-image: url(Images/statusbarMenuButton.png) 0 17 0 2;
+ -webkit-border-radius: 0;
+ -webkit-appearance: none;
+}
+
+select.status-bar-item:active {
+ color: black;
+ -webkit-border-image: url(Images/statusbarMenuButtonSelected.png) 0 17 0 2;
+}
+
+#dock-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/undockButtonGlyph.png);
+}
+
+body.detached #dock-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/dockButtonGlyph.png);
+}
+
+#console-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/consoleButtonGlyph.png);
+}
+
+#clear-console-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/clearConsoleButtonGlyph.png);
+}
+
+#changes-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/consoleButtonGlyph.png); /* TODO: Needs Image for Changes Toggle Button */
+}
+
+#clear-changes-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/clearConsoleButtonGlyph.png);
+}
+
+#count-items {
+ position: absolute;
+ right: 16px;
+ top: 0;
+ cursor: pointer;
+ padding: 6px 2px;
+ font-size: 10px;
+ height: 19px;
+}
+
+#changes-count, #error-warning-count {
+ display: inline;
+}
+
+#error-warning-count:hover, #changes-count:hover {
+ border-bottom: 1px solid rgb(96, 96, 96);
+}
+
+#style-changes-count::before {
+ content: url(Images/styleIcon.png); /* TODO: Needs Image for Style Changes Icon */
+ width: 10px;
+ height: 10px;
+ vertical-align: -1px;
+ margin-right: 2px;
+}
+
+#error-count::before {
+ content: url(Images/errorIcon.png);
+ width: 10px;
+ height: 10px;
+ vertical-align: -1px;
+ margin-right: 2px;
+}
+
+#changes-count + #error-warning-count, #error-count + #warning-count {
+ margin-left: 6px;
+}
+
+#warning-count::before {
+ content: url(Images/warningIcon.png);
+ width: 10px;
+ height: 10px;
+ vertical-align: -1px;
+ margin-right: 2px;
+}
+
+#drawer {
+ display: none;
+ position: absolute;
+ bottom: 0;
+ left: 0;
+ right: 0;
+ height: 200px;
+ background-color: white;
+ background-image: url(Images/statusbarBottomBackground.png);
+ background-repeat: repeat-x;
+ background-position: bottom;
+}
+
+body.drawer-visible #drawer {
+ display: block;
+}
+
+#drawer-status-bar {
+ position: absolute;
+ bottom: 0;
+ left: 0;
+ right: 0;
+ background: none;
+}
+
+#console-messages {
+ position: absolute;
+ z-index: 0;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 23px;
+ font-size: initial;
+ font-family: monospace;
+ padding: 2px 0;
+ overflow-y: overlay;
+ -webkit-user-select: text;
+ -webkit-text-size-adjust: auto;
+}
+
+#console-prompt {
+ position: relative;
+ padding: 1px 22px 1px 24px;
+ min-height: 16px;
+ white-space: pre-wrap;
+ -webkit-user-modify: read-write-plaintext-only;
+}
+
+#console-prompt::before {
+ background-image: url(Images/userInputIcon.png);
+}
+
+.console-user-command-result.console-log-level::before {
+ background-image: url(Images/userInputResultIcon.png);
+}
+
+.console-message, .console-user-command {
+ position: relative;
+ border-bottom: 1px solid rgb(240, 240, 240);
+ padding: 1px 22px 1px 24px;
+ min-height: 16px;
+}
+
+.console-adjacent-user-command-result {
+ border-bottom: none;
+}
+
+.console-adjacent-user-command-result + .console-user-command-result.console-log-level::before {
+ background-image: none;
+}
+
+.console-message::before, .console-user-command::before, #console-prompt::before, .console-group-title::before {
+ position: absolute;
+ display: block;
+ content: "";
+ left: 7px;
+ top: 0.8em;
+ width: 10px;
+ height: 10px;
+ margin-top: -5px;
+ -webkit-user-select: none;
+}
+
+.console-message .bubble {
+ display: inline-block;
+ height: 14px;
+ background-color: rgb(128, 151, 189);
+ vertical-align: middle;
+ white-space: nowrap;
+ padding: 1px 4px;
+ margin-top: -2px;
+ margin-right: 4px;
+ text-align: left;
+ font-size: 11px;
+ font-family: Helvetia, Arial, sans-serif;
+ font-weight: bold;
+ text-shadow: none;
+ color: white;
+ -webkit-border-radius: 7px;
+}
+
+.console-message-text {
+ white-space: pre-wrap;
+}
+
+.repeated-message {
+ padding-left: 6px;
+}
+
+.repeated-message.console-error-level::before, .repeated-message.console-warning-level:before, .repeated-message.console-debug-level:before {
+ visibility: hidden;
+}
+
+.console-group .console-group > .console-group-messages {
+ margin-left: 16px;
+}
+
+.console-group-title {
+ font-weight: bold;
+}
+
+.console-group-title::before {
+ background-image: url(Images/disclosureTriangleSmallDown.png);
+ top: 0.6em;
+ width: 11px;
+ height: 12px;
+}
+
+.console-group.collapsed .console-group-title::before {
+ background-image: url(Images/disclosureTriangleSmallRight.png);
+}
+
+.console-group.collapsed > .console-group-messages {
+ display: none;
+}
+
+.console-error-level .console-message-text {
+ color: red;
+}
+
+.console-debug-level .console-message-text {
+ color: blue;
+}
+
+.console-debug-level::before {
+ background-image: url(Images/searchSmallBrightBlue.png);
+}
+
+.console-error-level::before {
+ background-image: url(Images/errorIcon.png);
+}
+
+.console-warning-level::before {
+ background-image: url(Images/warningIcon.png);
+}
+
+.console-user-command .console-message {
+ margin-left: -24px;
+ padding-right: 0;
+ border-bottom: none;
+}
+
+.console-user-command::before {
+ background-image: url(Images/userInputPreviousIcon.png);
+}
+
+.console-user-command > .console-message-text {
+ color: rgb(0, 128, 255);
+}
+
+#console-messages a {
+ color: rgb(33%, 33%, 33%);
+ cursor: pointer;
+}
+
+#console-messages a:hover {
+ color: rgb(15%, 15%, 15%);
+}
+
+.console-message-url {
+ float: right;
+}
+
+.console-group-messages .section {
+ margin: 0 0 0 12px !important;
+}
+
+.console-group-messages .section .header {
+ padding: 0 8px 0 0;
+ background-image: none;
+ border: none;
+ min-height: 0;
+}
+
+.console-group-messages .section .header::before {
+ position: absolute;
+ top: 1px;
+ left: 1px;
+ width: 8px;
+ height: 8px;
+ content: url(Images/treeRightTriangleBlack.png);
+}
+
+.console-group-messages .section.expanded .header::before {
+ content: url(Images/treeDownTriangleBlack.png);
+}
+
+.console-group-messages .section .header .title {
+ color: black;
+ font-weight: normal;
+}
+
+.console-group-messages .section .properties li .info {
+ padding-top: 0;
+ padding-bottom: 0;
+ color: rgb(60%, 60%, 60%);
+}
+
+.console-group-messages .outline-disclosure {
+ padding-left: 0;
+}
+
+.console-group-messages .outline-disclosure > ol {
+ padding: 0 0 0 12px !important;
+}
+
+.console-group-messages .outline-disclosure, .console-group-messages .outline-disclosure ol {
+ font-size: inherit;
+ line-height: 1em;
+}
+
+.console-group-messages .outline-disclosure.single-node li {
+ padding-left: 2px;
+}
+
+.console-group-messages .outline-disclosure li .selection {
+ margin-left: -6px;
+ margin-right: -6px;
+}
+
+.console-group-messages .add-attribute {
+ display: none;
+}
+
+.console-formatted-object, .console-formatted-node {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+}
+
+.console-formatted-object .section, .console-formatted-node .section {
+ position: static;
+}
+
+.console-formatted-object .properties, .console-formatted-node .properties {
+ padding-left: 0 !important;
+}
+
+.error-message {
+ color: red;
+}
+
+.auto-complete-text {
+ color: rgb(128, 128, 128);
+ -webkit-user-select: none;
+ -webkit-user-modify: read-only;
+}
+
+.panel {
+ display: none;
+ overflow: hidden;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+}
+
+.panel.visible {
+ display: block;
+}
+
+.resource-view {
+ display: none;
+ overflow: hidden;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ overflow: hidden;
+}
+
+.resource-view.visible {
+ display: block;
+}
+
+.resource-view.headers-visible {
+ overflow-y: auto;
+ overflow-x: hidden;
+}
+
+.resource-view-headers {
+ display: none;
+ padding: 6px;
+ border-bottom: 1px solid rgb(64%, 64%, 64%);
+ background-color: white;
+ -webkit-user-select: text;
+}
+
+.resource-view-headers .outline-disclosure .parent {
+ -webkit-user-select: none;
+ font-weight: bold;
+}
+
+.resource-view.headers-visible .resource-view-headers {
+ display: block;
+}
+
+.resource-view-headers .outline-disclosure .children li {
+ white-space: nowrap;
+}
+
+.resource-view-headers .outline-disclosure li.expanded .header-count {
+ display: none;
+}
+
+.resource-view-headers .outline-disclosure .header-name {
+ color: rgb(33%, 33%, 33%);
+ display: inline-block;
+ width: 105px;
+ text-align: right;
+ margin-right: 0.5em;
+ font-weight: bold;
+ vertical-align: top;
+ overflow: hidden;
+ text-overflow: ellipsis;
+}
+
+.resource-view-headers .outline-disclosure .header-value {
+ display: inline-block;
+ white-space: normal;
+ word-break: break-word;
+ vertical-align: top;
+ margin-right: 100px;
+}
+
+.resource-view-headers .outline-disclosure .raw-form-data {
+ white-space:pre-wrap;
+}
+
+.resource-view .resource-view-content {
+ position: absolute;
+ top: 0;
+ right: 0;
+ left: 0;
+ bottom: 0;
+}
+
+.resource-view.headers-visible .resource-view-content {
+ position: relative;
+ top: auto;
+ right: auto;
+ left: auto;
+ bottom: auto;
+}
+
+.resource-view.headers-visible .source-view-frame {
+ height: auto;
+ vertical-align: top;
+}
+
+.invisible {
+ color: inherit;
+ text-decoration: none;
+}
+
+.webkit-line-gutter-backdrop {
+ /* Keep this in sync with view-source.css (.webkit-line-gutter-backdrop) */
+ width: 31px;
+ background-color: rgb(240, 240, 240);
+ border-right: 1px solid rgb(187, 187, 187);
+ position: absolute;
+ z-index: -1;
+ left: 0;
+ top: 0;
+ height: 100%
+}
+
+.resource-view.font .resource-view-content {
+ font-size: 60px;
+ white-space: pre-wrap;
+ word-wrap: break-word;
+ text-align: center;
+ padding: 15px;
+}
+
+.resource-view.image .resource-view-content > .image {
+ padding: 20px 20px 10px 20px;
+ text-align: center;
+}
+
+.resource-view.image .resource-view-content > .info {
+ padding-bottom: 10px;
+ font-size: 11px;
+ -webkit-user-select: text;
+}
+
+.resource-view.image img {
+ max-width: 100%;
+ max-height: 1000px;
+ background-image: url(Images/checker.png);
+ -webkit-box-shadow: 0px 5px 10px rgba(0, 0, 0, 0.5);
+ -webkit-user-select: text;
+ -webkit-user-drag: auto;
+}
+
+.resource-view.image .title {
+ text-align: center;
+ font-size: 13px;
+}
+
+.resource-view.image .infoList {
+ margin: 0;
+}
+
+.resource-view.image .infoList dt {
+ font-weight: bold;
+ display: inline-block;
+ width: 50%;
+ text-align: right;
+ color: rgb(76, 76, 76);
+}
+
+.resource-view.image .infoList dd {
+ display: inline-block;
+ padding-left: 8px;
+ width: 50%;
+ text-align: left;
+ margin: 0;
+}
+
+.resource-view.image .infoList dd::after {
+ white-space: pre;
+ content: "\A";
+}
+
+#elements-content {
+ display: block;
+ overflow: auto;
+ padding: 0;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 225px;
+ bottom: 0;
+}
+
+#elements-sidebar {
+ position: absolute;
+ top: 0;
+ right: 0;
+ bottom: 0;
+ width: 225px;
+ background-color: rgb(245, 245, 245);
+ border-left: 1px solid rgb(64%, 64%, 64%);
+ cursor: default;
+ overflow: auto;
+}
+
+.crumbs {
+ display: inline-block;
+ font-size: 11px;
+ line-height: 19px;
+ text-shadow: rgba(255, 255, 255, 0.75) 0 1px 0;
+ color: rgb(20, 20, 20);
+ margin-left: -1px;
+ padding-right: 12px;
+}
+
+.crumbs .crumb {
+ height: 24px;
+ border-width: 0 12px 0 2px;
+ -webkit-border-image: url(Images/segment.png) 0 12 0 2;
+ margin-right: -12px;
+ padding-left: 18px;
+ padding-right: 2px;
+ white-space: nowrap;
+ line-height: 23px;
+ float: right;
+}
+
+.crumbs .crumb.collapsed > * {
+ display: none;
+}
+
+.crumbs .crumb.collapsed::before {
+ content: "\2026";
+ font-weight: bold;
+}
+
+.crumbs .crumb.compact .extra {
+ display: none;
+}
+
+.crumbs .crumb.dimmed {
+ color: rgba(0, 0, 0, 0.45);
+}
+
+.crumbs .crumb.start {
+ padding-left: 7px;
+}
+
+.crumbs .crumb.end {
+ border-width: 0 2px 0 2px;
+ padding-right: 6px;
+ -webkit-border-image: url(Images/segmentEnd.png) 0 2 0 2;
+}
+
+.crumbs .crumb.selected {
+ -webkit-border-image: url(Images/segmentSelected.png) 0 12 0 2;
+ color: black;
+ text-shadow: rgba(255, 255, 255, 0.5) 0 1px 0;
+}
+
+.crumbs .crumb.selected:hover {
+ -webkit-border-image: url(Images/segmentSelected.png) 0 12 0 2;
+}
+
+.crumbs .crumb.selected.end, .crumbs .crumb.selected.end:hover {
+ -webkit-border-image: url(Images/segmentSelectedEnd.png) 0 2 0 2;
+}
+
+.crumbs .crumb:hover {
+ -webkit-border-image: url(Images/segmentHover.png) 0 12 0 2;
+ color: black;
+}
+
+.crumbs .crumb.dimmed:hover {
+ -webkit-border-image: url(Images/segmentHover.png) 0 12 0 2;
+ color: rgba(0, 0, 0, 0.75);
+}
+
+.crumbs .crumb.end:hover {
+ -webkit-border-image: url(Images/segmentHoverEnd.png) 0 2 0 2;
+}
+
+.outline-disclosure li.hovered:not(.selected) .selection {
+ display: block;
+ left: 3px;
+ right: 3px;
+ background-color: rgba(56, 121, 217, 0.1);
+ -webkit-border-radius: 5px;
+}
+
+.outline-disclosure li.highlighted .highlight {
+ background-color: rgb(255, 230, 179);
+ -webkit-border-radius: 4px;
+ padding-bottom: 2px;
+ margin-bottom: -2px;
+}
+
+.outline-disclosure li.selected.highlighted .highlight {
+ background-color: transparent;
+ padding-bottom: 0;
+ margin-bottom: 0;
+}
+
+.outline-disclosure li .selection {
+ display: none;
+ position: absolute;
+ left: 0;
+ right: 0;
+ height: 15px;
+ z-index: -1;
+}
+
+.outline-disclosure li.selected .selection {
+ display: block;
+ background-color: rgb(212, 212, 212);
+}
+
+:focus .outline-disclosure li.selected .selection {
+ background-color: rgb(56, 121, 217);
+}
+
+.outline-disclosure > ol {
+ position: relative;
+ padding: 2px 6px !important;
+ margin: 0;
+ color: black;
+ cursor: default;
+ min-width: 100%;
+}
+
+.outline-disclosure, .outline-disclosure ol {
+ list-style-type: none;
+ font-size: 11px;
+ -webkit-padding-start: 12px;
+ margin: 0;
+}
+
+.outline-disclosure li {
+ padding: 0 0 2px 14px;
+ margin-top: 1px;
+ margin-bottom: 1px;
+ word-wrap: break-word;
+ text-indent: -2px
+}
+
+:focus .outline-disclosure li.selected {
+ color: white;
+}
+
+:focus .outline-disclosure li.selected * {
+ color: inherit;
+}
+
+.outline-disclosure li.parent {
+ text-indent: -12px
+}
+
+.outline-disclosure li .webkit-html-tag.close {
+ margin-left: -12px;
+}
+
+.outline-disclosure li.parent::before {
+ content: url(Images/treeRightTriangleBlack.png);
+ float: left;
+ width: 8px;
+ height: 8px;
+ margin-top: 1px;
+ padding-right: 2px;
+}
+
+.outline-disclosure li.parent::before {
+ content: url(Images/treeRightTriangleBlack.png);
+}
+
+:focus .outline-disclosure li.parent.selected::before {
+ content: url(Images/treeRightTriangleWhite.png);
+}
+
+.outline-disclosure li.parent.expanded::before {
+ content: url(Images/treeDownTriangleBlack.png);
+}
+
+:focus .outline-disclosure li.parent.expanded.selected::before {
+ content: url(Images/treeDownTriangleWhite.png);
+}
+
+.outline-disclosure ol.children {
+ display: none;
+}
+
+.outline-disclosure ol.children.expanded {
+ display: block;
+}
+
+.webkit-html-comment {
+ /* Keep this in sync with view-source.css (.webkit-html-comment) */
+ color: rgb(35, 110, 37);
+}
+
+.webkit-html-tag {
+ /* Keep this in sync with view-source.css (.webkit-html-tag) */
+ color: rgb(136, 18, 128);
+}
+
+.webkit-html-doctype {
+ /* Keep this in sync with view-source.css (.webkit-html-doctype) */
+ color: rgb(192, 192, 192);
+}
+
+.webkit-html-attribute-name {
+ /* Keep this in sync with view-source.css (.webkit-html-attribute-name) */
+ color: rgb(153, 69, 0);
+}
+
+.webkit-html-attribute-value {
+ /* Keep this in sync with view-source.css (.webkit-html-attribute-value) */
+ color: rgb(26, 26, 166);
+}
+
+.webkit-html-external-link, .webkit-html-resource-link {
+ /* Keep this in sync with view-source.css (.webkit-html-external-link, .webkit-html-resource-link) */
+ color: #00e;
+}
+
+.webkit-html-external-link {
+ /* Keep this in sync with view-source.css (.webkit-html-external-link) */
+ text-decoration: none;
+}
+
+.webkit-html-external-link:hover {
+ /* Keep this in sync with view-source.css (.webkit-html-external-link:hover) */
+ text-decoration: underline;
+}
+
+.add-attribute {
+ margin-left: 1px;
+ margin-right: 1px;
+}
+
+.placard {
+ position: relative;
+ margin-top: 1px;
+ padding: 3px 8px 4px 18px;
+ min-height: 18px;
+ white-space: nowrap;
+}
+
+.placard:nth-of-type(2n) {
+ background-color: rgb(234, 243, 255);
+}
+
+.placard.selected {
+ border-top: 1px solid rgb(145, 160, 192);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(162, 177, 207)), to(rgb(120, 138, 177)));
+ -webkit-background-origin: padding;
+ -webkit-background-clip: padding;
+}
+
+:focus .placard.selected {
+ border-top: 1px solid rgb(68, 128, 200);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(92, 147, 213)), to(rgb(21, 83, 170)));
+}
+
+body.inactive .placard.selected {
+ border-top: 1px solid rgb(151, 151, 151);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(180, 180, 180)), to(rgb(138, 138, 138)));
+}
+
+.placard .title {
+ color: black;
+ font-weight: normal;
+ word-wrap: break-word;
+ white-space: normal;
+}
+
+.placard.selected .title {
+ color: white;
+ font-weight: bold;
+}
+
+.placard .subtitle {
+ float: right;
+ font-size: 10px;
+ margin-left: 5px;
+ max-width: 55%;
+ color: rgba(0, 0, 0, 0.7);
+ text-overflow: ellipsis;
+ overflow: hidden;
+}
+
+.placard.selected .subtitle {
+ color: rgba(255, 255, 255, 0.7);
+}
+
+.placard .subtitle a {
+ color: inherit;
+}
+
+.section {
+ position: relative;
+ margin-top: 1px;
+}
+
+.section:nth-last-of-type(1) {
+ margin-bottom: 1px;
+}
+
+.watch-expressions-buttons-container {
+ text-align: center;
+}
+
+.section .header {
+ padding: 2px 8px 4px 18px;
+ border-top: 1px solid rgb(145, 160, 192);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(162, 177, 207)), to(rgb(120, 138, 177)));
+ min-height: 18px;
+ white-space: nowrap;
+ -webkit-background-origin: padding;
+ -webkit-background-clip: padding;
+}
+
+.section.no-affect .header {
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(167, 167, 167)), to(rgb(123, 123, 123)))
+}
+
+.section .header::before {
+ position: absolute;
+ top: 4px;
+ left: 7px;
+ width: 8px;
+ height: 8px;
+ content: url(Images/treeRightTriangleWhite.png);
+}
+
+.section.blank-section .header::before {
+ display: none;
+}
+
+.section.expanded .header::before {
+ content: url(Images/treeDownTriangleWhite.png);
+}
+
+.section .header .title {
+ color: white;
+ font-weight: bold;
+ word-wrap: break-word;
+ white-space: normal;
+}
+
+.section .header .title.blank-title {
+ font-style: italic;
+}
+
+.section .header label {
+ display: none;
+}
+
+.section.expanded .header label {
+ display: inline;
+}
+
+.section .header input[type=checkbox] {
+ height: 10px;
+ width: 10px;
+ margin-left: 0;
+ margin-top: 0;
+ margin-bottom: 0;
+ vertical-align: 2px;
+}
+
+.section .header .subtitle {
+ float: right;
+ font-size: 10px;
+ margin-left: 5px;
+ max-width: 55%;
+ color: rgba(255, 255, 255, 0.7);
+ text-overflow: ellipsis;
+ overflow: hidden;
+}
+
+.section .header .subtitle a {
+ color: inherit;
+}
+
+.section .properties {
+ display: none;
+ margin: 0;
+ padding: 2px 6px 3px;
+ list-style: none;
+ background-color: white;
+ min-height: 18px;
+}
+
+.section.no-affect .properties li {
+ opacity: 0.5;
+}
+
+.section.no-affect .properties li.editing {
+ opacity: 1.0;
+}
+
+.section.expanded .properties {
+ display: block;
+}
+
+.section .properties li {
+ margin-left: 12px;
+ white-space: nowrap;
+ text-overflow: ellipsis;
+ overflow: hidden;
+ -webkit-user-select: text;
+ cursor: auto;
+}
+
+.section .properties li.parent {
+ margin-left: 1px;
+}
+
+.section .properties ol {
+ display: none;
+ margin: 0;
+ -webkit-padding-start: 12px;
+ list-style: none;
+}
+
+.section .properties ol.expanded {
+ display: block;
+}
+
+.section .properties li.parent::before {
+ content: url(Images/treeRightTriangleBlack.png);
+ opacity: 0.75;
+ float: left;
+ width: 8px;
+ height: 8px;
+ margin-top: 0;
+ padding-right: 3px;
+ -webkit-user-select: none;
+ cursor: default;
+}
+
+.section .properties li.parent.expanded::before {
+ content: url(Images/treeDownTriangleBlack.png);
+ margin-top: 1px;
+}
+
+.section .properties li .info {
+ padding-top: 4px;
+ padding-bottom: 3px;
+}
+
+.editing {
+ -webkit-user-select: text;
+ -webkit-box-shadow: rgba(0, 0, 0, .5) 3px 3px 4px;
+ outline: 1px solid rgb(66%, 66%, 66%) !important;
+ background-color: white;
+ -webkit-user-modify: read-write-plaintext-only;
+ text-overflow: clip;
+ padding-left: 2px;
+ margin-left: -2px;
+ padding-right: 2px;
+ margin-right: -2px;
+ margin-bottom: -1px;
+ padding-bottom: 1px;
+ opacity: 1.0 !important;
+}
+
+.editing, .editing * {
+ color: black !important;
+ text-decoration: none !important;
+}
+
+.section .properties li.editing {
+ margin-left: 10px;
+ text-overflow: clip;
+}
+
+li.editing .swatch, li.editing .enabled-button, li.editing-sub-part .delete-button {
+ display: none !important;
+}
+
+.watch-expressions > li.editing-sub-part .name {
+ display: block;
+ width: 100%;
+}
+
+.watch-expressions > li.editing-sub-part .value, .watch-expressions > li.editing-sub-part .separator {
+ display: none;
+}
+
+.section .properties li.editing-sub-part {
+ padding: 3px 6px 8px 18px;
+ margin: -3px -6px -8px -6px;
+ text-overflow: clip;
+}
+
+.section .properties .overloaded, .section .properties .disabled {
+ text-decoration: line-through;
+}
+
+.section.computed-style .properties .disabled {
+ text-decoration: none;
+ opacity: 0.5;
+}
+
+.section .properties .implicit, .section .properties .inherited {
+ opacity: 0.5;
+}
+
+.section:not(.show-inherited) .properties .inherited {
+ display: none;
+}
+
+.section .properties .enabled-button {
+ display: none;
+ float: right;
+ font-size: 10px;
+ margin: 0 0 0 4px;
+ vertical-align: top;
+ position: relative;
+ z-index: 1;
+}
+
+/* FIXME: need a better icon (comment in bug 27514) */
+.section .properties .delete-button {
+ width: 10px;
+ height: 10px;
+ background-image: url(Images/errorIcon.png);
+ background-position: 0 0;
+ background-color: transparent;
+ background-repeat: no-repeat;
+ border: 0 none transparent;
+}
+
+.section:hover .properties .enabled-button {
+ display: block;
+}
+
+.section .properties .name {
+ color: rgb(136, 19, 145);
+}
+
+.section .properties .value.dimmed {
+ color: rgb(100, 100, 100);
+}
+
+.section .properties .number {
+ color: blue;
+}
+
+.section .properties .priority {
+ color: rgb(128, 0, 0);
+}
+
+.section .properties .keyword {
+ color: rgb(136, 19, 79);
+}
+
+.section .properties .color {
+ color: rgb(118, 15, 21);
+}
+
+.swatch {
+ display: inline-block;
+ vertical-align: baseline;
+ margin-left: 1px;
+ margin-right: 2px;
+ margin-bottom: -1px;
+ width: 1em;
+ height: 1em;
+ border: 1px solid rgba(128, 128, 128, 0.6);
+}
+
+.swatch:hover {
+ border: 1px solid rgba(64, 64, 64, 0.8);
+}
+
+.pane:not(.expanded) + .pane, .pane:first-of-type {
+ margin-top: -1px;
+}
+
+.pane > .title {
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(243, 243, 243)), color-stop(0.05, rgb(243, 243, 243)), color-stop(0.05, rgb(230, 230, 230)), to(rgb(209, 209, 209)));
+ height: 20px;
+ padding: 0 5px;
+ border-top: 1px solid rgb(189, 189, 189);
+ border-bottom: 1px solid rgb(189, 189, 189);
+ font-weight: bold;
+ font-size: 12px;
+ line-height: 18px;
+ color: rgb(110, 110, 110);
+ text-shadow: white 0 1px 0;
+ -webkit-background-origin: padding;
+ -webkit-background-clip: padding;
+}
+
+.pane > .title:active {
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(231, 231, 231)), color-stop(0.05, rgb(231, 231, 231)), color-stop(0.05, rgb(207, 207, 207)), to(rgb(186, 186, 186)));
+ border-top: 1px solid rgb(178, 178, 178);
+ border-bottom: 1px solid rgb(178, 178, 178);
+}
+
+.pane > .title::before {
+ content: url(Images/disclosureTriangleSmallRightBlack.png);
+ float: left;
+ width: 11px;
+ height: 12px;
+ margin-right: 2px;
+ margin-top: 1px;
+}
+
+.pane.expanded > .title::before {
+ content: url(Images/disclosureTriangleSmallDownBlack.png);
+}
+
+.pane > .title > select {
+ display: none;
+ float: right;
+ width: 23px;
+ height: 17px;
+ color: transparent;
+ background-color: transparent;
+ border: none;
+ background-image: url(Images/paneSettingsButtons.png);
+ background-repeat: no-repeat;
+ margin: 1px 0 0 0;
+ padding: 0;
+ -webkit-border-radius: 0;
+ -webkit-appearance: none;
+}
+
+.pane.expanded:hover > .title > select {
+ display: inline-block;
+}
+
+.pane > .title > select:hover {
+ background-position: -23px 0px;
+}
+
+.pane > .title > select:active {
+ background-position: -46px 0px;
+}
+
+.pane > .body {
+ position: relative;
+ display: none;
+ background-color: white;
+ overflow-y: auto;
+ overflow-x: hidden;
+}
+
+.pane > .body .info {
+ text-align: center;
+ font-style: italic;
+ font-size: 10px;
+ padding: 6px;
+ color: gray;
+}
+
+.pane.expanded > .body, .pane.expanded > .growbar {
+ display: block;
+}
+
+.pane.expanded:nth-last-of-type(1) {
+ border-bottom: 1px solid rgb(189, 189, 189);
+}
+
+.pane > .growbar {
+ display: none;
+ background-image: url(Images/paneGrowHandleLine.png), url(Images/paneBottomGrow.png);
+ background-repeat: no-repeat, repeat-x;
+ background-position: center center, bottom;
+ height: 5px;
+}
+
+.metrics {
+ padding: 8px;
+ font-size: 10px;
+ text-align: center;
+ white-space: nowrap;
+}
+
+.metrics .label {
+ position: absolute;
+ margin-top: -10px;
+ font-size: 9px;
+ color: grey;
+ background-color: white;
+ margin-left: 3px;
+ padding-left: 2px;
+ padding-right: 2px;
+}
+
+.metrics .position {
+ border: 1px rgb(66%, 66%, 66%) dotted;
+ display: inline-block;
+ text-align: center;
+ padding: 3px;
+ margin: 3px;
+}
+
+.metrics .margin {
+ border: 1px dashed;
+ display: inline-block;
+ text-align: center;
+ vertical-align: middle;
+ padding: 3px;
+ margin: 3px;
+}
+
+.metrics .border {
+ border: 1px black solid;
+ display: inline-block;
+ text-align: center;
+ vertical-align: middle;
+ padding: 3px;
+ margin: 3px;
+}
+
+.metrics .padding {
+ border: 1px grey dashed;
+ display: inline-block;
+ text-align: center;
+ vertical-align: middle;
+ padding: 3px;
+ margin: 3px;
+}
+
+.metrics .content {
+ position: static;
+ border: 1px grey solid;
+ display: inline-block;
+ text-align: center;
+ vertical-align: middle;
+ padding: 3px;
+ margin: 3px;
+ min-width: 80px;
+ text-align: center;
+ overflow: visible;
+}
+
+.metrics .content span {
+ display: inline-block;
+}
+
+.metrics .editing {
+ position: relative;
+ z-index: 100;
+}
+
+.metrics .left {
+ display: inline-block;
+ vertical-align: middle;
+}
+
+.metrics .right {
+ display: inline-block;
+ vertical-align: middle;
+}
+
+.metrics .top {
+ display: inline-block;
+}
+
+.metrics .bottom {
+ display: inline-block;
+}
+
+.sidebar {
+ position: absolute;
+ top: 0;
+ left: 0;
+ bottom: 0;
+ width: 200px;
+ overflow-y: auto;
+ overflow-x: hidden;
+ background-color: rgb(214, 221, 229);
+ border-right: 1px solid rgb(64%, 64%, 64%);
+}
+
+body.inactive .sidebar {
+ background-color: rgb(232, 232, 232);
+}
+
+.database-sidebar-tree-item .icon {
+ content: url(Images/database.png);
+}
+
+.database-table-sidebar-tree-item .icon {
+ content: url(Images/databaseTable.png);
+}
+
+.domstorage-sidebar-tree-item.local-storage .icon {
+ content: url(Images/localStorage.png);
+}
+
+.domstorage-sidebar-tree-item.session-storage .icon {
+ content: url(Images/sessionStorage.png);
+}
+
+.cookie-sidebar-tree-item .icon {
+ content: url(Images/cookie.png);
+}
+
+#storage-views {
+ position: absolute;
+ top: 0;
+ right: 0;
+ left: 200px;
+ bottom: 0;
+}
+
+.storage-view {
+ display: none;
+ overflow: hidden;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+}
+
+.storage-view.visible {
+ display: block;
+}
+
+.storage-view.table {
+ overflow: hidden;
+}
+
+.storage-view.table .data-grid {
+ border: none;
+ height: 100%;
+}
+
+.storage-view.table .storage-table-empty, .storage-view.table .storage-table-error {
+ position: absolute;
+ top: 0;
+ bottom: 25%;
+ left: 0;
+ right: 0;
+ font-size: 24px;
+ color: rgb(75%, 75%, 75%);
+ margin-top: auto;
+ margin-bottom: auto;
+ height: 50px;
+ line-height: 26px;
+ text-align: center;
+ font-weight: bold;
+ padding: 10px;
+ white-space: pre-wrap;
+}
+
+.storage-view.table .storage-table-error {
+ color: rgb(66%, 33%, 33%);
+}
+
+.data-grid {
+ position: relative;
+ border: 1px solid #aaa;
+}
+
+.data-grid .highlight {
+ background-color: rgb(255, 230, 179);
+}
+
+.data-grid tr.selected .highlight {
+ background-color: transparent;
+}
+
+.data-grid table {
+ table-layout: fixed;
+ border-spacing: 0;
+ border-collapse: collapse;
+ width: 100%;
+ font-size: 10px;
+ font-family: Lucida Grande, sans-serif;
+}
+
+.data-grid .data-container {
+ position: absolute;
+ top: 16px;
+ bottom: 0;
+ left: 0;
+ right: 0;
+ padding-right: 14px;
+ overflow-x: hidden;
+ overflow-y: overlay;
+ background-image: -webkit-gradient(linear, left top, left bottom, from(white), color-stop(0.5, white), color-stop(0.5, rgb(234, 243, 255)), to(rgb(234, 243, 255)));
+ -webkit-background-size: 1px 32px;
+}
+
+.data-grid.inline .data-container {
+ position: static;
+}
+
+.data-grid th {
+ text-align: left;
+ background-image: url(Images/glossyHeader.png);
+ background-repeat: repeat-x;
+ border-right: 1px solid rgb(179, 179, 179);
+ border-bottom: 1px solid rgb(179, 179, 179);
+ height: 15px;
+ font-weight: normal;
+ vertical-align: middle;
+ padding: 0 4px;
+ white-space: nowrap;
+}
+
+.data-grid th.corner {
+ width: 15px;
+ border-right: 0 none transparent;
+}
+
+.data-grid tr.filler {
+ display: table-row !important;
+ height: auto !important;
+}
+
+.data-grid tr.filler td {
+ height: auto !important;
+ padding: 0 !important;
+}
+
+.data-grid table.data {
+ position: absolute;
+ left: 0;
+ top: 0;
+ right: 16px;
+ bottom: 0;
+ height: 100%;
+ border-top: 0 none transparent;
+ background-image: -webkit-gradient(linear, left top, left bottom, from(white), color-stop(0.5, white), color-stop(0.5, rgb(234, 243, 255)), to(rgb(234, 243, 255)));
+ -webkit-background-size: 1px 32px;
+}
+
+.data-grid.inline table.data {
+ position: static;
+}
+
+.data-grid table.data tr {
+ display: none;
+}
+
+.data-grid table.data tr.revealed {
+ display: table-row;
+}
+
+.data-grid td {
+ vertical-align: top;
+ height: 12px;
+ padding: 2px 4px;
+ white-space: nowrap;
+ border-right: 1px solid #aaa;
+ -webkit-user-select: text;
+}
+
+.data-grid td > div, .data-grid th > div {
+ white-space: nowrap;
+ text-overflow: ellipsis;
+ overflow: hidden;
+}
+
+.data-grid .centered div {
+ text-align: center;
+}
+
+.data-grid .right div {
+ text-align: right;
+}
+
+.data-grid th.sortable div {
+ position: relative;
+}
+
+.data-grid th.sortable:active {
+ background-image: url(Images/glossyHeaderPressed.png);
+}
+
+.data-grid th.sort-ascending, .data-grid th.sort-descending {
+ border-right: 1px solid rgb(107, 140, 196);
+ border-bottom: 1px solid rgb(107, 140, 196);
+ background-image: url(Images/glossyHeaderSelected.png);
+ background-repeat: repeat-x;
+}
+
+.data-grid th.sortable.sort-ascending:active, .data-grid th.sortable.sort-descending:active {
+ background-image: url(Images/glossyHeaderSelectedPressed.png);
+}
+
+.data-grid th.sort-ascending div::after {
+ position: absolute;
+ top: 0;
+ right: 0;
+ width: 8px;
+ height: 8px;
+ content: url(Images/treeUpTriangleBlack.png);
+}
+
+.data-grid th.sort-descending div::after {
+ position: absolute;
+ top: 0;
+ right: 0;
+ margin-top: 1px;
+ width: 8px;
+ height: 8px;
+ content: url(Images/treeDownTriangleBlack.png);
+}
+
+body.inactive .data-grid th.sort-ascending, body.inactive .data-grid th.sort-descending {
+ background-image: url(Images/glossyHeader.png);
+ border-right: 1px solid rgb(179, 179, 179);
+ border-bottom: 1px solid rgb(179, 179, 179);
+}
+
+.data-grid tr.parent td.disclosure::before {
+ float: left;
+ content: url(Images/treeRightTriangleBlack.png);
+ width: 8px;
+ height: 8px;
+ margin-right: 2px;
+ -webkit-user-select: none;
+}
+
+.data-grid tr.expanded td.disclosure::before {
+ content: url(Images/treeDownTriangleBlack.png);
+ width: 8px;
+ height: 8px;
+ margin-top: 1px;
+}
+
+.data-grid tr.selected {
+ background-color: rgb(212, 212, 212);
+ color: inherit;
+}
+
+.data-grid:focus tr.selected {
+ background-color: rgb(56, 121, 217);
+ color: white;
+}
+
+.data-grid:focus tr.parent.selected td.disclosure::before {
+ content: url(Images/treeRightTriangleWhite.png);
+}
+
+.data-grid:focus tr.expanded.selected td.disclosure::before {
+ content: url(Images/treeDownTriangleWhite.png);
+}
+
+.data-grid tr:not(.parent) td.disclosure {
+ text-indent: 10px;
+}
+
+.data-grid-resizer {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ width: 5px;
+ z-index: 500;
+ cursor: col-resize;
+}
+
+.storage-view.query {
+ font-size: initial;
+ font-family: monospace;
+ padding: 2px 0;
+ overflow-y: overlay;
+ overflow-x: hidden;
+ -webkit-text-size-adjust: auto;
+}
+
+.database-query-prompt {
+ position: relative;
+ padding: 1px 22px 1px 24px;
+ min-height: 16px;
+ white-space: pre-wrap;
+ -webkit-user-modify: read-write-plaintext-only;
+ -webkit-user-select: text;
+}
+
+.database-user-query::before, .database-query-prompt::before, .database-query-result::before {
+ position: absolute;
+ display: block;
+ content: "";
+ left: 7px;
+ top: 0.8em;
+ width: 10px;
+ height: 10px;
+ margin-top: -5px;
+ -webkit-user-select: none;
+}
+
+.database-query-prompt::before {
+ background-image: url(Images/userInputIcon.png);
+}
+
+.database-user-query {
+ position: relative;
+ border-bottom: 1px solid rgb(245, 245, 245);
+ padding: 1px 22px 1px 24px;
+ min-height: 16px;
+}
+
+.database-user-query::before {
+ background-image: url(Images/userInputPreviousIcon.png);
+}
+
+.database-query-text {
+ color: rgb(0, 128, 255);
+ -webkit-user-select: text;
+}
+
+.database-query-result {
+ position: relative;
+ padding: 1px 22px 1px 24px;
+ min-height: 16px;
+ margin-left: -24px;
+ padding-right: 0;
+}
+
+.database-query-result.error {
+ color: red;
+ -webkit-user-select: text;
+}
+
+.database-query-result.error::before {
+ background-image: url(Images/errorIcon.png);
+}
+
+.panel-enabler-view {
+ z-index: 1000;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ background-color: white;
+ font-size: 13px;
+ text-align: center;
+ overflow-x: hidden;
+ overflow-y: overlay;
+ display: none;
+}
+
+.panel-enabler-view.visible {
+ display: block;
+}
+
+.panel-enabler-view .panel-enabler-view-content {
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ max-height: 390px;
+ margin: auto;
+ white-space: nowrap;
+}
+
+.panel-enabler-view h1 {
+ color: rgb(110, 116, 128);
+ font-size: 16px;
+ line-height: 20px;
+ font-weight: normal;
+ margin-top: 0;
+}
+
+.panel-enabler-disclaimer {
+ font-size: 10px;
+ color: rgb(110, 116, 128);
+ margin-bottom: 12px;
+ margin-left: 20px;
+}
+
+.panel-enabler-disclaimer:empty {
+ display: none;
+}
+
+.panel-enabler-view img {
+ height: 100%;
+ min-height: 200px;
+ max-width: 100%;
+ top: 0;
+ bottom: 0;
+ padding: 20px 0 20px 20px;
+ margin: auto;
+ vertical-align: middle;
+}
+
+.panel-enabler-view img.hidden {
+ display: initial !important;
+ width: 0;
+}
+
+.panel-enabler-view form {
+ display: inline-block;
+ vertical-align: middle;
+ width: 330px;
+ margin: 0;
+ padding: 15px;
+ white-space: normal;
+}
+
+.panel-enabler-view label {
+ position: relative;
+ display: block;
+ text-align: left;
+ word-break: break-word;
+ margin: 0 0 5px 20px;
+}
+
+.panel-enabler-view button {
+ font-size: 13px;
+ margin: 6px 0 0 0;
+ padding: 3px 20px;
+ color: rgb(6, 6, 6);
+ height: 24px;
+ background-color: transparent;
+ border: 1px solid rgb(165, 165, 165);
+ background-color: rgb(237, 237, 237);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(252, 252, 252)), to(rgb(223, 223, 223)));
+ -webkit-border-radius: 12px;
+ -webkit-appearance: none;
+}
+
+.panel-enabler-view button:active {
+ background-color: rgb(215, 215, 215);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(194, 194, 194)), to(rgb(239, 239, 239)));
+}
+
+body.inactive .panel-enabler-view button, .panel-enabler-view button:disabled {
+ color: rgb(130, 130, 130);
+ border-color: rgb(212, 212, 212);
+ background-color: rgb(239, 239, 239);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(250, 250, 250)), to(rgb(235, 235, 235)));
+}
+
+.panel-enabler-view input {
+ height: 17px;
+ width: 17px;
+ border: 1px solid rgb(165, 165, 165);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(252, 252, 252)), to(rgb(223, 223, 223)));
+ -webkit-border-radius: 8px;
+ -webkit-appearance: none;
+ vertical-align: middle;
+ margin: 0 5px 5px 0;
+}
+
+.panel-enabler-view input:active {
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(194, 194, 194)), to(rgb(239, 239, 239)));
+}
+
+.panel-enabler-view input:checked {
+ background: url(Images/radioDot.png) center no-repeat,
+ -webkit-gradient(linear, left top, left bottom, from(rgb(252, 252, 252)), to(rgb(223, 223, 223)));
+}
+
+.panel-enabler-view.resources img {
+ content: url(Images/resourcesSilhouette.png);
+}
+
+.panel-enabler-view.scripts img {
+ content: url(Images/scriptsSilhouette.png);
+}
+
+.panel-enabler-view.profiles img {
+ content: url(Images/profilesSilhouette.png);
+}
+
+button.enable-toggle-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/enableOutlineButtonGlyph.png);
+}
+
+button.enable-toggle-status-bar-item.toggled-on .glyph {
+ -webkit-mask-image: url(Images/enableSolidButtonGlyph.png);
+}
+
+.scripts-pause-on-exceptions-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/pauseOnExceptionButtonGlyph.png);
+}
+
+#scripts-status-bar {
+ position: absolute;
+ top: -1px;
+ left: 0;
+ right: 0;
+ height: 24px;
+}
+
+#scripts-files {
+ max-width: 250px;
+}
+
+#scripts-functions {
+ max-width: 150px;
+}
+
+#scripts-status-bar .status-bar-item img {
+ margin-top: 2px;
+}
+
+#scripts-back img {
+ content: url(Images/back.png);
+}
+
+#scripts-forward img {
+ content: url(Images/forward.png);
+}
+
+#scripts-pause img {
+ content: url(Images/debuggerPause.png);
+}
+
+#scripts-pause.paused img {
+ content: url(Images/debuggerContinue.png);
+}
+
+#scripts-step-over img {
+ content: url(Images/debuggerStepOver.png);
+}
+
+#scripts-step-into img {
+ content: url(Images/debuggerStepInto.png);
+}
+
+#scripts-step-out img {
+ content: url(Images/debuggerStepOut.png);
+}
+
+#scripts-debugger-status {
+ position: absolute;
+ line-height: 24px;
+ top: 0;
+ right: 8px;
+}
+
+#scripts-sidebar-resizer-widget {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ right: 225px;
+ width: 16px;
+ cursor: col-resize;
+ background-image: url(Images/statusbarResizerHorizontal.png);
+ background-repeat: no-repeat;
+ background-position: center;
+}
+
+#scripts-sidebar-buttons {
+ position: absolute;
+ right: 0;
+ top: 0;
+ bottom: 0;
+ width: 225px;
+ overflow: hidden;
+ border-left: 1px solid rgb(64%, 64%, 64%);
+}
+
+#script-resource-views {
+ display: block;
+ overflow: auto;
+ padding: 0;
+ position: absolute;
+ top: 23px;
+ left: 0;
+ right: 225px;
+ bottom: 0;
+}
+
+.script-view {
+ display: none;
+ overflow: hidden;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+}
+
+.script-view.visible {
+ display: block;
+}
+
+#scripts-sidebar {
+ position: absolute;
+ top: 23px;
+ right: 0;
+ bottom: 0;
+ width: 225px;
+ background-color: rgb(245, 245, 245);
+ border-left: 1px solid rgb(64%, 64%, 64%);
+ cursor: default;
+ overflow: auto;
+}
+
+.resources-larger-resources-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/largerResourcesButtonGlyph.png);
+}
+
+#resources-filter {
+ height: 24px;
+ padding: 2px 10px 0;
+ background: -webkit-gradient(linear, left top, left bottom, from(rgb(233, 233, 233)), to(rgb(207, 207, 207)));
+ border-bottom: 1px solid rgb(177, 177, 177);
+ overflow: hidden;
+}
+
+#console-filter {
+ height: 24px;
+ padding: 2px 10px 0;
+ overflow: hidden;
+}
+
+#resources-filter li, #console-filter li {
+ display: inline-block;
+ margin: 1px 1px 0 0;
+ padding: 0 6px 3px;
+ font-size: 12px;
+ line-height: 12px;
+ font-weight: bold;
+ color: rgb(40, 40, 40);
+ border: 1px solid transparent;
+ border-bottom: 0;
+ background: transparent;
+ -webkit-border-radius: 8px;
+ text-shadow: rgba(255, 255, 255, 0.5) 1px 1px 0;
+}
+
+#resources-filter li.selected, #resources-filter li:hover, #resources-filter li:active,
+#console-filter li.selected, #console-filter li:hover, #console-filter li:active {
+ color: white;
+ text-shadow: rgb(80, 80, 80) 1px 1px 1px;
+ background: rgba(20, 20, 20, 0.4);
+ border-color: rgba(20, 20, 20, 0.2);
+ -webkit-box-shadow: 0 1px 0px rgba(255, 255, 255, 0.5);
+}
+
+#resources-filter li:hover,
+#console-filter li:hover {
+ background: rgba(20, 20, 20, 0.4);
+ border-color: transparent;
+ -webkit-box-shadow: none;
+}
+
+#resources-filter li:active,
+#console-filter li:active {
+ background: rgba(20, 20, 20, 0.6);
+}
+
+#resources-container {
+ position: absolute;
+ top: 24px;
+ left: 0;
+ bottom: 0;
+ right: 0;
+ border-right: 0 none transparent;
+ overflow-y: auto;
+ overflow-x: hidden;
+}
+
+#resources-container.viewing-resource {
+ right: auto;
+ width: 200px;
+ border-right: 1px solid rgb(64%, 64%, 64%);
+}
+
+#resources-container.viewing-resource #resources-sidebar {
+ width: 100%;
+ border-right: 0 none transparent;
+}
+
+#resources-sidebar {
+ min-height: 100%;
+ bottom: auto;
+ overflow: visible;
+}
+
+#resources-container-content {
+ position: absolute;
+ top: 0;
+ right: 0;
+ left: 200px;
+ min-height: 100%;
+}
+
+#resources-container.viewing-resource #resources-container-content {
+ display: none;
+}
+
+#resources-summary {
+ position: absolute;
+ padding-top: 20px;
+ top: 0;
+ left: 0;
+ right: 0;
+ height: 93px;
+ margin-left: -1px;
+ border-left: 1px solid rgb(102, 102, 102);
+ background-color: rgb(101, 111, 130);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgba(0, 0, 0, 0)), to(rgba(0, 0, 0, 0.5)));
+ background-repeat: repeat-x;
+ background-position: bottom;
+ text-align: center;
+ text-shadow: black 0 1px 1px;
+ white-space: nowrap;
+ color: white;
+ -webkit-background-size: 1px 6px;
+ -webkit-background-origin: padding;
+ -webkit-background-clip: padding;
+}
+
+.summary-graph-legend {
+ margin-top: -10px;
+ padding-left: 15px;
+}
+
+.summary-graph-legend-item {
+ display: inline-block;
+ font-weight: bold;
+ margin-right: 15px;
+ vertical-align: top;
+}
+
+.summary-graph-legend-item.total {
+ margin-left: 10px;
+}
+
+.summary-graph-legend-label {
+ display: inline-block;
+ text-align: left;
+}
+
+.summary-graph-legend-header {
+ font-size: 12px;
+}
+
+.summary-graph-legend-value {
+ font-size: 10px;
+}
+
+.summary-graph-legend-swatch {
+ vertical-align: top;
+ margin-top: 1px;
+ margin-right: 3px;
+}
+
+#resources-dividers {
+ position: absolute;
+ left: 0;
+ right: 0;
+ height: 100%;
+ top: 0;
+ z-index: -100;
+}
+
+#resources-dividers-label-bar {
+ position: absolute;
+ top: 93px;
+ left: 0px;
+ right: 0;
+ background-color: rgba(255, 255, 255, 0.8);
+ background-clip: padding;
+ border-bottom: 1px solid rgba(0, 0, 0, 0.3);
+ height: 20px;
+ z-index: 200;
+}
+
+.resources-divider {
+ position: absolute;
+ width: 1px;
+ top: 0;
+ bottom: 0;
+ background-color: rgba(0, 0, 0, 0.1);
+}
+
+.resources-divider.last {
+ background-color: transparent;
+}
+
+.resources-divider-label {
+ position: absolute;
+ top: 4px;
+ right: 3px;
+ font-size: 9px;
+ color: rgb(50%, 50%, 50%);
+ white-space: nowrap;
+}
+
+.resources-graph-label {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ margin: auto -7px;
+ height: 13px;
+ line-height: 13px;
+ font-size: 9px;
+ color: rgba(0, 0, 0, 0.75);
+ text-shadow: rgba(255, 255, 255, 0.25) 1px 0 0, rgba(255, 255, 255, 0.25) -1px 0 0, rgba(255, 255, 255, 0.333) 0 1px 0, rgba(255, 255, 255, 0.25) 0 -1px 0;
+ z-index: 150;
+ overflow: hidden;
+ text-align: center;
+ font-weight: bold;
+ opacity: 0;
+ -webkit-transition: opacity 250ms ease-in-out;
+}
+
+.resources-graph-side:hover .resources-graph-label {
+ opacity: 1;
+}
+
+.resources-graph-label:empty {
+ display: none;
+}
+
+.resources-graph-label.waiting {
+ margin-right: 5px;
+}
+
+.resources-graph-label.before {
+ color: rgba(0, 0, 0, 0.7);
+ text-shadow: none;
+ text-align: right;
+ margin-right: 2px;
+}
+
+.resources-graph-label.before::after {
+ padding-left: 2px;
+ height: 6px;
+ content: url(Images/graphLabelCalloutLeft.png);
+}
+
+.resources-graph-label.after {
+ color: rgba(0, 0, 0, 0.7);
+ text-shadow: none;
+ text-align: left;
+ margin-left: 2px;
+}
+
+.resources-graph-label.after::before {
+ padding-right: 2px;
+ height: 6px;
+ content: url(Images/graphLabelCalloutRight.png);
+}
+
+.resources-graph-bar {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ margin: auto -7px;
+ border-width: 6px 7px;
+ height: 13px;
+ min-width: 14px;
+ opacity: 0.65;
+ -webkit-border-image: url(Images/timelinePillGray.png) 6 7 6 7;
+}
+
+.resources-category-documents, .resources-category-stylesheets, .resources-category-images,
+.resources-category-scripts, .resources-category-xhr, .resources-category-fonts, .resources-category-other {
+ display: none;
+}
+
+.filter-all .resources-category-documents, .filter-documents .resources-category-documents,
+.filter-all .resources-category-stylesheets, .filter-stylesheets .resources-category-stylesheets,
+.filter-all .resources-category-images, .filter-images .resources-category-images,
+.filter-all .resources-category-scripts, .filter-scripts .resources-category-scripts,
+.filter-all .resources-category-xhr, .filter-xhr .resources-category-xhr,
+.filter-all .resources-category-fonts, .filter-fonts .resources-category-fonts,
+.filter-all .resources-category-other, .filter-other .resources-category-other,
+.resource-sidebar-tree-item.selected {
+ display: list-item;
+}
+
+.console-warning-level, .console-error-level, .console-log-level {
+ display: none;
+}
+
+.filter-all .console-warning-level, .filter-warnings .console-warning-level,
+.filter-all .console-error-level, .filter-errors .console-error-level,
+.filter-all .console-log-level, .filter-logs .console-log-level {
+ display: block;
+}
+
+.resources-graph-bar.waiting {
+ opacity: 0.35;
+}
+
+.resource-cached .resources-graph-bar {
+ -webkit-border-image: url(Images/timelineHollowPillGray.png) 6 7 6 7;
+}
+
+.resources-category-documents .resources-graph-bar {
+ -webkit-border-image: url(Images/timelinePillBlue.png) 6 7 6 7;
+}
+
+.resources-category-documents.resource-cached .resources-graph-bar {
+ -webkit-border-image: url(Images/timelineHollowPillBlue.png) 6 7 6 7;
+}
+
+.resources-category-stylesheets .resources-graph-bar {
+ -webkit-border-image: url(Images/timelinePillGreen.png) 6 7 6 7;
+}
+
+.resources-category-stylesheets.resource-cached .resources-graph-bar {
+ -webkit-border-image: url(Images/timelineHollowPillGreen.png) 6 7 6 7;
+}
+
+.resources-category-images .resources-graph-bar {
+ -webkit-border-image: url(Images/timelinePillPurple.png) 6 7 6 7;
+}
+
+.resources-category-images.resource-cached .resources-graph-bar {
+ -webkit-border-image: url(Images/timelineHollowPillPurple.png) 6 7 6 7;
+}
+
+.resources-category-fonts .resources-graph-bar {
+ -webkit-border-image: url(Images/timelinePillRed.png) 6 7 6 7;
+}
+
+.resources-category-fonts.resource-cached .resources-graph-bar {
+ -webkit-border-image: url(Images/timelineHollowPillRed.png) 6 7 6 7;
+}
+
+.resources-category-scripts .resources-graph-bar {
+ -webkit-border-image: url(Images/timelinePillOrange.png) 6 7 6 7;
+}
+
+.resources-category-scripts.resource-cached .resources-graph-bar {
+ -webkit-border-image: url(Images/timelineHollowPillOrange.png) 6 7 6 7;
+}
+
+.resources-category-xhr .resources-graph-bar {
+ -webkit-border-image: url(Images/timelinePillYellow.png) 6 7 6 7;
+}
+
+.resources-category-xhr.resource-cached .resources-graph-bar {
+ -webkit-border-image: url(Images/timelineHollowPillYellow.png) 6 7 6 7;
+}
+
+.tip-button {
+ background-image: url(Images/tipIcon.png);
+ border: none;
+ width: 16px;
+ height: 16px;
+ float: right;
+ background-color: transparent;
+ margin-top: 1px;
+}
+
+.tip-button:active {
+ background-image: url(Images/tipIconPressed.png);
+}
+
+.tip-balloon {
+ position: absolute;
+ left: 145px;
+ top: -5px;
+ z-index: 1000;
+ border-width: 51px 15px 18px 37px;
+ -webkit-border-image: url(Images/tipBalloon.png) 51 15 18 37;
+ width: 265px;
+}
+
+.tip-balloon.bottom {
+ position: absolute;
+ left: 145px;
+ top: auto;
+ bottom: -7px;
+ z-index: 1000;
+ border-width: 18px 15px 51px 37px;
+ -webkit-border-image: url(Images/tipBalloonBottom.png) 18 15 51 37;
+}
+
+.tip-balloon-content {
+ margin-top: -40px;
+ margin-bottom: -2px;
+ margin-left: 2px;
+}
+
+.tip-balloon.bottom .tip-balloon-content {
+ margin-top: -10px;
+ margin-bottom: -35px;
+}
+
+#resource-views {
+ position: absolute;
+ top: 24px;
+ right: 0;
+ left: 200px;
+ bottom: 0;
+}
+
+.source-view-frame {
+ width: 100%;
+ height: 100%;
+}
+
+.sidebar-resizer-vertical {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ width: 5px;
+ z-index: 500;
+ cursor: col-resize;
+}
+
+.resources .sidebar-resizer-vertical {
+ top: 24px;
+}
+
+.sidebar-tree, .sidebar-tree .children {
+ position: relative;
+ padding: 0;
+ margin: 0;
+ list-style: none;
+ font-size: 11px;
+}
+
+.sidebar-tree-section {
+ position: relative;
+ height: 18px;
+ padding: 4px 10px 6px 10px;
+ white-space: nowrap;
+ margin-top: 1px;
+ color: rgb(92, 110, 129);
+ font-weight: bold;
+ text-shadow: rgba(255, 255, 255, 0.75) 0 1px 0;
+}
+
+.sidebar-tree-item {
+ position: relative;
+ height: 36px;
+ padding: 0 5px 0 5px;
+ white-space: nowrap;
+ margin-top: 1px;
+ line-height: 34px;
+ border-top: 1px solid transparent;
+}
+
+.sidebar-tree .children {
+ display: none;
+}
+
+.sidebar-tree .children.expanded {
+ display: block;
+}
+
+.sidebar-tree-section + .children > .sidebar-tree-item {
+ padding-left: 10px !important;
+}
+
+.sidebar-tree-section + .children.small > .sidebar-tree-item {
+ padding-left: 17px !important;
+}
+
+.sidebar-tree > .children > .sidebar-tree-item {
+ padding-left: 37px;
+}
+
+.sidebar-tree > .children > .children > .sidebar-tree-item {
+ padding-left: 37px;
+}
+
+.sidebar-tree.hide-disclosure-buttons > .children {
+ display: none;
+}
+
+.sidebar-tree > .children.hide-disclosure-buttons > .children {
+ display: none;
+}
+
+.sidebar-tree.some-expandable:not(.hide-disclosure-buttons) > .sidebar-tree-item:not(.parent) .icon {
+ margin-left: 16px;
+}
+
+.sidebar-tree-item .disclosure-button {
+ float: left;
+ width: 16px;
+ height: 100%;
+ border: 0;
+ background-color: transparent;
+ background-image: url(Images/disclosureTriangleSmallRight.png);
+ background-repeat: no-repeat;
+ background-position: center;
+ -webkit-apearance: none;
+}
+
+.sidebar-tree.hide-disclosure-buttons .sidebar-tree-item .disclosure-button {
+ display: none;
+}
+
+body.inactive .sidebar-tree-item .disclosure-button {
+ background-image: url(Images/disclosureTriangleSmallRightBlack.png);
+}
+
+body.inactive .sidebar-tree-item.expanded .disclosure-button {
+ background-image: url(Images/disclosureTriangleSmallDownBlack.png);
+}
+
+body.inactive .sidebar-tree-item .disclosure-button:active {
+ background-image: url(Images/disclosureTriangleSmallRightDownBlack.png);
+}
+
+.sidebar-tree-item.selected .disclosure-button {
+ background-image: url(Images/disclosureTriangleSmallRightWhite.png) !important;
+}
+
+.sidebar-tree-item.expanded .disclosure-button {
+ background-image: url(Images/disclosureTriangleSmallDown.png);
+}
+
+.sidebar-tree-item.selected.expanded .disclosure-button {
+ background-image: url(Images/disclosureTriangleSmallDownWhite.png) !important;
+}
+
+.sidebar-tree-item.selected .disclosure-button:active {
+ background-image: url(Images/disclosureTriangleSmallRightDownWhite.png) !important;
+}
+
+.sidebar-tree-item .disclosure-button:active {
+ background-image: url(Images/disclosureTriangleSmallRightDown.png);
+}
+
+.sidebar-tree-item .icon {
+ float: left;
+ width: 32px;
+ height: 32px;
+ margin-top: 1px;
+ margin-right: 3px;
+}
+
+.sidebar-tree-item .status {
+ float: right;
+ height: 16px;
+ margin-top: 9px;
+ margin-left: 4px;
+ line-height: 1em;
+}
+
+.sidebar-tree-item .status:empty {
+ display: none;
+}
+
+.sidebar-tree-item .status .bubble {
+ display: inline-block;
+ height: 14px;
+ min-width: 16px;
+ margin-top: 1px;
+ background-color: rgb(128, 151, 189);
+ vertical-align: middle;
+ white-space: nowrap;
+ padding: 1px 4px;
+ text-align: center;
+ font-size: 11px;
+ font-family: Helvetia, Arial, sans-serif;
+ font-weight: bold;
+ text-shadow: none;
+ color: white;
+ -webkit-border-radius: 7px;
+}
+
+.sidebar-tree-item .status .bubble:empty {
+ display: none;
+}
+
+.sidebar-tree-item.selected .status .bubble {
+ background-color: white !important;
+ color: rgb(132, 154, 190) !important;
+}
+
+:focus .sidebar-tree-item.selected .status .bubble {
+ color: rgb(36, 98, 172) !important;
+}
+
+body.inactive .sidebar-tree-item.selected .status .bubble {
+ color: rgb(159, 159, 159) !important;
+}
+
+.sidebar-tree.small .sidebar-tree-item, .sidebar-tree .children.small .sidebar-tree-item, .sidebar-tree-item.small, .small .resources-graph-side {
+ height: 20px;
+}
+
+.sidebar-tree.small .sidebar-tree-item .icon, .sidebar-tree .children.small .sidebar-tree-item .icon, .sidebar-tree-item.small .icon {
+ width: 16px;
+ height: 16px;
+}
+
+.sidebar-tree.small .sidebar-tree-item .status, .sidebar-tree .children.small .sidebar-tree-item .status, .sidebar-tree-item.small .status {
+ margin-top: 1px;
+}
+
+.sidebar-tree-item.selected {
+ color: white;
+ border-top: 1px solid rgb(145, 160, 192);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(162, 177, 207)), to(rgb(120, 138, 177)));
+ text-shadow: rgba(0, 0, 0, 0.33) 0 1px 0;
+ font-weight: bold;
+ -webkit-background-origin: padding;
+ -webkit-background-clip: padding;
+}
+
+:focus .sidebar-tree-item.selected {
+ border-top: 1px solid rgb(68, 128, 200);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(92, 147, 213)), to(rgb(21, 83, 170)));
+}
+
+body.inactive .sidebar-tree-item.selected {
+ border-top: 1px solid rgb(151, 151, 151);
+ background-image: -webkit-gradient(linear, left top, left bottom, from(rgb(180, 180, 180)), to(rgb(138, 138, 138)));
+}
+
+.sidebar-tree-item .titles {
+ position: relative;
+ top: 5px;
+ line-height: 11px;
+ padding-bottom: 1px;
+ text-overflow: ellipsis;
+ overflow: hidden;
+ white-space: nowrap;
+}
+
+.sidebar-tree-item .titles.no-subtitle {
+ top: 10px;
+}
+
+.sidebar-tree.small .sidebar-tree-item .titles, .sidebar-tree .children.small .sidebar-tree-item .titles, .sidebar-tree-item.small .titles {
+ top: 2px;
+ line-height: normal;
+}
+
+.sidebar-tree:not(.small) .sidebar-tree-item:not(.small) .title::after, .sidebar-tree .children:not(.small) .sidebar-tree-item .title::after {
+ content: "\A";
+ white-space: pre;
+}
+
+.sidebar-tree-item .subtitle {
+ font-size: 9px;
+ color: rgba(0, 0, 0, 0.7);
+}
+
+.sidebar-tree.small .sidebar-tree-item .subtitle, .sidebar-tree .children.small .sidebar-tree-item .subtitle, .sidebar-tree-item.small .subtitle {
+ display: none;
+}
+
+.sidebar-tree-item.selected .subtitle {
+ color: rgba(255, 255, 255, 0.9);
+}
+
+#resources-graphs {
+ position: absolute;
+ left: 0;
+ right: 0;
+ max-height: 100%;
+ top: 112px;
+}
+
+.resources-graph-side {
+ position: relative;
+ height: 36px;
+ padding: 0 5px;
+ white-space: nowrap;
+ margin-top: 1px;
+ border-top: 1px solid transparent;
+ overflow: hidden;
+}
+
+.resources-graph-bar-area {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ right: 8px;
+ left: 9px;
+}
+
+#resources-container:not(.viewing-resource) .resource-sidebar-tree-item:nth-of-type(2n) {
+ background-color: rgba(0, 0, 0, 0.05);
+}
+
+#resources-container:not(.viewing-resource) .resources-graph-side:nth-of-type(2n) {
+ background-color: rgba(0, 0, 0, 0.05);
+}
+
+.resources-time-graph-sidebar-item .icon {
+ content: url(Images/resourcesTimeGraphIcon.png);
+}
+
+.resources-size-graph-sidebar-item .icon {
+ content: url(Images/resourcesSizeGraphIcon.png);
+}
+
+.resources-size-graph-sidebar-item .icon {
+ content: url(Images/resourcesSizeGraphIcon.png);
+}
+
+.resource-sidebar-tree-item .icon {
+ content: url(Images/resourcePlainIcon.png);
+}
+
+.children.small .resource-sidebar-tree-item .icon {
+ content: url(Images/resourcePlainIconSmall.png);
+}
+
+.resource-sidebar-tree-item.resources-category-documents .icon {
+ content: url(Images/resourceDocumentIcon.png);
+}
+
+.children.small .resource-sidebar-tree-item.resources-category-documents .icon {
+ content: url(Images/resourceDocumentIconSmall.png);
+}
+
+.resource-sidebar-tree-item.resources-category-stylesheets .icon {
+ content: url(Images/resourceCSSIcon.png);
+}
+
+.children.small .resource-sidebar-tree-item.resources-category-stylesheets .icon {
+ content: url(Images/resourceDocumentIconSmall.png);
+}
+
+.resource-sidebar-tree-item.resources-category-images .icon {
+ position: relative;
+ background-image: url(Images/resourcePlainIcon.png);
+ background-repeat: no-repeat;
+ content: "";
+}
+
+.resource-sidebar-tree-item.resources-category-images .image-resource-icon-preview {
+ position: absolute;
+ margin: auto;
+ top: 3px;
+ bottom: 4px;
+ left: 5px;
+ right: 5px;
+ max-width: 18px;
+ max-height: 21px;
+ min-width: 1px;
+ min-height: 1px;
+}
+
+.children.small .resource-sidebar-tree-item.resources-category-images .icon {
+ background-image: url(Images/resourcePlainIconSmall.png);
+ content: "";
+}
+
+.children.small .resource-sidebar-tree-item.resources-category-images .image-resource-icon-preview {
+ top: 2px;
+ bottom: 1px;
+ left: 3px;
+ right: 3px;
+ max-width: 8px;
+ max-height: 11px;
+}
+
+.resource-sidebar-tree-item.resources-category-fonts .icon {
+ content: url(Images/resourcePlainIcon.png);
+}
+
+.children.small .resource-sidebar-tree-item.resources-category-fonts .icon {
+ content: url(Images/resourcePlainIconSmall.png);
+}
+
+.resource-sidebar-tree-item.resources-category-scripts .icon {
+ content: url(Images/resourceJSIcon.png);
+}
+
+.children.small .resource-sidebar-tree-item.resources-category-scripts .icon {
+ content: url(Images/resourceDocumentIconSmall.png);
+}
+
+.resource-sidebar-tree-item.resources-category-xhr .icon {
+ content: url(Images/resourcePlainIcon.png);
+}
+
+.children.small .resource-sidebar-tree-item.resources-category-xhr .icon {
+ content: url(Images/resourceDocumentIconSmall.png);
+}
+
+.bubble.debug, .console-debug-level .bubble {
+ background-color: rgb(0, 0, 255) !important;
+}
+
+.bubble.warning, .console-warning-level .bubble {
+ background-color: rgb(232, 164, 0) !important;
+}
+
+.bubble.error, .console-error-level .bubble {
+ background-color: rgb(216, 35, 35) !important;
+}
+
+.bubble.search-matches {
+ background-image: url(Images/searchSmallWhite.png);
+ background-repeat: no-repeat;
+ background-position: 3px 2px;
+ padding-left: 13px !important;
+}
+
+.sidebar-tree-item.selected .bubble.search-matches {
+ background-image: url(Images/searchSmallBlue.png);
+}
+
+:focus .sidebar-tree-item.selected .bubble.search-matches {
+ background-image: url(Images/searchSmallBrightBlue.png);
+}
+
+body.inactive .sidebar-tree-item.selected .bubble.search-matches {
+ background-image: url(Images/searchSmallGray.png);
+}
+
+/* Profiler Style */
+
+#profile-views {
+ position: absolute;
+ top: 0;
+ right: 0;
+ left: 200px;
+ bottom: 0;
+}
+
+#profile-view-status-bar-items {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ left: 200px;
+ overflow: hidden;
+ border-left: 1px solid rgb(184, 184, 184);
+ margin-left: -1px;
+}
+
+.profile-sidebar-tree-item .icon {
+ content: url(Images/profileIcon.png);
+}
+
+.profile-sidebar-tree-item.small .icon {
+ content: url(Images/profileSmallIcon.png);
+}
+
+.profile-group-sidebar-tree-item .icon {
+ content: url(Images/profileGroupIcon.png);
+}
+
+.profile-view {
+ display: none;
+ overflow: hidden;
+ position: absolute;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+}
+
+.profile-view.visible {
+ display: block;
+}
+
+.profile-view .data-grid {
+ border: none;
+ height: 100%;
+}
+
+.profile-view .data-grid th.average-column {
+ text-align: center;
+}
+
+.profile-view .data-grid td.average-column {
+ text-align: right;
+}
+
+.profile-view .data-grid th.self-column {
+ text-align: center;
+}
+
+.profile-view .data-grid td.self-column {
+ text-align: right;
+}
+
+.profile-view .data-grid th.total-column {
+ text-align: center;
+}
+
+.profile-view .data-grid td.total-column {
+ text-align: right;
+}
+
+.profile-view .data-grid .calls-column {
+ text-align: center;
+}
+
+.profile-node-file {
+ float: right;
+ color: gray;
+ margin-top: -1px;
+}
+
+.data-grid tr.selected .profile-node-file {
+ color: rgb(33%, 33%, 33%);
+}
+
+.data-grid:focus tr.selected .profile-node-file {
+ color: white;
+}
+
+button.enable-toggle-status-bar-item .glyph {
+}
+
+.record-profile-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/recordButtonGlyph.png);
+}
+
+.record-profile-status-bar-item.toggled-on .glyph {
+ -webkit-mask-image: url(Images/recordToggledButtonGlyph.png);
+ background-color: rgb(216, 0, 0) !important;
+}
+
+/* FIXME: should have its own glyph. */
+.heap-snapshot-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/focusButtonGlyph.png);
+}
+
+.node-search-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/nodeSearchButtonGlyph.png);
+}
+
+.percent-time-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/percentButtonGlyph.png);
+}
+
+.focus-profile-node-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/focusButtonGlyph.png);
+}
+
+.exclude-profile-node-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/excludeButtonGlyph.png);
+}
+
+.reset-profile-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/reloadButtonGlyph.png);
+}
+
+.delete-storage-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/excludeButtonGlyph.png);
+}
+
+#storage-view-status-bar-items {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ left: 200px;
+ overflow: hidden;
+ border-left: 1px solid rgb(184, 184, 184);
+ margin-left: -1px;
+}
+
+.refresh-storage-status-bar-item .glyph {
+ -webkit-mask-image: url(Images/reloadButtonGlyph.png);
+}
+
+#storage-view-status-bar-items {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ left: 200px;
+ overflow: hidden;
+ border-left: 1px solid rgb(184, 184, 184);
+ margin-left: -1px;
+}
+
+ol.breakpoint-list {
+ -webkit-padding-start: 2px;
+ list-style: none;
+ margin: 0;
+}
+
+.breakpoint-list li {
+ white-space: nowrap;
+ text-overflow: ellipsis;
+ overflow: hidden;
+ margin: 4px 0;
+}
+
+.breakpoint-list .checkbox-elem {
+ font-size: 10px;
+ margin: 0 4px;
+ vertical-align: top;
+ position: relative;
+ z-index: 1;
+}
+
+.breakpoint-list .source-text {
+ font-family: monospace;
+ white-space: nowrap;
+ text-overflow: ellipsis;
+ overflow: hidden;
+ margin: 2px 0 0px 20px;
+}
+
+.breakpoint-list a {
+ color: rgb(33%, 33%, 33%);
+ cursor: pointer;
+}
+
+.breakpoint-list a:hover {
+ color: rgb(15%, 15%, 15%);
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.html b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.html
new file mode 100644
index 0000000..7f544fe
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.html
@@ -0,0 +1,112 @@
+<!--
+Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions
+are met:
+
+1. Redistributions of source code must retain the above copyright
+ notice, this list of conditions and the following disclaimer.
+2. Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in the
+ documentation and/or other materials provided with the distribution.
+3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ its contributors may be used to endorse or promote products derived
+ from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+-->
+<!DOCTYPE html>
+<html>
+<head>
+ <meta http-equiv="content-type" content="text/html; charset=utf-8">
+ <link rel="stylesheet" type="text/css" href="inspector.css">
+ <script type="text/javascript" src="utilities.js"></script>
+ <script type="text/javascript" src="treeoutline.js"></script>
+ <script type="text/javascript" src="inspector.js"></script>
+ <script type="text/javascript" src="Object.js"></script>
+ <script type="text/javascript" src="KeyboardShortcut.js"></script>
+ <script type="text/javascript" src="TextPrompt.js"></script>
+ <script type="text/javascript" src="Popup.js"></script>
+ <script type="text/javascript" src="Placard.js"></script>
+ <script type="text/javascript" src="View.js"></script>
+ <script type="text/javascript" src="Callback.js"></script>
+ <script type="text/javascript" src="Drawer.js"></script>
+ <script type="text/javascript" src="ChangesView.js"></script>
+ <script type="text/javascript" src="ConsoleView.js"></script>
+ <script type="text/javascript" src="Resource.js"></script>
+ <script type="text/javascript" src="ResourceCategory.js"></script>
+ <script type="text/javascript" src="Database.js"></script>
+ <script type="text/javascript" src="DOMStorage.js"></script>
+ <script type="text/javascript" src="DOMStorageItemsView.js"></script>
+ <script type="text/javascript" src="DataGrid.js"></script>
+ <script type="text/javascript" src="DOMStorageDataGrid.js"></script>
+ <script type="text/javascript" src="CookieItemsView.js"></script>
+ <script type="text/javascript" src="Script.js"></script>
+ <script type="text/javascript" src="Breakpoint.js"></script>
+ <script type="text/javascript" src="SidebarPane.js"></script>
+ <script type="text/javascript" src="ElementsTreeOutline.js"></script>
+ <script type="text/javascript" src="SidebarTreeElement.js"></script>
+ <script type="text/javascript" src="PropertiesSection.js"></script>
+ <script type="text/javascript" src="ObjectProxy.js"></script>
+ <script type="text/javascript" src="ObjectPropertiesSection.js"></script>
+ <script type="text/javascript" src="BreakpointsSidebarPane.js"></script>
+ <script type="text/javascript" src="CallStackSidebarPane.js"></script>
+ <script type="text/javascript" src="ScopeChainSidebarPane.js"></script>
+ <script type="text/javascript" src="WatchExpressionsSidebarPane.js"></script>
+ <script type="text/javascript" src="MetricsSidebarPane.js"></script>
+ <script type="text/javascript" src="PropertiesSidebarPane.js"></script>
+ <script type="text/javascript" src="Color.js"></script>
+ <script type="text/javascript" src="StylesSidebarPane.js"></script>
+ <script type="text/javascript" src="Panel.js"></script>
+ <script type="text/javascript" src="PanelEnablerView.js"></script>
+ <script type="text/javascript" src="StatusBarButton.js"></script>
+ <script type="text/javascript" src="SummaryBar.js"></script>
+ <script type="text/javascript" src="ElementsPanel.js"></script>
+ <script type="text/javascript" src="ResourcesPanel.js"></script>
+ <script type="text/javascript" src="ScriptsPanel.js"></script>
+ <script type="text/javascript" src="StoragePanel.js"></script>
+ <script type="text/javascript" src="ProfilesPanel.js"></script>
+ <script type="text/javascript" src="ResourceView.js"></script>
+ <script type="text/javascript" src="SourceFrame.js"></script>
+ <script type="text/javascript" src="SourceView.js"></script>
+ <script type="text/javascript" src="FontView.js"></script>
+ <script type="text/javascript" src="ImageView.js"></script>
+ <script type="text/javascript" src="DatabaseTableView.js"></script>
+ <script type="text/javascript" src="DatabaseQueryView.js"></script>
+ <script type="text/javascript" src="ScriptView.js"></script>
+ <script type="text/javascript" src="ProfileDataGridTree.js"></script>
+ <script type="text/javascript" src="BottomUpProfileDataGridTree.js"></script>
+ <script type="text/javascript" src="TopDownProfileDataGridTree.js"></script>
+ <script type="text/javascript" src="ProfileView.js"></script>
+ <script type="text/javascript" src="DOMAgent.js"></script>
+ <script type="text/javascript" src="InjectedScript.js"></script>
+ <script type="text/javascript" src="InjectedScriptAccess.js"></script>
+ <script type="text/javascript" src="TimelineAgent.js"></script>
+</head>
+<body class="detached">
+ <div id="toolbar">
+ <div class="toolbar-item close"><button id="close-button"></button></div>
+ <div class="toolbar-item flexable-space"></div>
+ <div class="toolbar-item hidden" id="search-results-matches"></div>
+ <div class="toolbar-item"><input id="search" type="search" incremental results="0"><div id="search-toolbar-label" class="toolbar-label"></div></div>
+ </div>
+ <div id="main">
+ <div id="main-panels" tabindex="0" spellcheck="false"></div>
+ <div id="main-status-bar" class="status-bar"><div id="anchored-status-bar-items"><button id="dock-status-bar-item" class="status-bar-item toggled"><div class="glyph"></div><div class="glyph shadow"></div></button><button id="console-status-bar-item" class="status-bar-item"><div class="glyph"></div><div class="glyph shadow"></div></button><button id="changes-status-bar-item" class="status-bar-item hidden"></button><div id="count-items"><div id="changes-count" class="hidden"></div><div id="error-warning-count" class="hidden"></div></div></div></div>
+ </div>
+ <div id="drawer">
+ <div id="console-view"><div id="console-messages"><div id="console-prompt" spellcheck="false"><br></div></div></div>
+ <div id="drawer-status-bar" class="status-bar"><div id="other-drawer-status-bar-items"><button id="clear-console-status-bar-item" class="status-bar-item"><div class="glyph"></div><div class="glyph shadow"></div></button><div id="console-filter" class="status-bar-item"></div></div></div>
+ </div>
+</body>
+</html>
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.js
new file mode 100644
index 0000000..902dd94
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector.js
@@ -0,0 +1,1553 @@
+/*
+ * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
+ * Copyright (C) 2007 Matt Lilek (pewtermoose@gmail.com).
+ * Copyright (C) 2009 Joseph Pecoraro
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+var Preferences = {
+ ignoreWhitespace: true,
+ showUserAgentStyles: true,
+ maxInlineTextChildLength: 80,
+ minConsoleHeight: 75,
+ minSidebarWidth: 100,
+ minElementsSidebarWidth: 200,
+ minScriptsSidebarWidth: 200,
+ showInheritedComputedStyleProperties: false,
+ styleRulesExpandedState: {},
+ showMissingLocalizedStrings: false,
+ heapProfilerPresent: false,
+ samplingCPUProfiler: false,
+ showColorNicknames: true,
+ colorFormat: "hex"
+}
+
+var WebInspector = {
+ resources: {},
+ resourceURLMap: {},
+ missingLocalizedStrings: {},
+
+ get previousFocusElement()
+ {
+ return this._previousFocusElement;
+ },
+
+ get currentFocusElement()
+ {
+ return this._currentFocusElement;
+ },
+
+ set currentFocusElement(x)
+ {
+ if (this._currentFocusElement !== x)
+ this._previousFocusElement = this._currentFocusElement;
+ this._currentFocusElement = x;
+
+ if (this._currentFocusElement) {
+ this._currentFocusElement.focus();
+
+ // Make a caret selection inside the new element if there isn't a range selection and
+ // there isn't already a caret selection inside.
+ var selection = window.getSelection();
+ if (selection.isCollapsed && !this._currentFocusElement.isInsertionCaretInside()) {
+ var selectionRange = this._currentFocusElement.ownerDocument.createRange();
+ selectionRange.setStart(this._currentFocusElement, 0);
+ selectionRange.setEnd(this._currentFocusElement, 0);
+
+ selection.removeAllRanges();
+ selection.addRange(selectionRange);
+ }
+ } else if (this._previousFocusElement)
+ this._previousFocusElement.blur();
+ },
+
+ get currentPanel()
+ {
+ return this._currentPanel;
+ },
+
+ set currentPanel(x)
+ {
+ if (this._currentPanel === x)
+ return;
+
+ if (this._currentPanel)
+ this._currentPanel.hide();
+
+ this._currentPanel = x;
+
+ this.updateSearchLabel();
+
+ if (x) {
+ x.show();
+
+ if (this.currentQuery) {
+ if (x.performSearch) {
+ function performPanelSearch()
+ {
+ this.updateSearchMatchesCount();
+
+ x.currentQuery = this.currentQuery;
+ x.performSearch(this.currentQuery);
+ }
+
+ // Perform the search on a timeout so the panel switches fast.
+ setTimeout(performPanelSearch.bind(this), 0);
+ } else {
+ // Update to show Not found for panels that can't be searched.
+ this.updateSearchMatchesCount();
+ }
+ }
+ }
+
+ for (var panelName in WebInspector.panels) {
+ if (WebInspector.panels[panelName] == x)
+ InspectorController.storeLastActivePanel(panelName);
+ }
+ },
+
+ _createPanels: function()
+ {
+ var hiddenPanels = (InspectorController.hiddenPanels() || "").split(',');
+ if (hiddenPanels.indexOf("elements") === -1)
+ this.panels.elements = new WebInspector.ElementsPanel();
+ if (hiddenPanels.indexOf("resources") === -1)
+ this.panels.resources = new WebInspector.ResourcesPanel();
+ if (hiddenPanels.indexOf("scripts") === -1)
+ this.panels.scripts = new WebInspector.ScriptsPanel();
+ if (hiddenPanels.indexOf("profiles") === -1)
+ this.panels.profiles = new WebInspector.ProfilesPanel();
+ if (hiddenPanels.indexOf("storage") === -1 && hiddenPanels.indexOf("databases") === -1)
+ this.panels.storage = new WebInspector.StoragePanel();
+ },
+
+ get attached()
+ {
+ return this._attached;
+ },
+
+ set attached(x)
+ {
+ if (this._attached === x)
+ return;
+
+ this._attached = x;
+
+ this.updateSearchLabel();
+
+ var dockToggleButton = document.getElementById("dock-status-bar-item");
+ var body = document.body;
+
+ if (x) {
+ InspectorController.attach();
+ body.removeStyleClass("detached");
+ body.addStyleClass("attached");
+ dockToggleButton.title = WebInspector.UIString("Undock into separate window.");
+ } else {
+ InspectorController.detach();
+ body.removeStyleClass("attached");
+ body.addStyleClass("detached");
+ dockToggleButton.title = WebInspector.UIString("Dock to main window.");
+ }
+ },
+
+ get errors()
+ {
+ return this._errors || 0;
+ },
+
+ set errors(x)
+ {
+ x = Math.max(x, 0);
+
+ if (this._errors === x)
+ return;
+ this._errors = x;
+ this._updateErrorAndWarningCounts();
+ },
+
+ get warnings()
+ {
+ return this._warnings || 0;
+ },
+
+ set warnings(x)
+ {
+ x = Math.max(x, 0);
+
+ if (this._warnings === x)
+ return;
+ this._warnings = x;
+ this._updateErrorAndWarningCounts();
+ },
+
+ _updateErrorAndWarningCounts: function()
+ {
+ var errorWarningElement = document.getElementById("error-warning-count");
+ if (!errorWarningElement)
+ return;
+
+ if (!this.errors && !this.warnings) {
+ errorWarningElement.addStyleClass("hidden");
+ return;
+ }
+
+ errorWarningElement.removeStyleClass("hidden");
+
+ errorWarningElement.removeChildren();
+
+ if (this.errors) {
+ var errorElement = document.createElement("span");
+ errorElement.id = "error-count";
+ errorElement.textContent = this.errors;
+ errorWarningElement.appendChild(errorElement);
+ }
+
+ if (this.warnings) {
+ var warningsElement = document.createElement("span");
+ warningsElement.id = "warning-count";
+ warningsElement.textContent = this.warnings;
+ errorWarningElement.appendChild(warningsElement);
+ }
+
+ if (this.errors) {
+ if (this.warnings) {
+ if (this.errors == 1) {
+ if (this.warnings == 1)
+ errorWarningElement.title = WebInspector.UIString("%d error, %d warning", this.errors, this.warnings);
+ else
+ errorWarningElement.title = WebInspector.UIString("%d error, %d warnings", this.errors, this.warnings);
+ } else if (this.warnings == 1)
+ errorWarningElement.title = WebInspector.UIString("%d errors, %d warning", this.errors, this.warnings);
+ else
+ errorWarningElement.title = WebInspector.UIString("%d errors, %d warnings", this.errors, this.warnings);
+ } else if (this.errors == 1)
+ errorWarningElement.title = WebInspector.UIString("%d error", this.errors);
+ else
+ errorWarningElement.title = WebInspector.UIString("%d errors", this.errors);
+ } else if (this.warnings == 1)
+ errorWarningElement.title = WebInspector.UIString("%d warning", this.warnings);
+ else if (this.warnings)
+ errorWarningElement.title = WebInspector.UIString("%d warnings", this.warnings);
+ else
+ errorWarningElement.title = null;
+ },
+
+ get styleChanges()
+ {
+ return this._styleChanges;
+ },
+
+ set styleChanges(x)
+ {
+ x = Math.max(x, 0);
+
+ if (this._styleChanges === x)
+ return;
+ this._styleChanges = x;
+ this._updateChangesCount();
+ },
+
+ _updateChangesCount: function()
+ {
+ // TODO: Remove immediate return when enabling the Changes Panel
+ return;
+
+ var changesElement = document.getElementById("changes-count");
+ if (!changesElement)
+ return;
+
+ if (!this.styleChanges) {
+ changesElement.addStyleClass("hidden");
+ return;
+ }
+
+ changesElement.removeStyleClass("hidden");
+ changesElement.removeChildren();
+
+ if (this.styleChanges) {
+ var styleChangesElement = document.createElement("span");
+ styleChangesElement.id = "style-changes-count";
+ styleChangesElement.textContent = this.styleChanges;
+ changesElement.appendChild(styleChangesElement);
+ }
+
+ if (this.styleChanges) {
+ if (this.styleChanges === 1)
+ changesElement.title = WebInspector.UIString("%d style change", this.styleChanges);
+ else
+ changesElement.title = WebInspector.UIString("%d style changes", this.styleChanges);
+ }
+ },
+
+ get hoveredDOMNode()
+ {
+ return this._hoveredDOMNode;
+ },
+
+ set hoveredDOMNode(x)
+ {
+ if (this._hoveredDOMNode === x)
+ return;
+
+ this._hoveredDOMNode = x;
+
+ if (this._hoveredDOMNode)
+ this._updateHoverHighlightSoon(this.showingDOMNodeHighlight ? 50 : 500);
+ else
+ this._updateHoverHighlight();
+ },
+
+ _updateHoverHighlightSoon: function(delay)
+ {
+ if ("_updateHoverHighlightTimeout" in this)
+ clearTimeout(this._updateHoverHighlightTimeout);
+ this._updateHoverHighlightTimeout = setTimeout(this._updateHoverHighlight.bind(this), delay);
+ },
+
+ _updateHoverHighlight: function()
+ {
+ if ("_updateHoverHighlightTimeout" in this) {
+ clearTimeout(this._updateHoverHighlightTimeout);
+ delete this._updateHoverHighlightTimeout;
+ }
+
+ if (this._hoveredDOMNode) {
+ InspectorController.highlightDOMNode(this._hoveredDOMNode.id);
+ this.showingDOMNodeHighlight = true;
+ } else {
+ InspectorController.hideDOMNodeHighlight();
+ this.showingDOMNodeHighlight = false;
+ }
+ }
+}
+
+WebInspector.loaded = function()
+{
+ var platform = InspectorController.platform();
+ document.body.addStyleClass("platform-" + platform);
+
+ var colorFormat = InspectorController.setting("color-format");
+ if (colorFormat)
+ Preferences.colorFormat = colorFormat;
+
+ this.drawer = new WebInspector.Drawer();
+ this.console = new WebInspector.ConsoleView(this.drawer);
+ // TODO: Uncomment when enabling the Changes Panel
+ // this.changes = new WebInspector.ChangesView(this.drawer);
+ // TODO: Remove class="hidden" from inspector.html on button#changes-status-bar-item
+ this.drawer.visibleView = this.console;
+ this.domAgent = new WebInspector.DOMAgent();
+
+ this.resourceCategories = {
+ documents: new WebInspector.ResourceCategory(WebInspector.UIString("Documents"), "documents"),
+ stylesheets: new WebInspector.ResourceCategory(WebInspector.UIString("Stylesheets"), "stylesheets"),
+ images: new WebInspector.ResourceCategory(WebInspector.UIString("Images"), "images"),
+ scripts: new WebInspector.ResourceCategory(WebInspector.UIString("Scripts"), "scripts"),
+ xhr: new WebInspector.ResourceCategory(WebInspector.UIString("XHR"), "xhr"),
+ fonts: new WebInspector.ResourceCategory(WebInspector.UIString("Fonts"), "fonts"),
+ other: new WebInspector.ResourceCategory(WebInspector.UIString("Other"), "other")
+ };
+
+ this.panels = {};
+ this._createPanels();
+
+ var toolbarElement = document.getElementById("toolbar");
+ var previousToolbarItem = toolbarElement.children[0];
+
+ this.panelOrder = [];
+ for (var panelName in this.panels) {
+ var panel = this.panels[panelName];
+ var panelToolbarItem = panel.toolbarItem;
+ this.panelOrder.push(panel);
+ panelToolbarItem.addEventListener("click", this._toolbarItemClicked.bind(this));
+ if (previousToolbarItem)
+ toolbarElement.insertBefore(panelToolbarItem, previousToolbarItem.nextSibling);
+ else
+ toolbarElement.insertBefore(panelToolbarItem, toolbarElement.firstChild);
+ previousToolbarItem = panelToolbarItem;
+ }
+
+ this.Tips = {
+ ResourceNotCompressed: {id: 0, message: WebInspector.UIString("You could save bandwidth by having your web server compress this transfer with gzip or zlib.")}
+ };
+
+ this.Warnings = {
+ IncorrectMIMEType: {id: 0, message: WebInspector.UIString("Resource interpreted as %s but transferred with MIME type %s.")}
+ };
+
+ this.addMainEventListeners(document);
+
+ window.addEventListener("unload", this.windowUnload.bind(this), true);
+ window.addEventListener("resize", this.windowResize.bind(this), true);
+
+ document.addEventListener("focus", this.focusChanged.bind(this), true);
+ document.addEventListener("keydown", this.documentKeyDown.bind(this), true);
+ document.addEventListener("keyup", this.documentKeyUp.bind(this), true);
+ document.addEventListener("beforecopy", this.documentCanCopy.bind(this), true);
+ document.addEventListener("copy", this.documentCopy.bind(this), true);
+ document.addEventListener("contextmenu", this.contextMenu.bind(this), true);
+
+ var mainPanelsElement = document.getElementById("main-panels");
+ mainPanelsElement.handleKeyEvent = this.mainKeyDown.bind(this);
+ mainPanelsElement.handleKeyUpEvent = this.mainKeyUp.bind(this);
+ mainPanelsElement.handleCopyEvent = this.mainCopy.bind(this);
+
+ // Focus the mainPanelsElement in a timeout so it happens after the initial focus,
+ // so it doesn't get reset to the first toolbar button. This initial focus happens
+ // on Mac when the window is made key and the WebHTMLView becomes the first responder.
+ setTimeout(function() { WebInspector.currentFocusElement = mainPanelsElement }, 0);
+
+ var dockToggleButton = document.getElementById("dock-status-bar-item");
+ dockToggleButton.addEventListener("click", this.toggleAttach.bind(this), false);
+
+ if (this.attached)
+ dockToggleButton.title = WebInspector.UIString("Undock into separate window.");
+ else
+ dockToggleButton.title = WebInspector.UIString("Dock to main window.");
+
+ var errorWarningCount = document.getElementById("error-warning-count");
+ errorWarningCount.addEventListener("click", this.showConsole.bind(this), false);
+ this._updateErrorAndWarningCounts();
+
+ this.styleChanges = 0;
+ // TODO: Uncomment when enabling the Changes Panel
+ // var changesElement = document.getElementById("changes-count");
+ // changesElement.addEventListener("click", this.showChanges.bind(this), false);
+ // this._updateErrorAndWarningCounts();
+
+ var searchField = document.getElementById("search");
+ searchField.addEventListener("keydown", this.searchKeyDown.bind(this), false);
+ searchField.addEventListener("keyup", this.searchKeyUp.bind(this), false);
+ searchField.addEventListener("search", this.performSearch.bind(this), false); // when the search is emptied
+
+ document.getElementById("toolbar").addEventListener("mousedown", this.toolbarDragStart, true);
+ document.getElementById("close-button").addEventListener("click", this.close, true);
+
+ InspectorController.loaded();
+}
+
+var windowLoaded = function()
+{
+ var localizedStringsURL = InspectorController.localizedStringsURL();
+ if (localizedStringsURL) {
+ var localizedStringsScriptElement = document.createElement("script");
+ localizedStringsScriptElement.addEventListener("load", WebInspector.loaded.bind(WebInspector), false);
+ localizedStringsScriptElement.type = "text/javascript";
+ localizedStringsScriptElement.src = localizedStringsURL;
+ document.getElementsByTagName("head").item(0).appendChild(localizedStringsScriptElement);
+ } else
+ WebInspector.loaded();
+
+ window.removeEventListener("load", windowLoaded, false);
+ delete windowLoaded;
+};
+
+window.addEventListener("load", windowLoaded, false);
+
+WebInspector.dispatch = function() {
+ var methodName = arguments[0];
+ var parameters = Array.prototype.slice.call(arguments, 1);
+ WebInspector[methodName].apply(this, parameters);
+}
+
+WebInspector.windowUnload = function(event)
+{
+ InspectorController.windowUnloading();
+}
+
+WebInspector.windowResize = function(event)
+{
+ if (this.currentPanel && this.currentPanel.resize)
+ this.currentPanel.resize();
+}
+
+WebInspector.windowFocused = function(event)
+{
+ if (event.target.nodeType === Node.DOCUMENT_NODE)
+ document.body.removeStyleClass("inactive");
+}
+
+WebInspector.windowBlured = function(event)
+{
+ if (event.target.nodeType === Node.DOCUMENT_NODE)
+ document.body.addStyleClass("inactive");
+}
+
+WebInspector.focusChanged = function(event)
+{
+ this.currentFocusElement = event.target;
+}
+
+WebInspector.setAttachedWindow = function(attached)
+{
+ this.attached = attached;
+}
+
+WebInspector.close = function(event)
+{
+ InspectorController.closeWindow();
+}
+
+WebInspector.documentClick = function(event)
+{
+ var anchor = event.target.enclosingNodeOrSelfWithNodeName("a");
+ if (!anchor)
+ return;
+
+ // Prevent the link from navigating, since we don't do any navigation by following links normally.
+ event.preventDefault();
+
+ function followLink()
+ {
+ // FIXME: support webkit-html-external-link links here.
+ if (anchor.href in WebInspector.resourceURLMap) {
+ if (anchor.hasStyleClass("webkit-html-external-link")) {
+ anchor.removeStyleClass("webkit-html-external-link");
+ anchor.addStyleClass("webkit-html-resource-link");
+ }
+
+ WebInspector.showResourceForURL(anchor.href, anchor.lineNumber, anchor.preferredPanel);
+ } else {
+ var profileStringRegEx = new RegExp("webkit-profile://.+/([0-9]+)");
+ var profileString = profileStringRegEx.exec(anchor.href);
+ if (profileString)
+ WebInspector.showProfileById(profileString[1])
+ }
+ }
+
+ if (WebInspector.followLinkTimeout)
+ clearTimeout(WebInspector.followLinkTimeout);
+
+ if (anchor.preventFollowOnDoubleClick) {
+ // Start a timeout if this is the first click, if the timeout is canceled
+ // before it fires, then a double clicked happened or another link was clicked.
+ if (event.detail === 1)
+ WebInspector.followLinkTimeout = setTimeout(followLink, 333);
+ return;
+ }
+
+ followLink();
+}
+
+WebInspector.documentKeyDown = function(event)
+{
+ if (!this.currentFocusElement)
+ return;
+ if (this.currentFocusElement.handleKeyEvent)
+ this.currentFocusElement.handleKeyEvent(event);
+ else if (this.currentFocusElement.id && this.currentFocusElement.id.length && WebInspector[this.currentFocusElement.id + "KeyDown"])
+ WebInspector[this.currentFocusElement.id + "KeyDown"](event);
+
+ if (!event.handled) {
+ var isMac = InspectorController.platform().indexOf("mac-") === 0;
+
+ switch (event.keyIdentifier) {
+ case "U+001B": // Escape key
+ this.drawer.visible = !this.drawer.visible;
+ event.preventDefault();
+ break;
+
+ case "U+0046": // F key
+ if (isMac)
+ var isFindKey = event.metaKey && !event.ctrlKey && !event.altKey && !event.shiftKey;
+ else
+ var isFindKey = event.ctrlKey && !event.metaKey && !event.altKey && !event.shiftKey;
+
+ if (isFindKey) {
+ var searchField = document.getElementById("search");
+ searchField.focus();
+ searchField.select();
+ event.preventDefault();
+ }
+
+ break;
+
+ case "U+0047": // G key
+ if (isMac)
+ var isFindAgainKey = event.metaKey && !event.ctrlKey && !event.altKey;
+ else
+ var isFindAgainKey = event.ctrlKey && !event.metaKey && !event.altKey;
+
+ if (isFindAgainKey) {
+ if (event.shiftKey) {
+ if (this.currentPanel.jumpToPreviousSearchResult)
+ this.currentPanel.jumpToPreviousSearchResult();
+ } else if (this.currentPanel.jumpToNextSearchResult)
+ this.currentPanel.jumpToNextSearchResult();
+ event.preventDefault();
+ }
+
+ break;
+
+ case "U+005B": // [ key
+ if (isMac)
+ var isRotateLeft = event.metaKey && !event.shiftKey && !event.ctrlKey && !event.altKey;
+ else
+ var isRotateLeft = event.ctrlKey && !event.shiftKey && !event.metaKey && !event.altKey;
+
+ if (isRotateLeft) {
+ var index = this.panelOrder.indexOf(this.currentPanel);
+ index = (index === 0) ? this.panelOrder.length - 1 : index - 1;
+ this.panelOrder[index].toolbarItem.click();
+ event.preventDefault();
+ }
+
+ break;
+
+ case "U+005D": // ] key
+ if (isMac)
+ var isRotateRight = event.metaKey && !event.shiftKey && !event.ctrlKey && !event.altKey;
+ else
+ var isRotateRight = event.ctrlKey && !event.shiftKey && !event.metaKey && !event.altKey;
+
+ if (isRotateRight) {
+ var index = this.panelOrder.indexOf(this.currentPanel);
+ index = (index + 1) % this.panelOrder.length;
+ this.panelOrder[index].toolbarItem.click();
+ event.preventDefault();
+ }
+
+ break;
+ }
+ }
+}
+
+WebInspector.documentKeyUp = function(event)
+{
+ if (!this.currentFocusElement || !this.currentFocusElement.handleKeyUpEvent)
+ return;
+ this.currentFocusElement.handleKeyUpEvent(event);
+}
+
+WebInspector.documentCanCopy = function(event)
+{
+ if (!this.currentFocusElement)
+ return;
+ // Calling preventDefault() will say "we support copying, so enable the Copy menu".
+ if (this.currentFocusElement.handleCopyEvent)
+ event.preventDefault();
+ else if (this.currentFocusElement.id && this.currentFocusElement.id.length && WebInspector[this.currentFocusElement.id + "Copy"])
+ event.preventDefault();
+}
+
+WebInspector.documentCopy = function(event)
+{
+ if (!this.currentFocusElement)
+ return;
+ if (this.currentFocusElement.handleCopyEvent)
+ this.currentFocusElement.handleCopyEvent(event);
+ else if (this.currentFocusElement.id && this.currentFocusElement.id.length && WebInspector[this.currentFocusElement.id + "Copy"])
+ WebInspector[this.currentFocusElement.id + "Copy"](event);
+}
+
+WebInspector.contextMenu = function(event)
+{
+ if (event.handled || event.target.hasStyleClass("popup-glasspane"))
+ event.preventDefault();
+}
+
+WebInspector.mainKeyDown = function(event)
+{
+ if (this.currentPanel && this.currentPanel.handleKeyEvent)
+ this.currentPanel.handleKeyEvent(event);
+}
+
+WebInspector.mainKeyUp = function(event)
+{
+ if (this.currentPanel && this.currentPanel.handleKeyUpEvent)
+ this.currentPanel.handleKeyUpEvent(event);
+}
+
+WebInspector.mainCopy = function(event)
+{
+ if (this.currentPanel && this.currentPanel.handleCopyEvent)
+ this.currentPanel.handleCopyEvent(event);
+}
+
+WebInspector.animateStyle = function(animations, duration, callback, complete)
+{
+ if (complete === undefined)
+ complete = 0;
+ var slice = (1000 / 30); // 30 frames per second
+
+ var defaultUnit = "px";
+ var propertyUnit = {opacity: ""};
+
+ for (var i = 0; i < animations.length; ++i) {
+ var animation = animations[i];
+ var element = null;
+ var start = null;
+ var current = null;
+ var end = null;
+ var key = null;
+ for (key in animation) {
+ if (key === "element")
+ element = animation[key];
+ else if (key === "start")
+ start = animation[key];
+ else if (key === "current")
+ current = animation[key];
+ else if (key === "end")
+ end = animation[key];
+ }
+
+ if (!element || !end)
+ continue;
+
+ var computedStyle = element.ownerDocument.defaultView.getComputedStyle(element);
+ if (!start) {
+ start = {};
+ for (key in end)
+ start[key] = parseInt(computedStyle.getPropertyValue(key));
+ animation.start = start;
+ } else if (complete == 0)
+ for (key in start)
+ element.style.setProperty(key, start[key] + (key in propertyUnit ? propertyUnit[key] : defaultUnit));
+
+ if (!current) {
+ current = {};
+ for (key in start)
+ current[key] = start[key];
+ animation.current = current;
+ }
+
+ function cubicInOut(t, b, c, d)
+ {
+ if ((t/=d/2) < 1) return c/2*t*t*t + b;
+ return c/2*((t-=2)*t*t + 2) + b;
+ }
+
+ var style = element.style;
+ for (key in end) {
+ var startValue = start[key];
+ var currentValue = current[key];
+ var endValue = end[key];
+ if ((complete + slice) < duration) {
+ var delta = (endValue - startValue) / (duration / slice);
+ var newValue = cubicInOut(complete, startValue, endValue - startValue, duration);
+ style.setProperty(key, newValue + (key in propertyUnit ? propertyUnit[key] : defaultUnit));
+ current[key] = newValue;
+ } else {
+ style.setProperty(key, endValue + (key in propertyUnit ? propertyUnit[key] : defaultUnit));
+ }
+ }
+ }
+
+ if (complete < duration)
+ setTimeout(WebInspector.animateStyle, slice, animations, duration, callback, complete + slice);
+ else if (callback)
+ callback();
+}
+
+WebInspector.updateSearchLabel = function()
+{
+ if (!this.currentPanel)
+ return;
+
+ var newLabel = WebInspector.UIString("Search %s", this.currentPanel.toolbarItemLabel);
+ if (this.attached)
+ document.getElementById("search").setAttribute("placeholder", newLabel);
+ else {
+ document.getElementById("search").removeAttribute("placeholder");
+ document.getElementById("search-toolbar-label").textContent = newLabel;
+ }
+}
+
+WebInspector.toggleAttach = function()
+{
+ this.attached = !this.attached;
+}
+
+WebInspector.toolbarDragStart = function(event)
+{
+ if (!WebInspector.attached && InspectorController.platform() !== "mac-leopard")
+ return;
+
+ var target = event.target;
+ if (target.hasStyleClass("toolbar-item") && target.hasStyleClass("toggleable"))
+ return;
+
+ var toolbar = document.getElementById("toolbar");
+ if (target !== toolbar && !target.hasStyleClass("toolbar-item"))
+ return;
+
+ toolbar.lastScreenX = event.screenX;
+ toolbar.lastScreenY = event.screenY;
+
+ WebInspector.elementDragStart(toolbar, WebInspector.toolbarDrag, WebInspector.toolbarDragEnd, event, (WebInspector.attached ? "row-resize" : "default"));
+}
+
+WebInspector.toolbarDragEnd = function(event)
+{
+ var toolbar = document.getElementById("toolbar");
+
+ WebInspector.elementDragEnd(event);
+
+ delete toolbar.lastScreenX;
+ delete toolbar.lastScreenY;
+}
+
+WebInspector.toolbarDrag = function(event)
+{
+ var toolbar = document.getElementById("toolbar");
+
+ if (WebInspector.attached) {
+ var height = window.innerHeight - (event.screenY - toolbar.lastScreenY);
+
+ InspectorController.setAttachedWindowHeight(height);
+ } else {
+ var x = event.screenX - toolbar.lastScreenX;
+ var y = event.screenY - toolbar.lastScreenY;
+
+ // We cannot call window.moveBy here because it restricts the movement
+ // of the window at the edges.
+ InspectorController.moveByUnrestricted(x, y);
+ }
+
+ toolbar.lastScreenX = event.screenX;
+ toolbar.lastScreenY = event.screenY;
+
+ event.preventDefault();
+}
+
+WebInspector.elementDragStart = function(element, dividerDrag, elementDragEnd, event, cursor)
+{
+ if (this._elementDraggingEventListener || this._elementEndDraggingEventListener)
+ this.elementDragEnd(event);
+
+ this._elementDraggingEventListener = dividerDrag;
+ this._elementEndDraggingEventListener = elementDragEnd;
+
+ document.addEventListener("mousemove", dividerDrag, true);
+ document.addEventListener("mouseup", elementDragEnd, true);
+
+ document.body.style.cursor = cursor;
+
+ event.preventDefault();
+}
+
+WebInspector.elementDragEnd = function(event)
+{
+ document.removeEventListener("mousemove", this._elementDraggingEventListener, true);
+ document.removeEventListener("mouseup", this._elementEndDraggingEventListener, true);
+
+ document.body.style.removeProperty("cursor");
+
+ delete this._elementDraggingEventListener;
+ delete this._elementEndDraggingEventListener;
+
+ event.preventDefault();
+}
+
+WebInspector.showConsole = function()
+{
+ this.drawer.showView(this.console);
+}
+
+WebInspector.showChanges = function()
+{
+ this.drawer.showView(this.changes);
+}
+
+WebInspector.showElementsPanel = function()
+{
+ this.currentPanel = this.panels.elements;
+}
+
+WebInspector.showResourcesPanel = function()
+{
+ this.currentPanel = this.panels.resources;
+}
+
+WebInspector.showScriptsPanel = function()
+{
+ this.currentPanel = this.panels.scripts;
+}
+
+WebInspector.showProfilesPanel = function()
+{
+ this.currentPanel = this.panels.profiles;
+}
+
+WebInspector.showStoragePanel = function()
+{
+ this.currentPanel = this.panels.storage;
+}
+
+WebInspector.addResource = function(identifier, payload)
+{
+ var resource = new WebInspector.Resource(
+ payload.requestHeaders,
+ payload.requestURL,
+ payload.host,
+ payload.path,
+ payload.lastPathComponent,
+ identifier,
+ payload.isMainResource,
+ payload.cached,
+ payload.requestMethod,
+ payload.requestFormData);
+ this.resources[identifier] = resource;
+ this.resourceURLMap[resource.url] = resource;
+
+ if (resource.mainResource) {
+ this.mainResource = resource;
+ this.panels.elements.reset();
+ }
+
+ if (this.panels.resources)
+ this.panels.resources.addResource(resource);
+}
+
+WebInspector.clearConsoleMessages = function()
+{
+ WebInspector.console.clearMessages(false);
+}
+
+WebInspector.selectDatabase = function(o)
+{
+ WebInspector.showStoragePanel();
+ WebInspector.panels.storage.selectDatabase(o);
+}
+
+WebInspector.selectDOMStorage = function(o)
+{
+ WebInspector.showStoragePanel();
+ WebInspector.panels.storage.selectDOMStorage(o);
+}
+
+WebInspector.updateResource = function(identifier, payload)
+{
+ var resource = this.resources[identifier];
+ if (!resource)
+ return;
+
+ if (payload.didRequestChange) {
+ resource.url = payload.url;
+ resource.domain = payload.domain;
+ resource.path = payload.path;
+ resource.lastPathComponent = payload.lastPathComponent;
+ resource.requestHeaders = payload.requestHeaders;
+ resource.mainResource = payload.mainResource;
+ resource.requestMethod = payload.requestMethod;
+ resource.requestFormData = payload.requestFormData;
+ }
+
+ if (payload.didResponseChange) {
+ resource.mimeType = payload.mimeType;
+ resource.suggestedFilename = payload.suggestedFilename;
+ resource.expectedContentLength = payload.expectedContentLength;
+ resource.statusCode = payload.statusCode;
+ resource.suggestedFilename = payload.suggestedFilename;
+ resource.responseHeaders = payload.responseHeaders;
+ }
+
+ if (payload.didTypeChange) {
+ resource.type = payload.type;
+ }
+
+ if (payload.didLengthChange) {
+ resource.contentLength = payload.contentLength;
+ }
+
+ if (payload.didCompletionChange) {
+ resource.failed = payload.failed;
+ resource.finished = payload.finished;
+ }
+
+ if (payload.didTimingChange) {
+ if (payload.startTime)
+ resource.startTime = payload.startTime;
+ if (payload.responseReceivedTime)
+ resource.responseReceivedTime = payload.responseReceivedTime;
+ if (payload.endTime)
+ resource.endTime = payload.endTime;
+ }
+}
+
+WebInspector.removeResource = function(identifier)
+{
+ var resource = this.resources[identifier];
+ if (!resource)
+ return;
+
+ resource.category.removeResource(resource);
+ delete this.resourceURLMap[resource.url];
+ delete this.resources[identifier];
+
+ if (this.panels.resources)
+ this.panels.resources.removeResource(resource);
+}
+
+WebInspector.addDatabase = function(payload)
+{
+ var database = new WebInspector.Database(
+ payload.database,
+ payload.domain,
+ payload.name,
+ payload.version);
+ this.panels.storage.addDatabase(database);
+}
+
+WebInspector.addDOMStorage = function(payload)
+{
+ var domStorage = new WebInspector.DOMStorage(
+ payload.domStorage,
+ payload.host,
+ payload.isLocalStorage);
+ this.panels.storage.addDOMStorage(domStorage);
+}
+
+WebInspector.resourceTrackingWasEnabled = function()
+{
+ this.panels.resources.resourceTrackingWasEnabled();
+}
+
+WebInspector.resourceTrackingWasDisabled = function()
+{
+ this.panels.resources.resourceTrackingWasDisabled();
+}
+
+WebInspector.attachDebuggerWhenShown = function()
+{
+ this.panels.scripts.attachDebuggerWhenShown();
+}
+
+WebInspector.debuggerWasEnabled = function()
+{
+ this.panels.scripts.debuggerWasEnabled();
+}
+
+WebInspector.debuggerWasDisabled = function()
+{
+ this.panels.scripts.debuggerWasDisabled();
+}
+
+WebInspector.profilerWasEnabled = function()
+{
+ this.panels.profiles.profilerWasEnabled();
+}
+
+WebInspector.profilerWasDisabled = function()
+{
+ this.panels.profiles.profilerWasDisabled();
+}
+
+WebInspector.parsedScriptSource = function(sourceID, sourceURL, source, startingLine)
+{
+ this.panels.scripts.addScript(sourceID, sourceURL, source, startingLine);
+}
+
+WebInspector.failedToParseScriptSource = function(sourceURL, source, startingLine, errorLine, errorMessage)
+{
+ this.panels.scripts.addScript(null, sourceURL, source, startingLine, errorLine, errorMessage);
+}
+
+WebInspector.pausedScript = function(callFrames)
+{
+ this.panels.scripts.debuggerPaused(callFrames);
+}
+
+WebInspector.resumedScript = function()
+{
+ this.panels.scripts.debuggerResumed();
+}
+
+WebInspector.populateInterface = function()
+{
+ for (var panelName in this.panels) {
+ var panel = this.panels[panelName];
+ if ("populateInterface" in panel)
+ panel.populateInterface();
+ }
+}
+
+WebInspector.reset = function()
+{
+ for (var panelName in this.panels) {
+ var panel = this.panels[panelName];
+ if ("reset" in panel)
+ panel.reset();
+ }
+
+ for (var category in this.resourceCategories)
+ this.resourceCategories[category].removeAllResources();
+
+ this.resources = {};
+ this.resourceURLMap = {};
+ this.hoveredDOMNode = null;
+
+ delete this.mainResource;
+
+ this.console.clearMessages();
+}
+
+WebInspector.resourceURLChanged = function(resource, oldURL)
+{
+ delete this.resourceURLMap[oldURL];
+ this.resourceURLMap[resource.url] = resource;
+}
+
+WebInspector.addMessageToConsole = function(payload)
+{
+ var consoleMessage = new WebInspector.ConsoleMessage(
+ payload.source,
+ payload.type,
+ payload.level,
+ payload.line,
+ payload.url,
+ payload.groupLevel,
+ payload.repeatCount);
+ consoleMessage.setMessageBody(Array.prototype.slice.call(arguments, 1));
+ this.console.addMessage(consoleMessage);
+}
+
+WebInspector.log = function(message)
+{
+ var msg = new WebInspector.ConsoleMessage(
+ WebInspector.ConsoleMessage.MessageSource.Other,
+ WebInspector.ConsoleMessage.MessageType.Log,
+ WebInspector.ConsoleMessage.MessageLevel.Debug,
+ -1,
+ null,
+ null,
+ 1,
+ message);
+ this.console.addMessage(msg);
+}
+
+WebInspector.addProfile = function(profile)
+{
+ this.panels.profiles.addProfile(profile);
+}
+
+WebInspector.setRecordingProfile = function(isProfiling)
+{
+ this.panels.profiles.setRecordingProfile(isProfiling);
+}
+
+WebInspector.drawLoadingPieChart = function(canvas, percent) {
+ var g = canvas.getContext("2d");
+ var darkColor = "rgb(122, 168, 218)";
+ var lightColor = "rgb(228, 241, 251)";
+ var cx = 8;
+ var cy = 8;
+ var r = 7;
+
+ g.beginPath();
+ g.arc(cx, cy, r, 0, Math.PI * 2, false);
+ g.closePath();
+
+ g.lineWidth = 1;
+ g.strokeStyle = darkColor;
+ g.fillStyle = lightColor;
+ g.fill();
+ g.stroke();
+
+ var startangle = -Math.PI / 2;
+ var endangle = startangle + (percent * Math.PI * 2);
+
+ g.beginPath();
+ g.moveTo(cx, cy);
+ g.arc(cx, cy, r, startangle, endangle, false);
+ g.closePath();
+
+ g.fillStyle = darkColor;
+ g.fill();
+}
+
+WebInspector.updateFocusedNode = function(nodeId)
+{
+ var node = WebInspector.domAgent.nodeForId(nodeId);
+ if (!node)
+ // FIXME: Should we deselect if null is passed in?
+ return;
+
+ this.currentPanel = this.panels.elements;
+ this.panels.elements.focusedDOMNode = node;
+}
+
+WebInspector.displayNameForURL = function(url)
+{
+ if (!url)
+ return "";
+ var resource = this.resourceURLMap[url];
+ if (resource)
+ return resource.displayName;
+ return url.trimURL(WebInspector.mainResource ? WebInspector.mainResource.domain : "");
+}
+
+WebInspector.resourceForURL = function(url)
+{
+ if (url in this.resourceURLMap)
+ return this.resourceURLMap[url];
+
+ // No direct match found. Search for resources that contain
+ // a substring of the URL.
+ for (var resourceURL in this.resourceURLMap) {
+ if (resourceURL.hasSubstring(url))
+ return this.resourceURLMap[resourceURL];
+ }
+
+ return null;
+}
+
+WebInspector.showResourceForURL = function(url, line, preferredPanel)
+{
+ var resource = this.resourceForURL(url);
+ if (!resource)
+ return false;
+
+ if (preferredPanel && preferredPanel in WebInspector.panels) {
+ var panel = this.panels[preferredPanel];
+ if (!("showResource" in panel))
+ panel = null;
+ else if ("canShowResource" in panel && !panel.canShowResource(resource))
+ panel = null;
+ }
+
+ this.currentPanel = panel || this.panels.resources;
+ if (!this.currentPanel)
+ return false;
+ this.currentPanel.showResource(resource, line);
+ return true;
+}
+
+WebInspector.linkifyStringAsFragment = function(string)
+{
+ var container = document.createDocumentFragment();
+ var linkStringRegEx = new RegExp("(?:[a-zA-Z][a-zA-Z0-9+.-]{2,}://|www\\.)[\\w$\\-_+*'=\\|/\\\\(){}[\\]%@&#~,:;.!?]{2,}[\\w$\\-_+*=\\|/\\\\({%@&#~]");
+
+ while (string) {
+ var linkString = linkStringRegEx.exec(string);
+ if (!linkString)
+ break;
+
+ linkString = linkString[0];
+ var title = linkString;
+ var linkIndex = string.indexOf(linkString);
+ var nonLink = string.substring(0, linkIndex);
+ container.appendChild(document.createTextNode(nonLink));
+
+ var profileStringRegEx = new RegExp("webkit-profile://(.+)/[0-9]+");
+ var profileStringMatches = profileStringRegEx.exec(title);
+ var profileTitle;
+ if (profileStringMatches)
+ profileTitle = profileStringMatches[1];
+ if (profileTitle)
+ title = WebInspector.panels.profiles.displayTitleForProfileLink(profileTitle);
+
+ var realURL = (linkString.indexOf("www.") === 0 ? "http://" + linkString : linkString);
+ container.appendChild(WebInspector.linkifyURLAsNode(realURL, title, null, (realURL in WebInspector.resourceURLMap)));
+ string = string.substring(linkIndex + linkString.length, string.length);
+ }
+
+ if (string)
+ container.appendChild(document.createTextNode(string));
+
+ return container;
+}
+
+WebInspector.showProfileById = function(uid) {
+ WebInspector.showProfilesPanel();
+ WebInspector.panels.profiles.showProfileById(uid);
+}
+
+WebInspector.linkifyURLAsNode = function(url, linkText, classes, isExternal)
+{
+ if (!linkText)
+ linkText = url;
+ classes = (classes ? classes + " " : "");
+ classes += isExternal ? "webkit-html-external-link" : "webkit-html-resource-link";
+
+ var a = document.createElement("a");
+ a.href = url;
+ a.className = classes;
+ a.title = url;
+ a.target = "_blank";
+ a.textContent = linkText;
+
+ return a;
+}
+
+WebInspector.linkifyURL = function(url, linkText, classes, isExternal)
+{
+ // Use the DOM version of this function so as to avoid needing to escape attributes.
+ // FIXME: Get rid of linkifyURL entirely.
+ return WebInspector.linkifyURLAsNode(url, linkText, classes, isExternal).outerHTML;
+}
+
+WebInspector.addMainEventListeners = function(doc)
+{
+ doc.defaultView.addEventListener("focus", this.windowFocused.bind(this), true);
+ doc.defaultView.addEventListener("blur", this.windowBlured.bind(this), true);
+ doc.addEventListener("click", this.documentClick.bind(this), true);
+}
+
+WebInspector.searchKeyDown = function(event)
+{
+ if (event.keyIdentifier !== "Enter")
+ return;
+
+ // Call preventDefault since this was the Enter key. This prevents a "search" event
+ // from firing for key down. We handle the Enter key on key up in searchKeyUp. This
+ // stops performSearch from being called twice in a row.
+ event.preventDefault();
+}
+
+WebInspector.searchKeyUp = function(event)
+{
+ if (event.keyIdentifier !== "Enter")
+ return;
+
+ // Select all of the text so the user can easily type an entirely new query.
+ event.target.select();
+
+ // Only call performSearch if the Enter key was pressed. Otherwise the search
+ // performance is poor because of searching on every key. The search field has
+ // the incremental attribute set, so we still get incremental searches.
+ this.performSearch(event);
+}
+
+WebInspector.performSearch = function(event)
+{
+ var query = event.target.value;
+ var forceSearch = event.keyIdentifier === "Enter";
+
+ if (!query || !query.length || (!forceSearch && query.length < 3)) {
+ delete this.currentQuery;
+
+ for (var panelName in this.panels) {
+ var panel = this.panels[panelName];
+ if (panel.currentQuery && panel.searchCanceled)
+ panel.searchCanceled();
+ delete panel.currentQuery;
+ }
+
+ this.updateSearchMatchesCount();
+
+ return;
+ }
+
+ if (query === this.currentPanel.currentQuery && this.currentPanel.currentQuery === this.currentQuery) {
+ // When this is the same query and a forced search, jump to the next
+ // search result for a good user experience.
+ if (forceSearch && this.currentPanel.jumpToNextSearchResult)
+ this.currentPanel.jumpToNextSearchResult();
+ return;
+ }
+
+ this.currentQuery = query;
+
+ this.updateSearchMatchesCount();
+
+ if (!this.currentPanel.performSearch)
+ return;
+
+ this.currentPanel.currentQuery = query;
+ this.currentPanel.performSearch(query);
+}
+
+WebInspector.addNodesToSearchResult = function(nodeIds)
+{
+ WebInspector.panels.elements.addNodesToSearchResult(nodeIds);
+}
+
+WebInspector.updateSearchMatchesCount = function(matches, panel)
+{
+ if (!panel)
+ panel = this.currentPanel;
+
+ panel.currentSearchMatches = matches;
+
+ if (panel !== this.currentPanel)
+ return;
+
+ if (!this.currentPanel.currentQuery) {
+ document.getElementById("search-results-matches").addStyleClass("hidden");
+ return;
+ }
+
+ if (matches) {
+ if (matches === 1)
+ var matchesString = WebInspector.UIString("1 match");
+ else
+ var matchesString = WebInspector.UIString("%d matches", matches);
+ } else
+ var matchesString = WebInspector.UIString("Not Found");
+
+ var matchesToolbarElement = document.getElementById("search-results-matches");
+ matchesToolbarElement.removeStyleClass("hidden");
+ matchesToolbarElement.textContent = matchesString;
+}
+
+WebInspector.UIString = function(string)
+{
+ if (window.localizedStrings && string in window.localizedStrings)
+ string = window.localizedStrings[string];
+ else {
+ if (!(string in this.missingLocalizedStrings)) {
+ console.error("Localized string \"" + string + "\" not found.");
+ this.missingLocalizedStrings[string] = true;
+ }
+
+ if (Preferences.showMissingLocalizedStrings)
+ string += " (not localized)";
+ }
+
+ return String.vsprintf(string, Array.prototype.slice.call(arguments, 1));
+}
+
+WebInspector.isBeingEdited = function(element)
+{
+ return element.__editing;
+}
+
+WebInspector.startEditing = function(element, committedCallback, cancelledCallback, context)
+{
+ if (element.__editing)
+ return;
+ element.__editing = true;
+
+ var oldText = getContent(element);
+ var oldHandleKeyEvent = element.handleKeyEvent;
+ var moveDirection = "";
+
+ element.addStyleClass("editing");
+
+ var oldTabIndex = element.tabIndex;
+ if (element.tabIndex < 0)
+ element.tabIndex = 0;
+
+ function blurEventListener() {
+ editingCommitted.call(element);
+ }
+
+ function getContent(element) {
+ if (element.tagName === "INPUT" && element.type === "text")
+ return element.value;
+ else
+ return element.textContent;
+ }
+
+ function cleanUpAfterEditing() {
+ delete this.__editing;
+
+ this.removeStyleClass("editing");
+ this.tabIndex = oldTabIndex;
+ this.scrollTop = 0;
+ this.scrollLeft = 0;
+
+ this.handleKeyEvent = oldHandleKeyEvent;
+ element.removeEventListener("blur", blurEventListener, false);
+
+ if (element === WebInspector.currentFocusElement || element.isAncestor(WebInspector.currentFocusElement))
+ WebInspector.currentFocusElement = WebInspector.previousFocusElement;
+ }
+
+ function editingCancelled() {
+ if (this.tagName === "INPUT" && this.type === "text")
+ this.value = oldText;
+ else
+ this.textContent = oldText;
+
+ cleanUpAfterEditing.call(this);
+
+ if (cancelledCallback)
+ cancelledCallback(this, context);
+ }
+
+ function editingCommitted() {
+ cleanUpAfterEditing.call(this);
+
+ if (committedCallback)
+ committedCallback(this, getContent(this), oldText, context, moveDirection);
+ }
+
+ element.handleKeyEvent = function(event) {
+ if (oldHandleKeyEvent)
+ oldHandleKeyEvent(event);
+ if (event.handled)
+ return;
+
+ if (event.keyIdentifier === "Enter") {
+ editingCommitted.call(element);
+ event.preventDefault();
+ } else if (event.keyCode === 27) { // Escape key
+ editingCancelled.call(element);
+ event.preventDefault();
+ event.handled = true;
+ } else if (event.keyIdentifier === "U+0009") // Tab key
+ moveDirection = (event.shiftKey ? "backward" : "forward");
+ }
+
+ element.addEventListener("blur", blurEventListener, false);
+
+ WebInspector.currentFocusElement = element;
+}
+
+WebInspector._toolbarItemClicked = function(event)
+{
+ var toolbarItem = event.currentTarget;
+ this.currentPanel = toolbarItem.panel;
+}
+
+// This table maps MIME types to the Resource.Types which are valid for them.
+// The following line:
+// "text/html": {0: 1},
+// means that text/html is a valid MIME type for resources that have type
+// WebInspector.Resource.Type.Document (which has a value of 0).
+WebInspector.MIMETypes = {
+ "text/html": {0: true},
+ "text/xml": {0: true},
+ "text/plain": {0: true},
+ "application/xhtml+xml": {0: true},
+ "text/css": {1: true},
+ "text/xsl": {1: true},
+ "image/jpeg": {2: true},
+ "image/png": {2: true},
+ "image/gif": {2: true},
+ "image/bmp": {2: true},
+ "image/vnd.microsoft.icon": {2: true},
+ "image/x-icon": {2: true},
+ "image/x-xbitmap": {2: true},
+ "font/ttf": {3: true},
+ "font/opentype": {3: true},
+ "application/x-font-type1": {3: true},
+ "application/x-font-ttf": {3: true},
+ "application/x-truetype-font": {3: true},
+ "text/javascript": {4: true},
+ "text/ecmascript": {4: true},
+ "application/javascript": {4: true},
+ "application/ecmascript": {4: true},
+ "application/x-javascript": {4: true},
+ "text/javascript1.1": {4: true},
+ "text/javascript1.2": {4: true},
+ "text/javascript1.3": {4: true},
+ "text/jscript": {4: true},
+ "text/livescript": {4: true},
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller.js
new file mode 100644
index 0000000..383cba4
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller.js
@@ -0,0 +1,506 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview Stub implementation of the InspectorController API.
+ * This stub class is supposed to make front-end a standalone WebApp
+ * that can be implemented/refactored in isolation from the Web browser
+ * backend. Clients need to subclass it in order to wire calls to the
+ * non-stub backends.
+ */
+goog.provide('devtools.InspectorController');
+
+
+/**
+ * Creates inspector controller stub instance.
+ * @constructor.
+ */
+devtools.InspectorController = function() {
+ /**
+ * @type {boolean}
+ */
+ this.searchingForNode_ = false;
+
+ /**
+ * @type {boolean}
+ */
+ this.windowVisible_ = true;
+
+ /**
+ * @type {number}
+ */
+ this.attachedWindowHeight_ = 0;
+
+ /**
+ * @type {boolean}
+ */
+ this.debuggerEnabled_ = true;
+
+ /**
+ * @type {boolean}
+ */
+ this.profilerEnabled_ = true;
+
+ /**
+ * @type {boolean}
+ */
+ this.resourceTrackingEnabled_ = false;
+
+ /**
+ * @type {boolean}
+ */
+ this.timelineEnabled_ = false;
+
+ /**
+ * @type {Object}
+ */
+ this.settings_ = {};
+};
+
+
+/**
+ * Wraps javascript callback.
+ * @param {function():undefined} func The callback to wrap.
+ * @return {function():undefined} Callback wrapper.
+ */
+devtools.InspectorController.prototype.wrapCallback = function f(func) {
+ // Just return as is.
+ return func;
+};
+
+
+/**
+ * @return {boolean} True iff inspector window is currently visible.
+ */
+devtools.InspectorController.prototype.isWindowVisible = function() {
+ return this.windowVisible_;
+};
+
+
+/**
+ * @return {string} Platform identifier.
+ */
+devtools.InspectorController.prototype.platform = function() {
+ return 'windows';
+};
+
+
+/**
+ * Closes inspector window.
+ */
+devtools.InspectorController.prototype.closeWindow = function() {
+ this.windowVisible_ = false;
+};
+
+
+/**
+ * Attaches frontend to the backend.
+ */
+devtools.InspectorController.prototype.attach = function() {
+};
+
+
+/**
+ * Detaches frontend from the backend.
+ */
+devtools.InspectorController.prototype.detach = function() {
+};
+
+
+/**
+ * Tell host that the active panel has changed.
+ * @param {string} panel Panel name that was last active.
+ */
+devtools.InspectorController.prototype.storeLastActivePanel = function(panel) {
+};
+
+
+/**
+ * Clears console message log in the backend.
+ */
+devtools.InspectorController.prototype.clearMessages = function() {
+};
+
+
+/**
+ * Returns true iff browser is currently in the search for node mode.
+ * @return {boolean} True is currently searching for a node.
+ */
+devtools.InspectorController.prototype.searchingForNode = function() {
+ return this.searchingForNode_;
+};
+
+
+/**
+ * Initiates search for a given query starting on a given row.
+ * @param {number} sourceRow Row to start searching from.
+ * @param {string} query Query string for search for.
+ */
+devtools.InspectorController.prototype.search = function(sourceRow, query) {
+};
+
+
+/**
+ * Toggles node search mode on/off.
+ */
+devtools.InspectorController.prototype.toggleNodeSearch = function() {
+ this.searchingForNode_ = !this.searchingForNode_;
+};
+
+
+/**
+ * Sets the inspector window height while in the attached mode.
+ * @param {number} height Window height being set.
+ */
+devtools.InspectorController.prototype.setAttachedWindowHeight =
+ function(height) {
+ this.attachedWindowHeight_ = height;
+};
+
+
+/**
+ * Moves window by the given offset.
+ * @param {number} x X offset.
+ * @param {number} y Y offset.
+ */
+devtools.InspectorController.prototype.moveByUnrestricted = function(x, y) {
+};
+
+
+/**
+ * Adds resource with given identifier into the given iframe element.
+ * @param {number} identifier Identifier of the resource to add into the frame.
+ * @param {Element} element Element to add resource content to.
+ */
+devtools.InspectorController.prototype.addResourceSourceToFrame =
+ function(identifier, element) {
+};
+
+
+/**
+ * Adds given source of a given mimeType into the given iframe element.
+ * @param {string} mimeType MIME type of the content to be added.
+ * @param {string} source String content to be added.
+ * @param {Element} element Element to add resource content to.
+ */
+devtools.InspectorController.prototype.addSourceToFrame =
+ function(mimeType, source, element) {
+ return false;
+};
+
+
+/**
+ * Returns document node corresponding to the resource with given id.
+ * @return {Node} Node containing the resource.
+ */
+devtools.InspectorController.prototype.getResourceDocumentNode =
+ function(identifier) {
+ return undefined;
+};
+
+
+/**
+ * Highlights the given node on the page.
+ * @param {Node} node Node to highlight.
+ */
+devtools.InspectorController.prototype.highlightDOMNode = function(node) {
+ // Does nothing in stub.
+};
+
+
+/**
+ * Clears current highlight.
+ */
+devtools.InspectorController.prototype.hideDOMNodeHighlight = function() {
+ // Does nothing in stub.
+};
+
+
+/**
+ * @return {window} Inspectable window instance.
+ */
+devtools.InspectorController.prototype.inspectedWindow = function() {
+ return window;
+};
+
+
+/**
+ * Notifies backend that the frontend has been successfully loaded.
+ */
+devtools.InspectorController.prototype.loaded = function() {
+ // Does nothing in stub.
+};
+
+
+/**
+ * @return {string} Url of the i18n-ed strings map.
+ */
+devtools.InspectorController.prototype.localizedStringsURL = function() {
+ return undefined;
+};
+
+
+/**
+ * @return {boolean} True iff window is currently unloading.
+ */
+devtools.InspectorController.prototype.windowUnloading = function() {
+ return false;
+};
+
+
+/**
+ * @return {string} Identifiers of the panels that should be hidden.
+ */
+devtools.InspectorController.prototype.hiddenPanels = function() {
+ return '';
+};
+
+
+/**
+ * @return {boolean} True iff debugger is enabled.
+ */
+devtools.InspectorController.prototype.debuggerEnabled = function() {
+ return this.debuggerEnabled_;
+};
+
+
+/**
+ * Enables resource tracking.
+ */
+devtools.InspectorController.prototype.enableResourceTracking = function() {
+ this.resourceTrackingEnabled_ = true;
+ WebInspector.resourceTrackingWasEnabled();
+};
+
+
+/**
+ * Disables resource tracking.
+ */
+devtools.InspectorController.prototype.disableResourceTracking = function() {
+ this.resourceTrackingEnabled_ = false;
+ WebInspector.resourceTrackingWasDisabled();
+};
+
+
+/**
+ * @return {boolean} True iff resource tracking is enabled.
+ */
+devtools.InspectorController.prototype.resourceTrackingEnabled = function() {
+ return this.resourceTrackingEnabled_;
+};
+
+
+/**
+ * Enables timeline.
+ */
+devtools.InspectorController.prototype.enableTimeline = function() {
+ this.timelineEnabled_ = true;
+ WebInspector.timelineWasEnabled();
+};
+
+
+/**
+ * Disables timeline.
+ */
+devtools.InspectorController.prototype.disableTimeline = function() {
+ this.timelineEnabled_ = false;
+ WebInspector.timelineWasDisabled();
+};
+
+/**
+ * @return {boolean} True iff timeline is enabled.
+ */
+devtools.InspectorController.prototype.timelineEnabled = function() {
+ return this.timelineEnabled_;
+};
+
+
+/**
+ * Enables debugger.
+ */
+devtools.InspectorController.prototype.enableDebugger = function() {
+ this.debuggerEnabled_ = true;
+};
+
+
+/**
+ * Disables debugger.
+ */
+devtools.InspectorController.prototype.disableDebugger = function() {
+ this.debuggerEnabled_ = false;
+};
+
+
+/**
+ * Adds breakpoint to the given line of the source with given ID.
+ * @param {string} sourceID Source Id to add breakpoint to.
+ * @param {number} line Line number to add breakpoint to.
+ * @param {?string} condition The breakpoint condition.
+ */
+devtools.InspectorController.prototype.addBreakpoint =
+ function(sourceID, line, condition) {
+};
+
+
+/**
+ * Removes breakpoint from the given line of the source with given ID.
+ * @param {string} sourceID Source Id to remove breakpoint from.
+ * @param {number} line Line number to remove breakpoint from.
+ */
+devtools.InspectorController.prototype.removeBreakpoint =
+ function(sourceID, line) {
+};
+
+
+/**
+ * Sets a breakpoint condition given a line of the source and an ID.
+ * @param {string} sourceID Source Id to remove breakpoint from.
+ * @param {number} line Line number to remove breakpoint from.
+ * @param {?string} condition New breakpoint condition.
+ */
+devtools.InspectorController.prototype.updateBreakpoint =
+ function(sourceID, line, condition) {
+};
+
+
+/**
+ * Tells backend to pause in the debugger.
+ */
+devtools.InspectorController.prototype.pauseInDebugger = function() {
+ // Does nothing in stub.
+};
+
+
+/**
+ * @return {boolean} True iff the debugger will pause execution on the
+ * exceptions.
+ */
+devtools.InspectorController.prototype.pauseOnExceptions = function() {
+ // Does nothing in stub.
+ return false;
+};
+
+
+/**
+ * Tells whether to pause in the debugger on the exceptions or not.
+ * @param {boolean} value True iff execution should be stopped in the debugger
+ * on the exceptions.
+ */
+devtools.InspectorController.prototype.setPauseOnExceptions = function(value) {
+};
+
+
+/**
+ * Tells backend to resume execution.
+ */
+devtools.InspectorController.prototype.resumeDebugger = function() {
+};
+
+
+/**
+ * @return {boolean} True iff profiler is enabled.
+ */
+devtools.InspectorController.prototype.profilerEnabled = function() {
+ return true;
+};
+
+
+/**
+ * Enables profiler.
+ */
+devtools.InspectorController.prototype.enableProfiler = function() {
+ this.profilerEnabled_ = true;
+};
+
+
+/**
+ * Disables profiler.
+ */
+devtools.InspectorController.prototype.disableProfiler = function() {
+ this.profilerEnabled_ = false;
+};
+
+
+/**
+ * Returns given callframe while on a debugger break.
+ * @return {Object} Current call frame.
+ */
+devtools.InspectorController.prototype.currentCallFrame = function() {
+ return undefined;
+};
+
+
+/**
+ * Tells backend to start collecting profiler data.
+ */
+devtools.InspectorController.prototype.startProfiling = function() {
+};
+
+
+/**
+ * Tells backend to stop collecting profiler data.
+ */
+devtools.InspectorController.prototype.stopProfiling = function() {
+};
+
+
+/**
+ * @return {Array.<Object>} Profile snapshots array.
+ */
+devtools.InspectorController.prototype.profiles = function() {
+ return [];
+};
+
+
+/**
+ * @return {Array.<string>} Database table names available offline.
+ */
+devtools.InspectorController.prototype.databaseTableNames =
+ function(database) {
+ return [];
+};
+
+
+/**
+ * Tells backend to step into the function in debugger.
+ */
+devtools.InspectorController.prototype.stepIntoStatementInDebugger =
+ function() {
+};
+
+
+/**
+ * Tells backend to step out of the function in debugger.
+ */
+devtools.InspectorController.prototype.stepOutOfFunctionInDebugger =
+ function() {};
+
+
+/**
+ * Tells backend to step over the statement in debugger.
+ */
+devtools.InspectorController.prototype.stepOverStatementInDebugger =
+ function() {
+};
+
+
+/**
+ * Sets a setting value in backend.
+ */
+devtools.InspectorController.prototype.setSetting =
+ function(setting, value) {
+ this.settings_[setting] = value;
+};
+
+
+/**
+ * Retrieves a setting value stored in backend.
+ */
+devtools.InspectorController.prototype.setting =
+ function(setting) {
+ return this.settings_[setting];
+};
+
+
+var InspectorController = new devtools.InspectorController();
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller_impl.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller_impl.js
new file mode 100644
index 0000000..5bf19b7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/inspector_controller_impl.js
@@ -0,0 +1,286 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview DevTools' implementation of the InspectorController API.
+ */
+goog.require('devtools.InspectorController');
+
+goog.provide('devtools.InspectorControllerImpl');
+
+devtools.InspectorControllerImpl = function() {
+ devtools.InspectorController.call(this);
+ this.frame_element_id_ = 1;
+
+ this.installInspectorControllerDelegate_('clearMessages');
+ this.installInspectorControllerDelegate_('storeLastActivePanel');
+ this.installInspectorControllerDelegate_('highlightDOMNode');
+ this.installInspectorControllerDelegate_('hideDOMNodeHighlight');
+ this.installInspectorControllerDelegate_('getChildNodes');
+ this.installInspectorControllerDelegate_('setAttribute');
+ this.installInspectorControllerDelegate_('removeAttribute');
+ this.installInspectorControllerDelegate_('setTextNodeValue');
+ this.installInspectorControllerDelegate_('enableResourceTracking');
+ this.installInspectorControllerDelegate_('disableResourceTracking');
+ this.installInspectorControllerDelegate_('enableTimeline');
+ this.installInspectorControllerDelegate_('disableTimeline');
+ this.installInspectorControllerDelegate_('setting');
+ this.installInspectorControllerDelegate_('setSetting');
+};
+goog.inherits(devtools.InspectorControllerImpl,
+ devtools.InspectorController);
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.platform = function() {
+ return DevToolsHost.getPlatform();
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.closeWindow = function() {
+ DevToolsHost.closeWindow();
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.attach = function() {
+ DevToolsHost.dockWindow();
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.detach = function() {
+ DevToolsHost.undockWindow();
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.hiddenPanels = function() {
+ return 'databases';
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.search = function(sourceRow, query) {
+ return DevToolsHost.search(sourceRow, query);
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.toggleNodeSearch = function() {
+ devtools.InspectorController.prototype.toggleNodeSearch.call(this);
+ DevToolsHost.toggleInspectElementMode(this.searchingForNode());
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.localizedStringsURL =
+ function(opt_prefix) {
+ // l10n is turned off in test mode because delayed loading of strings
+ // causes test failures.
+ if (false) {
+ var locale = DevToolsHost.getApplicationLocale();
+ locale = locale.replace('_', '-');
+ return 'l10n/localizedStrings_' + locale + '.js';
+ } else {
+ return undefined;
+ }
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.addSourceToFrame =
+ function(mimeType, source, element) {
+ return DevToolsHost.addSourceToFrame(mimeType, source, element);
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.addResourceSourceToFrame =
+ function(identifier, element) {
+ var resource = WebInspector.resources[identifier];
+ if (!resource) {
+ return;
+ }
+ DevToolsHost.addResourceSourceToFrame(identifier, resource.mimeType, element);
+};
+
+
+/**
+ * {@inheritDoc}.
+ */
+devtools.InspectorControllerImpl.prototype.inspectedWindow = function() {
+ return null;
+};
+
+
+/**
+ * @override
+ */
+devtools.InspectorControllerImpl.prototype.debuggerEnabled = function() {
+ return true;
+};
+
+
+devtools.InspectorControllerImpl.prototype.addBreakpoint = function(
+ sourceID, line, condition) {
+ devtools.tools.getDebuggerAgent().addBreakpoint(sourceID, line, condition);
+};
+
+
+devtools.InspectorControllerImpl.prototype.removeBreakpoint = function(
+ sourceID, line) {
+ devtools.tools.getDebuggerAgent().removeBreakpoint(sourceID, line);
+};
+
+devtools.InspectorControllerImpl.prototype.updateBreakpoint = function(
+ sourceID, line, condition) {
+ devtools.tools.getDebuggerAgent().updateBreakpoint(
+ sourceID, line, condition);
+};
+
+devtools.InspectorControllerImpl.prototype.pauseInDebugger = function() {
+ devtools.tools.getDebuggerAgent().pauseExecution();
+};
+
+
+devtools.InspectorControllerImpl.prototype.resumeDebugger = function() {
+ devtools.tools.getDebuggerAgent().resumeExecution();
+};
+
+
+devtools.InspectorControllerImpl.prototype.stepIntoStatementInDebugger =
+ function() {
+ devtools.tools.getDebuggerAgent().stepIntoStatement();
+};
+
+
+devtools.InspectorControllerImpl.prototype.stepOutOfFunctionInDebugger =
+ function() {
+ devtools.tools.getDebuggerAgent().stepOutOfFunction();
+};
+
+
+devtools.InspectorControllerImpl.prototype.stepOverStatementInDebugger =
+ function() {
+ devtools.tools.getDebuggerAgent().stepOverStatement();
+};
+
+
+/**
+ * @override
+ */
+devtools.InspectorControllerImpl.prototype.pauseOnExceptions = function() {
+ return devtools.tools.getDebuggerAgent().pauseOnExceptions();
+};
+
+
+/**
+ * @override
+ */
+devtools.InspectorControllerImpl.prototype.setPauseOnExceptions = function(
+ value) {
+ return devtools.tools.getDebuggerAgent().setPauseOnExceptions(value);
+};
+
+
+/**
+ * @override
+ */
+devtools.InspectorControllerImpl.prototype.startProfiling = function() {
+ devtools.tools.getDebuggerAgent().startProfiling(
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_CPU);
+};
+
+
+/**
+ * @override
+ */
+devtools.InspectorControllerImpl.prototype.stopProfiling = function() {
+ devtools.tools.getDebuggerAgent().stopProfiling(
+ devtools.DebuggerAgent.ProfilerModules.PROFILER_MODULE_CPU);
+};
+
+
+/**
+ * @override
+ */
+devtools.InspectorControllerImpl.prototype.evaluateInCallFrame =
+ function(callFrameId, code, callback) {
+ devtools.tools.getDebuggerAgent().evaluateInCallFrame(callFrameId, code,
+ callback);
+};
+
+
+/**
+ * @override
+ */
+devtools.InspectorControllerImpl.prototype.dispatchOnInjectedScript = function(
+ callId, methodName, argsString) {
+ var callback = function(result, isException) {
+ WebInspector.didDispatchOnInjectedScript(callId,
+ isException ? result : JSON.parse(result),
+ isException);
+ };
+ RemoteToolsAgent.ExecuteUtilityFunction(
+ devtools.Callback.wrap(callback),
+ 'InjectedScript',
+ JSON.stringify(['dispatch', methodName, argsString]));
+};
+
+
+/**
+ * Installs delegating handler into the inspector controller.
+ * @param {string} methodName Method to install delegating handler for.
+ */
+devtools.InspectorControllerImpl.prototype.installInspectorControllerDelegate_
+ = function(methodName) {
+ this[methodName] = goog.bind(this.callInspectorController_, this,
+ methodName);
+};
+
+
+/**
+ * Bound function with the installInjectedScriptDelegate_ actual
+ * implementation.
+ */
+devtools.InspectorControllerImpl.prototype.callInspectorController_ =
+ function(methodName, var_arg) {
+ var args = Array.prototype.slice.call(arguments);
+ RemoteToolsAgent.ExecuteUtilityFunction(
+ devtools.Callback.wrap(function(){}),
+ 'InspectorController', JSON.stringify(args));
+};
+
+
+devtools.InspectorControllerImpl.parseWrap_ = function(callback) {
+ return devtools.Callback.wrap(
+ function(data) {
+ callback.call(this, JSON.parse(data));
+ });
+};
+
+
+InspectorController = new devtools.InspectorControllerImpl();
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/logreader.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/logreader.js
new file mode 100644
index 0000000..88ab907
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/logreader.js
@@ -0,0 +1,320 @@
+// Copyright 2009 the V8 project authors. All rights reserved.
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following
+// disclaimer in the documentation and/or other materials provided
+// with the distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived
+// from this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+/**
+ * @fileoverview Log Reader is used to process log file produced by V8.
+ */
+
+// Initlialize namespaces
+var devtools = devtools || {};
+devtools.profiler = devtools.profiler || {};
+
+
+/**
+ * Base class for processing log files.
+ *
+ * @param {Array.<Object>} dispatchTable A table used for parsing and processing
+ * log records.
+ * @constructor
+ */
+devtools.profiler.LogReader = function(dispatchTable) {
+ /**
+ * @type {Array.<Object>}
+ */
+ this.dispatchTable_ = dispatchTable;
+ this.dispatchTable_['alias'] =
+ { parsers: [null, null], processor: this.processAlias_ };
+ this.dispatchTable_['repeat'] =
+ { parsers: [parseInt, 'var-args'], processor: this.processRepeat_,
+ backrefs: true };
+
+ /**
+ * A key-value map for aliases. Translates short name -> full name.
+ * @type {Object}
+ */
+ this.aliases_ = {};
+
+ /**
+ * A key-value map for previous address values.
+ * @type {Object}
+ */
+ this.prevAddresses_ = {};
+
+ /**
+ * A key-value map for events than can be backreference-compressed.
+ * @type {Object}
+ */
+ this.backRefsCommands_ = {};
+ this.initBackRefsCommands_();
+
+ /**
+ * Back references for decompression.
+ * @type {Array.<string>}
+ */
+ this.backRefs_ = [];
+};
+
+
+/**
+ * Creates a parser for an address entry.
+ *
+ * @param {string} addressTag Address tag to perform offset decoding.
+ * @return {function(string):number} Address parser.
+ */
+devtools.profiler.LogReader.prototype.createAddressParser = function(
+ addressTag) {
+ var self = this;
+ return (function (str) {
+ var value = parseInt(str, 16);
+ var firstChar = str.charAt(0);
+ if (firstChar == '+' || firstChar == '-') {
+ var addr = self.prevAddresses_[addressTag];
+ addr += value;
+ self.prevAddresses_[addressTag] = addr;
+ return addr;
+ } else if (firstChar != '0' || str.charAt(1) != 'x') {
+ self.prevAddresses_[addressTag] = value;
+ }
+ return value;
+ });
+};
+
+
+/**
+ * Expands an alias symbol, if applicable.
+ *
+ * @param {string} symbol Symbol to expand.
+ * @return {string} Expanded symbol, or the input symbol itself.
+ */
+devtools.profiler.LogReader.prototype.expandAlias = function(symbol) {
+ return symbol in this.aliases_ ? this.aliases_[symbol] : symbol;
+};
+
+
+/**
+ * Used for printing error messages.
+ *
+ * @param {string} str Error message.
+ */
+devtools.profiler.LogReader.prototype.printError = function(str) {
+ // Do nothing.
+};
+
+
+/**
+ * Processes a portion of V8 profiler event log.
+ *
+ * @param {string} chunk A portion of log.
+ */
+devtools.profiler.LogReader.prototype.processLogChunk = function(chunk) {
+ this.processLog_(chunk.split('\n'));
+};
+
+
+/**
+ * Processes stack record.
+ *
+ * @param {number} pc Program counter.
+ * @param {Array.<string>} stack String representation of a stack.
+ * @return {Array.<number>} Processed stack.
+ */
+devtools.profiler.LogReader.prototype.processStack = function(pc, stack) {
+ var fullStack = [pc];
+ var prevFrame = pc;
+ for (var i = 0, n = stack.length; i < n; ++i) {
+ var frame = stack[i];
+ var firstChar = frame.charAt(0);
+ if (firstChar == '+' || firstChar == '-') {
+ // An offset from the previous frame.
+ prevFrame += parseInt(frame, 16);
+ fullStack.push(prevFrame);
+ // Filter out possible 'overflow' string.
+ } else if (firstChar != 'o') {
+ fullStack.push(parseInt(frame, 16));
+ }
+ }
+ return fullStack;
+};
+
+
+/**
+ * Returns whether a particular dispatch must be skipped.
+ *
+ * @param {!Object} dispatch Dispatch record.
+ * @return {boolean} True if dispatch must be skipped.
+ */
+devtools.profiler.LogReader.prototype.skipDispatch = function(dispatch) {
+ return false;
+};
+
+
+/**
+ * Does a dispatch of a log record.
+ *
+ * @param {Array.<string>} fields Log record.
+ * @private
+ */
+devtools.profiler.LogReader.prototype.dispatchLogRow_ = function(fields) {
+ // Obtain the dispatch.
+ var command = fields[0];
+ if (!(command in this.dispatchTable_)) {
+ throw new Error('unknown command: ' + command);
+ }
+ var dispatch = this.dispatchTable_[command];
+
+ if (dispatch === null || this.skipDispatch(dispatch)) {
+ return;
+ }
+
+ // Parse fields.
+ var parsedFields = [];
+ for (var i = 0; i < dispatch.parsers.length; ++i) {
+ var parser = dispatch.parsers[i];
+ if (parser === null) {
+ parsedFields.push(fields[1 + i]);
+ } else if (typeof parser == 'function') {
+ parsedFields.push(parser(fields[1 + i]));
+ } else {
+ // var-args
+ parsedFields.push(fields.slice(1 + i));
+ break;
+ }
+ }
+
+ // Run the processor.
+ dispatch.processor.apply(this, parsedFields);
+};
+
+
+/**
+ * Decompresses a line if it was backreference-compressed.
+ *
+ * @param {string} line Possibly compressed line.
+ * @return {string} Decompressed line.
+ * @private
+ */
+devtools.profiler.LogReader.prototype.expandBackRef_ = function(line) {
+ var backRefPos;
+ // Filter out case when a regexp is created containing '#'.
+ if (line.charAt(line.length - 1) != '"'
+ && (backRefPos = line.lastIndexOf('#')) != -1) {
+ var backRef = line.substr(backRefPos + 1);
+ var backRefIdx = parseInt(backRef, 10) - 1;
+ var colonPos = backRef.indexOf(':');
+ var backRefStart =
+ colonPos != -1 ? parseInt(backRef.substr(colonPos + 1), 10) : 0;
+ line = line.substr(0, backRefPos) +
+ this.backRefs_[backRefIdx].substr(backRefStart);
+ }
+ this.backRefs_.unshift(line);
+ if (this.backRefs_.length > 10) {
+ this.backRefs_.length = 10;
+ }
+ return line;
+};
+
+
+/**
+ * Initializes the map of backward reference compressible commands.
+ * @private
+ */
+devtools.profiler.LogReader.prototype.initBackRefsCommands_ = function() {
+ for (var event in this.dispatchTable_) {
+ var dispatch = this.dispatchTable_[event];
+ if (dispatch && dispatch.backrefs) {
+ this.backRefsCommands_[event] = true;
+ }
+ }
+};
+
+
+/**
+ * Processes alias log record. Adds an alias to a corresponding map.
+ *
+ * @param {string} symbol Short name.
+ * @param {string} expansion Long name.
+ * @private
+ */
+devtools.profiler.LogReader.prototype.processAlias_ = function(
+ symbol, expansion) {
+ if (expansion in this.dispatchTable_) {
+ this.dispatchTable_[symbol] = this.dispatchTable_[expansion];
+ if (expansion in this.backRefsCommands_) {
+ this.backRefsCommands_[symbol] = true;
+ }
+ } else {
+ this.aliases_[symbol] = expansion;
+ }
+};
+
+
+/**
+ * Processes log lines.
+ *
+ * @param {Array.<string>} lines Log lines.
+ * @private
+ */
+devtools.profiler.LogReader.prototype.processLog_ = function(lines) {
+ var csvParser = new devtools.profiler.CsvParser();
+ try {
+ for (var i = 0, n = lines.length; i < n; ++i) {
+ var line = lines[i];
+ if (!line) {
+ continue;
+ }
+ if (line.charAt(0) == '#' ||
+ line.substr(0, line.indexOf(',')) in this.backRefsCommands_) {
+ line = this.expandBackRef_(line);
+ }
+ var fields = csvParser.parseLine(line);
+ this.dispatchLogRow_(fields);
+ }
+ } catch (e) {
+ // An error on the last line is acceptable since log file can be truncated.
+ if (i < n - 1) {
+ this.printError('line ' + (i + 1) + ': ' + (e.message || e));
+ throw e;
+ }
+ }
+};
+
+
+/**
+ * Processes repeat log record. Expands it according to calls count and
+ * invokes processing.
+ *
+ * @param {number} count Count.
+ * @param {Array.<string>} cmd Parsed command.
+ * @private
+ */
+devtools.profiler.LogReader.prototype.processRepeat_ = function(count, cmd) {
+ // Replace the repeat-prefixed command from backrefs list with a non-prefixed.
+ this.backRefs_[0] = cmd.join(',');
+ for (var i = 0; i < count; ++i) {
+ this.dispatchLogRow_(cmd);
+ }
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile.js
new file mode 100644
index 0000000..db4b542
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile.js
@@ -0,0 +1,621 @@
+// Copyright 2009 the V8 project authors. All rights reserved.
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following
+// disclaimer in the documentation and/or other materials provided
+// with the distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived
+// from this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+// Initlialize namespaces
+var devtools = devtools || {};
+devtools.profiler = devtools.profiler || {};
+
+
+/**
+ * Creates a profile object for processing profiling-related events
+ * and calculating function execution times.
+ *
+ * @constructor
+ */
+devtools.profiler.Profile = function() {
+ this.codeMap_ = new devtools.profiler.CodeMap();
+ this.topDownTree_ = new devtools.profiler.CallTree();
+ this.bottomUpTree_ = new devtools.profiler.CallTree();
+};
+
+
+/**
+ * Returns whether a function with the specified name must be skipped.
+ * Should be overriden by subclasses.
+ *
+ * @param {string} name Function name.
+ */
+devtools.profiler.Profile.prototype.skipThisFunction = function(name) {
+ return false;
+};
+
+
+/**
+ * Enum for profiler operations that involve looking up existing
+ * code entries.
+ *
+ * @enum {number}
+ */
+devtools.profiler.Profile.Operation = {
+ MOVE: 0,
+ DELETE: 1,
+ TICK: 2
+};
+
+
+/**
+ * Called whenever the specified operation has failed finding a function
+ * containing the specified address. Should be overriden by subclasses.
+ * See the devtools.profiler.Profile.Operation enum for the list of
+ * possible operations.
+ *
+ * @param {number} operation Operation.
+ * @param {number} addr Address of the unknown code.
+ * @param {number} opt_stackPos If an unknown address is encountered
+ * during stack strace processing, specifies a position of the frame
+ * containing the address.
+ */
+devtools.profiler.Profile.prototype.handleUnknownCode = function(
+ operation, addr, opt_stackPos) {
+};
+
+
+/**
+ * Registers a library.
+ *
+ * @param {string} name Code entry name.
+ * @param {number} startAddr Starting address.
+ * @param {number} endAddr Ending address.
+ */
+devtools.profiler.Profile.prototype.addLibrary = function(
+ name, startAddr, endAddr) {
+ var entry = new devtools.profiler.CodeMap.CodeEntry(
+ endAddr - startAddr, name);
+ this.codeMap_.addLibrary(startAddr, entry);
+ return entry;
+};
+
+
+/**
+ * Registers statically compiled code entry.
+ *
+ * @param {string} name Code entry name.
+ * @param {number} startAddr Starting address.
+ * @param {number} endAddr Ending address.
+ */
+devtools.profiler.Profile.prototype.addStaticCode = function(
+ name, startAddr, endAddr) {
+ var entry = new devtools.profiler.CodeMap.CodeEntry(
+ endAddr - startAddr, name);
+ this.codeMap_.addStaticCode(startAddr, entry);
+ return entry;
+};
+
+
+/**
+ * Registers dynamic (JIT-compiled) code entry.
+ *
+ * @param {string} type Code entry type.
+ * @param {string} name Code entry name.
+ * @param {number} start Starting address.
+ * @param {number} size Code entry size.
+ */
+devtools.profiler.Profile.prototype.addCode = function(
+ type, name, start, size) {
+ var entry = new devtools.profiler.Profile.DynamicCodeEntry(size, type, name);
+ this.codeMap_.addCode(start, entry);
+ return entry;
+};
+
+
+/**
+ * Reports about moving of a dynamic code entry.
+ *
+ * @param {number} from Current code entry address.
+ * @param {number} to New code entry address.
+ */
+devtools.profiler.Profile.prototype.moveCode = function(from, to) {
+ try {
+ this.codeMap_.moveCode(from, to);
+ } catch (e) {
+ this.handleUnknownCode(devtools.profiler.Profile.Operation.MOVE, from);
+ }
+};
+
+
+/**
+ * Reports about deletion of a dynamic code entry.
+ *
+ * @param {number} start Starting address.
+ */
+devtools.profiler.Profile.prototype.deleteCode = function(start) {
+ try {
+ this.codeMap_.deleteCode(start);
+ } catch (e) {
+ this.handleUnknownCode(devtools.profiler.Profile.Operation.DELETE, start);
+ }
+};
+
+
+/**
+ * Records a tick event. Stack must contain a sequence of
+ * addresses starting with the program counter value.
+ *
+ * @param {Array<number>} stack Stack sample.
+ */
+devtools.profiler.Profile.prototype.recordTick = function(stack) {
+ var processedStack = this.resolveAndFilterFuncs_(stack);
+ this.bottomUpTree_.addPath(processedStack);
+ processedStack.reverse();
+ this.topDownTree_.addPath(processedStack);
+};
+
+
+/**
+ * Translates addresses into function names and filters unneeded
+ * functions.
+ *
+ * @param {Array<number>} stack Stack sample.
+ */
+devtools.profiler.Profile.prototype.resolveAndFilterFuncs_ = function(stack) {
+ var result = [];
+ for (var i = 0; i < stack.length; ++i) {
+ var entry = this.codeMap_.findEntry(stack[i]);
+ if (entry) {
+ var name = entry.getName();
+ if (!this.skipThisFunction(name)) {
+ result.push(name);
+ }
+ } else {
+ this.handleUnknownCode(
+ devtools.profiler.Profile.Operation.TICK, stack[i], i);
+ }
+ }
+ return result;
+};
+
+
+/**
+ * Performs a BF traversal of the top down call graph.
+ *
+ * @param {function(devtools.profiler.CallTree.Node)} f Visitor function.
+ */
+devtools.profiler.Profile.prototype.traverseTopDownTree = function(f) {
+ this.topDownTree_.traverse(f);
+};
+
+
+/**
+ * Performs a BF traversal of the bottom up call graph.
+ *
+ * @param {function(devtools.profiler.CallTree.Node)} f Visitor function.
+ */
+devtools.profiler.Profile.prototype.traverseBottomUpTree = function(f) {
+ this.bottomUpTree_.traverse(f);
+};
+
+
+/**
+ * Calculates a top down profile for a node with the specified label.
+ * If no name specified, returns the whole top down calls tree.
+ *
+ * @param {string} opt_label Node label.
+ */
+devtools.profiler.Profile.prototype.getTopDownProfile = function(opt_label) {
+ return this.getTreeProfile_(this.topDownTree_, opt_label);
+};
+
+
+/**
+ * Calculates a bottom up profile for a node with the specified label.
+ * If no name specified, returns the whole bottom up calls tree.
+ *
+ * @param {string} opt_label Node label.
+ */
+devtools.profiler.Profile.prototype.getBottomUpProfile = function(opt_label) {
+ return this.getTreeProfile_(this.bottomUpTree_, opt_label);
+};
+
+
+/**
+ * Helper function for calculating a tree profile.
+ *
+ * @param {devtools.profiler.Profile.CallTree} tree Call tree.
+ * @param {string} opt_label Node label.
+ */
+devtools.profiler.Profile.prototype.getTreeProfile_ = function(tree, opt_label) {
+ if (!opt_label) {
+ tree.computeTotalWeights();
+ return tree;
+ } else {
+ var subTree = tree.cloneSubtree(opt_label);
+ subTree.computeTotalWeights();
+ return subTree;
+ }
+};
+
+
+/**
+ * Calculates a flat profile of callees starting from a node with
+ * the specified label. If no name specified, starts from the root.
+ *
+ * @param {string} opt_label Starting node label.
+ */
+devtools.profiler.Profile.prototype.getFlatProfile = function(opt_label) {
+ var counters = new devtools.profiler.CallTree();
+ var rootLabel = opt_label || devtools.profiler.CallTree.ROOT_NODE_LABEL;
+ var precs = {};
+ precs[rootLabel] = 0;
+ var root = counters.findOrAddChild(rootLabel);
+
+ this.topDownTree_.computeTotalWeights();
+ this.topDownTree_.traverseInDepth(
+ function onEnter(node) {
+ if (!(node.label in precs)) {
+ precs[node.label] = 0;
+ }
+ var nodeLabelIsRootLabel = node.label == rootLabel;
+ if (nodeLabelIsRootLabel || precs[rootLabel] > 0) {
+ if (precs[rootLabel] == 0) {
+ root.selfWeight += node.selfWeight;
+ root.totalWeight += node.totalWeight;
+ } else {
+ var rec = root.findOrAddChild(node.label);
+ rec.selfWeight += node.selfWeight;
+ if (nodeLabelIsRootLabel || precs[node.label] == 0) {
+ rec.totalWeight += node.totalWeight;
+ }
+ }
+ precs[node.label]++;
+ }
+ },
+ function onExit(node) {
+ if (node.label == rootLabel || precs[rootLabel] > 0) {
+ precs[node.label]--;
+ }
+ },
+ null);
+
+ if (!opt_label) {
+ // If we have created a flat profile for the whole program, we don't
+ // need an explicit root in it. Thus, replace the counters tree
+ // root with the node corresponding to the whole program.
+ counters.root_ = root;
+ } else {
+ // Propagate weights so percents can be calculated correctly.
+ counters.getRoot().selfWeight = root.selfWeight;
+ counters.getRoot().totalWeight = root.totalWeight;
+ }
+ return counters;
+};
+
+
+/**
+ * Creates a dynamic code entry.
+ *
+ * @param {number} size Code size.
+ * @param {string} type Code type.
+ * @param {string} name Function name.
+ * @constructor
+ */
+devtools.profiler.Profile.DynamicCodeEntry = function(size, type, name) {
+ devtools.profiler.CodeMap.CodeEntry.call(this, size, name);
+ this.type = type;
+};
+
+
+/**
+ * Returns node name.
+ */
+devtools.profiler.Profile.DynamicCodeEntry.prototype.getName = function() {
+ var name = this.name;
+ if (name.length == 0) {
+ name = '<anonymous>';
+ } else if (name.charAt(0) == ' ') {
+ // An anonymous function with location: " aaa.js:10".
+ name = '<anonymous>' + name;
+ }
+ return this.type + ': ' + name;
+};
+
+
+/**
+ * Constructs a call graph.
+ *
+ * @constructor
+ */
+devtools.profiler.CallTree = function() {
+ this.root_ = new devtools.profiler.CallTree.Node(
+ devtools.profiler.CallTree.ROOT_NODE_LABEL);
+};
+
+
+/**
+ * The label of the root node.
+ */
+devtools.profiler.CallTree.ROOT_NODE_LABEL = '';
+
+
+/**
+ * @private
+ */
+devtools.profiler.CallTree.prototype.totalsComputed_ = false;
+
+
+/**
+ * Returns the tree root.
+ */
+devtools.profiler.CallTree.prototype.getRoot = function() {
+ return this.root_;
+};
+
+
+/**
+ * Adds the specified call path, constructing nodes as necessary.
+ *
+ * @param {Array<string>} path Call path.
+ */
+devtools.profiler.CallTree.prototype.addPath = function(path) {
+ if (path.length == 0) {
+ return;
+ }
+ var curr = this.root_;
+ for (var i = 0; i < path.length; ++i) {
+ curr = curr.findOrAddChild(path[i]);
+ }
+ curr.selfWeight++;
+ this.totalsComputed_ = false;
+};
+
+
+/**
+ * Finds an immediate child of the specified parent with the specified
+ * label, creates a child node if necessary. If a parent node isn't
+ * specified, uses tree root.
+ *
+ * @param {string} label Child node label.
+ */
+devtools.profiler.CallTree.prototype.findOrAddChild = function(label) {
+ return this.root_.findOrAddChild(label);
+};
+
+
+/**
+ * Creates a subtree by cloning and merging all subtrees rooted at nodes
+ * with a given label. E.g. cloning the following call tree on label 'A'
+ * will give the following result:
+ *
+ * <A>--<B> <B>
+ * / /
+ * <root> == clone on 'A' ==> <root>--<A>
+ * \ \
+ * <C>--<A>--<D> <D>
+ *
+ * And <A>'s selfWeight will be the sum of selfWeights of <A>'s from the
+ * source call tree.
+ *
+ * @param {string} label The label of the new root node.
+ */
+devtools.profiler.CallTree.prototype.cloneSubtree = function(label) {
+ var subTree = new devtools.profiler.CallTree();
+ this.traverse(function(node, parent) {
+ if (!parent && node.label != label) {
+ return null;
+ }
+ var child = (parent ? parent : subTree).findOrAddChild(node.label);
+ child.selfWeight += node.selfWeight;
+ return child;
+ });
+ return subTree;
+};
+
+
+/**
+ * Computes total weights in the call graph.
+ */
+devtools.profiler.CallTree.prototype.computeTotalWeights = function() {
+ if (this.totalsComputed_) {
+ return;
+ }
+ this.root_.computeTotalWeight();
+ this.totalsComputed_ = true;
+};
+
+
+/**
+ * Traverses the call graph in preorder. This function can be used for
+ * building optionally modified tree clones. This is the boilerplate code
+ * for this scenario:
+ *
+ * callTree.traverse(function(node, parentClone) {
+ * var nodeClone = cloneNode(node);
+ * if (parentClone)
+ * parentClone.addChild(nodeClone);
+ * return nodeClone;
+ * });
+ *
+ * @param {function(devtools.profiler.CallTree.Node, *)} f Visitor function.
+ * The second parameter is the result of calling 'f' on the parent node.
+ */
+devtools.profiler.CallTree.prototype.traverse = function(f) {
+ var pairsToProcess = new ConsArray();
+ pairsToProcess.concat([{node: this.root_, param: null}]);
+ while (!pairsToProcess.atEnd()) {
+ var pair = pairsToProcess.next();
+ var node = pair.node;
+ var newParam = f(node, pair.param);
+ var morePairsToProcess = [];
+ node.forEachChild(function (child) {
+ morePairsToProcess.push({node: child, param: newParam}); });
+ pairsToProcess.concat(morePairsToProcess);
+ }
+};
+
+
+/**
+ * Performs an indepth call graph traversal.
+ *
+ * @param {function(devtools.profiler.CallTree.Node)} enter A function called
+ * prior to visiting node's children.
+ * @param {function(devtools.profiler.CallTree.Node)} exit A function called
+ * after visiting node's children.
+ */
+devtools.profiler.CallTree.prototype.traverseInDepth = function(enter, exit) {
+ function traverse(node) {
+ enter(node);
+ node.forEachChild(traverse);
+ exit(node);
+ }
+ traverse(this.root_);
+};
+
+
+/**
+ * Constructs a call graph node.
+ *
+ * @param {string} label Node label.
+ * @param {devtools.profiler.CallTree.Node} opt_parent Node parent.
+ */
+devtools.profiler.CallTree.Node = function(label, opt_parent) {
+ this.label = label;
+ this.parent = opt_parent;
+ this.children = {};
+};
+
+
+/**
+ * Node self weight (how many times this node was the last node in
+ * a call path).
+ * @type {number}
+ */
+devtools.profiler.CallTree.Node.prototype.selfWeight = 0;
+
+
+/**
+ * Node total weight (includes weights of all children).
+ * @type {number}
+ */
+devtools.profiler.CallTree.Node.prototype.totalWeight = 0;
+
+
+/**
+ * Adds a child node.
+ *
+ * @param {string} label Child node label.
+ */
+devtools.profiler.CallTree.Node.prototype.addChild = function(label) {
+ var child = new devtools.profiler.CallTree.Node(label, this);
+ this.children[label] = child;
+ return child;
+};
+
+
+/**
+ * Computes node's total weight.
+ */
+devtools.profiler.CallTree.Node.prototype.computeTotalWeight =
+ function() {
+ var totalWeight = this.selfWeight;
+ this.forEachChild(function(child) {
+ totalWeight += child.computeTotalWeight(); });
+ return this.totalWeight = totalWeight;
+};
+
+
+/**
+ * Returns all node's children as an array.
+ */
+devtools.profiler.CallTree.Node.prototype.exportChildren = function() {
+ var result = [];
+ this.forEachChild(function (node) { result.push(node); });
+ return result;
+};
+
+
+/**
+ * Finds an immediate child with the specified label.
+ *
+ * @param {string} label Child node label.
+ */
+devtools.profiler.CallTree.Node.prototype.findChild = function(label) {
+ return this.children[label] || null;
+};
+
+
+/**
+ * Finds an immediate child with the specified label, creates a child
+ * node if necessary.
+ *
+ * @param {string} label Child node label.
+ */
+devtools.profiler.CallTree.Node.prototype.findOrAddChild = function(label) {
+ return this.findChild(label) || this.addChild(label);
+};
+
+
+/**
+ * Calls the specified function for every child.
+ *
+ * @param {function(devtools.profiler.CallTree.Node)} f Visitor function.
+ */
+devtools.profiler.CallTree.Node.prototype.forEachChild = function(f) {
+ for (var c in this.children) {
+ f(this.children[c]);
+ }
+};
+
+
+/**
+ * Walks up from the current node up to the call tree root.
+ *
+ * @param {function(devtools.profiler.CallTree.Node)} f Visitor function.
+ */
+devtools.profiler.CallTree.Node.prototype.walkUpToRoot = function(f) {
+ for (var curr = this; curr != null; curr = curr.parent) {
+ f(curr);
+ }
+};
+
+
+/**
+ * Tries to find a node with the specified path.
+ *
+ * @param {Array<string>} labels The path.
+ * @param {function(devtools.profiler.CallTree.Node)} opt_f Visitor function.
+ */
+devtools.profiler.CallTree.Node.prototype.descendToChild = function(
+ labels, opt_f) {
+ for (var pos = 0, curr = this; pos < labels.length && curr != null; pos++) {
+ var child = curr.findChild(labels[pos]);
+ if (opt_f) {
+ opt_f(child, pos);
+ }
+ curr = child;
+ }
+ return curr;
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile_view.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile_view.js
new file mode 100644
index 0000000..bdea631
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profile_view.js
@@ -0,0 +1,224 @@
+// Copyright 2009 the V8 project authors. All rights reserved.
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following
+// disclaimer in the documentation and/or other materials provided
+// with the distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived
+// from this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+// Initlialize namespaces
+var devtools = devtools || {};
+devtools.profiler = devtools.profiler || {};
+
+
+/**
+ * Creates a Profile View builder object.
+ *
+ * @param {number} samplingRate Number of ms between profiler ticks.
+ * @constructor
+ */
+devtools.profiler.ViewBuilder = function(samplingRate) {
+ this.samplingRate = samplingRate;
+};
+
+
+/**
+ * Builds a profile view for the specified call tree.
+ *
+ * @param {devtools.profiler.CallTree} callTree A call tree.
+ * @param {boolean} opt_bottomUpViewWeights Whether remapping
+ * of self weights for a bottom up view is needed.
+ */
+devtools.profiler.ViewBuilder.prototype.buildView = function(
+ callTree, opt_bottomUpViewWeights) {
+ var head;
+ var samplingRate = this.samplingRate;
+ var createViewNode = this.createViewNode;
+ callTree.traverse(function(node, viewParent) {
+ var totalWeight = node.totalWeight * samplingRate;
+ var selfWeight = node.selfWeight * samplingRate;
+ if (opt_bottomUpViewWeights === true) {
+ if (viewParent === head) {
+ selfWeight = totalWeight;
+ } else {
+ selfWeight = 0;
+ }
+ }
+ var viewNode = createViewNode(node.label, totalWeight, selfWeight, head);
+ if (viewParent) {
+ viewParent.addChild(viewNode);
+ } else {
+ head = viewNode;
+ }
+ return viewNode;
+ });
+ var view = this.createView(head);
+ return view;
+};
+
+
+/**
+ * Factory method for a profile view.
+ *
+ * @param {devtools.profiler.ProfileView.Node} head View head node.
+ * @return {devtools.profiler.ProfileView} Profile view.
+ */
+devtools.profiler.ViewBuilder.prototype.createView = function(head) {
+ return new devtools.profiler.ProfileView(head);
+};
+
+
+/**
+ * Factory method for a profile view node.
+ *
+ * @param {string} internalFuncName A fully qualified function name.
+ * @param {number} totalTime Amount of time that application spent in the
+ * corresponding function and its descendants (not that depending on
+ * profile they can be either callees or callers.)
+ * @param {number} selfTime Amount of time that application spent in the
+ * corresponding function only.
+ * @param {devtools.profiler.ProfileView.Node} head Profile view head.
+ * @return {devtools.profiler.ProfileView.Node} Profile view node.
+ */
+devtools.profiler.ViewBuilder.prototype.createViewNode = function(
+ funcName, totalTime, selfTime, head) {
+ return new devtools.profiler.ProfileView.Node(
+ funcName, totalTime, selfTime, head);
+};
+
+
+/**
+ * Creates a Profile View object. It allows to perform sorting
+ * and filtering actions on the profile.
+ *
+ * @param {devtools.profiler.ProfileView.Node} head Head (root) node.
+ * @constructor
+ */
+devtools.profiler.ProfileView = function(head) {
+ this.head = head;
+};
+
+
+/**
+ * Sorts the profile view using the specified sort function.
+ *
+ * @param {function(devtools.profiler.ProfileView.Node,
+ * devtools.profiler.ProfileView.Node):number} sortFunc A sorting
+ * functions. Must comply with Array.sort sorting function requirements.
+ */
+devtools.profiler.ProfileView.prototype.sort = function(sortFunc) {
+ this.traverse(function (node) {
+ node.sortChildren(sortFunc);
+ });
+};
+
+
+/**
+ * Traverses profile view nodes in preorder.
+ *
+ * @param {function(devtools.profiler.ProfileView.Node)} f Visitor function.
+ */
+devtools.profiler.ProfileView.prototype.traverse = function(f) {
+ var nodesToTraverse = new ConsArray();
+ nodesToTraverse.concat([this.head]);
+ while (!nodesToTraverse.atEnd()) {
+ var node = nodesToTraverse.next();
+ f(node);
+ nodesToTraverse.concat(node.children);
+ }
+};
+
+
+/**
+ * Constructs a Profile View node object. Each node object corresponds to
+ * a function call.
+ *
+ * @param {string} internalFuncName A fully qualified function name.
+ * @param {number} totalTime Amount of time that application spent in the
+ * corresponding function and its descendants (not that depending on
+ * profile they can be either callees or callers.)
+ * @param {number} selfTime Amount of time that application spent in the
+ * corresponding function only.
+ * @param {devtools.profiler.ProfileView.Node} head Profile view head.
+ * @constructor
+ */
+devtools.profiler.ProfileView.Node = function(
+ internalFuncName, totalTime, selfTime, head) {
+ this.internalFuncName = internalFuncName;
+ this.totalTime = totalTime;
+ this.selfTime = selfTime;
+ this.head = head;
+ this.parent = null;
+ this.children = [];
+};
+
+
+/**
+ * Returns a share of the function's total time in application's total time.
+ */
+devtools.profiler.ProfileView.Node.prototype.__defineGetter__(
+ 'totalPercent',
+ function() { return this.totalTime /
+ (this.head ? this.head.totalTime : this.totalTime) * 100.0; });
+
+
+/**
+ * Returns a share of the function's self time in application's total time.
+ */
+devtools.profiler.ProfileView.Node.prototype.__defineGetter__(
+ 'selfPercent',
+ function() { return this.selfTime /
+ (this.head ? this.head.totalTime : this.totalTime) * 100.0; });
+
+
+/**
+ * Returns a share of the function's total time in its parent's total time.
+ */
+devtools.profiler.ProfileView.Node.prototype.__defineGetter__(
+ 'parentTotalPercent',
+ function() { return this.totalTime /
+ (this.parent ? this.parent.totalTime : this.totalTime) * 100.0; });
+
+
+/**
+ * Adds a child to the node.
+ *
+ * @param {devtools.profiler.ProfileView.Node} node Child node.
+ */
+devtools.profiler.ProfileView.Node.prototype.addChild = function(node) {
+ node.parent = this;
+ this.children.push(node);
+};
+
+
+/**
+ * Sorts all the node's children recursively.
+ *
+ * @param {function(devtools.profiler.ProfileView.Node,
+ * devtools.profiler.ProfileView.Node):number} sortFunc A sorting
+ * functions. Must comply with Array.sort sorting function requirements.
+ */
+devtools.profiler.ProfileView.Node.prototype.sortChildren = function(
+ sortFunc) {
+ this.children.sort(sortFunc);
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profiler_processor.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profiler_processor.js
new file mode 100644
index 0000000..13aeee7
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/profiler_processor.js
@@ -0,0 +1,449 @@
+// Copyright (c) 2009 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.
+
+/**
+ * @fileoverview Profiler processor is used to process log file produced
+ * by V8 and produce an internal profile representation which is used
+ * for building profile views in 'Profiles' tab.
+ */
+goog.provide('devtools.profiler.Processor');
+
+
+/**
+ * Creates a Profile View builder object compatible with WebKit Profiler UI.
+ *
+ * @param {number} samplingRate Number of ms between profiler ticks.
+ * @constructor
+ */
+devtools.profiler.WebKitViewBuilder = function(samplingRate) {
+ devtools.profiler.ViewBuilder.call(this, samplingRate);
+};
+goog.inherits(devtools.profiler.WebKitViewBuilder,
+ devtools.profiler.ViewBuilder);
+
+
+/**
+ * @override
+ */
+devtools.profiler.WebKitViewBuilder.prototype.createViewNode = function(
+ funcName, totalTime, selfTime, head) {
+ return new devtools.profiler.WebKitViewNode(
+ funcName, totalTime, selfTime, head);
+};
+
+
+/**
+ * Constructs a Profile View node object for displaying in WebKit Profiler UI.
+ *
+ * @param {string} internalFuncName A fully qualified function name.
+ * @param {number} totalTime Amount of time that application spent in the
+ * corresponding function and its descendants (not that depending on
+ * profile they can be either callees or callers.)
+ * @param {number} selfTime Amount of time that application spent in the
+ * corresponding function only.
+ * @param {devtools.profiler.ProfileView.Node} head Profile view head.
+ * @constructor
+ */
+devtools.profiler.WebKitViewNode = function(
+ internalFuncName, totalTime, selfTime, head) {
+ devtools.profiler.ProfileView.Node.call(this,
+ internalFuncName, totalTime, selfTime, head);
+ this.initFuncInfo_();
+ this.callUID = internalFuncName;
+};
+goog.inherits(devtools.profiler.WebKitViewNode,
+ devtools.profiler.ProfileView.Node);
+
+
+/**
+ * RegEx for stripping V8's prefixes of compiled functions.
+ */
+devtools.profiler.WebKitViewNode.FUNC_NAME_STRIP_RE =
+ /^(?:LazyCompile|Function): (.*)$/;
+
+
+/**
+ * RegEx for extracting script source URL and line number.
+ */
+devtools.profiler.WebKitViewNode.FUNC_NAME_PARSE_RE =
+ /^([^ ]+) (.*):(\d+)( \{\d+\})?$/;
+
+
+/**
+ * Inits 'functionName', 'url', and 'lineNumber' fields using 'internalFuncName'
+ * field.
+ * @private
+ */
+devtools.profiler.WebKitViewNode.prototype.initFuncInfo_ = function() {
+ var nodeAlias = devtools.profiler.WebKitViewNode;
+ this.functionName = this.internalFuncName;
+
+ var strippedName = nodeAlias.FUNC_NAME_STRIP_RE.exec(this.functionName);
+ if (strippedName) {
+ this.functionName = strippedName[1];
+ }
+
+ var parsedName = nodeAlias.FUNC_NAME_PARSE_RE.exec(this.functionName);
+ if (parsedName) {
+ this.functionName = parsedName[1];
+ if (parsedName[4]) {
+ this.functionName += parsedName[4];
+ }
+ this.url = parsedName[2];
+ this.lineNumber = parsedName[3];
+ } else {
+ this.url = '';
+ this.lineNumber = 0;
+ }
+};
+
+
+/**
+ * Ancestor of a profile object that leaves out only JS-related functions.
+ * @constructor
+ */
+devtools.profiler.JsProfile = function() {
+ devtools.profiler.Profile.call(this);
+};
+goog.inherits(devtools.profiler.JsProfile, devtools.profiler.Profile);
+
+
+/**
+ * RegExp that leaves only JS functions.
+ * @type {RegExp}
+ */
+devtools.profiler.JsProfile.JS_FUNC_RE = /^(LazyCompile|Function|Script):/;
+
+/**
+ * RegExp that filters out native code (ending with "native src.js:xxx").
+ * @type {RegExp}
+ */
+devtools.profiler.JsProfile.JS_NATIVE_FUNC_RE = /\ native\ \w+\.js:\d+$/;
+
+/**
+ * RegExp that filters out native scripts.
+ * @type {RegExp}
+ */
+devtools.profiler.JsProfile.JS_NATIVE_SCRIPT_RE = /^Script:\ native/;
+
+/**
+ * RegExp that filters out devtools functions. See inject.js and
+ * inject_dispatch.js.
+ * @type {RegExp}
+ */
+devtools.profiler.JsProfile.JS_DEVTOOLS_FUNC_RE =
+ /^\w+:\ devtools(\$\$|\.Injected)/;
+
+
+/**
+ * @override
+ */
+devtools.profiler.JsProfile.prototype.skipThisFunction = function(name) {
+ return !devtools.profiler.JsProfile.JS_FUNC_RE.test(name) ||
+ // To profile V8's natives comment out two lines below and '||' above.
+ devtools.profiler.JsProfile.JS_NATIVE_FUNC_RE.test(name) ||
+ devtools.profiler.JsProfile.JS_NATIVE_SCRIPT_RE.test(name) ||
+ devtools.profiler.JsProfile.JS_DEVTOOLS_FUNC_RE.test(name);
+};
+
+
+/**
+ * Profiler processor. Consumes profiler log and builds profile views.
+ *
+ * @param {function(devtools.profiler.ProfileView)} newProfileCallback Callback
+ * that receives a new processed profile.
+ * @constructor
+ */
+devtools.profiler.Processor = function() {
+ devtools.profiler.LogReader.call(this, {
+ 'code-creation': {
+ parsers: [null, this.createAddressParser('code'), parseInt, null],
+ processor: this.processCodeCreation_, backrefs: true,
+ needsProfile: true },
+ 'code-move': { parsers: [this.createAddressParser('code'),
+ this.createAddressParser('code-move-to')],
+ processor: this.processCodeMove_, backrefs: true,
+ needsProfile: true },
+ 'code-delete': { parsers: [this.createAddressParser('code')],
+ processor: this.processCodeDelete_, backrefs: true,
+ needsProfile: true },
+ 'tick': { parsers: [this.createAddressParser('code'),
+ this.createAddressParser('stack'), parseInt, 'var-args'],
+ processor: this.processTick_, backrefs: true, needProfile: true },
+ 'profiler': { parsers: [null, 'var-args'],
+ processor: this.processProfiler_, needsProfile: false },
+ 'heap-sample-begin': { parsers: [null, null, parseInt],
+ processor: this.processHeapSampleBegin_ },
+ 'heap-sample-stats': { parsers: [null, null, parseInt, parseInt],
+ processor: this.processHeapSampleStats_ },
+ 'heap-sample-item': { parsers: [null, parseInt, parseInt],
+ processor: this.processHeapSampleItem_ },
+ 'heap-js-cons-item': { parsers: [null, parseInt, parseInt],
+ processor: this.processHeapJsConsItem_ },
+ 'heap-sample-end': { parsers: [null, null],
+ processor: this.processHeapSampleEnd_ },
+ // Not used in DevTools Profiler.
+ 'shared-library': null,
+ 'heap-js-ret-item': null,
+ // Obsolete row types.
+ 'code-allocate': null,
+ 'begin-code-region': null,
+ 'end-code-region': null});
+
+
+ /**
+ * Callback that is called when a new profile is encountered in the log.
+ * @type {function()}
+ */
+ this.startedProfileProcessing_ = null;
+
+ /**
+ * Callback that is called periodically to display processing status.
+ * @type {function()}
+ */
+ this.profileProcessingStatus_ = null;
+
+ /**
+ * Callback that is called when a profile has been processed and is ready
+ * to be shown.
+ * @type {function(devtools.profiler.ProfileView)}
+ */
+ this.finishedProfileProcessing_ = null;
+
+ /**
+ * The current profile.
+ * @type {devtools.profiler.JsProfile}
+ */
+ this.currentProfile_ = null;
+
+ /**
+ * Builder of profile views. Created during "profiler,begin" event processing.
+ * @type {devtools.profiler.WebKitViewBuilder}
+ */
+ this.viewBuilder_ = null;
+
+ /**
+ * Next profile id.
+ * @type {number}
+ */
+ this.profileId_ = 1;
+
+ /**
+ * Counter for processed ticks.
+ * @type {number}
+ */
+ this.ticksCount_ = 0;
+
+ /**
+ * The current heap snapshot.
+ * @type {string}
+ */
+ this.currentHeapSnapshot_ = null;
+
+ /**
+ * Next heap snapshot id.
+ * @type {number}
+ */
+ this.heapSnapshotId_ = 1;
+};
+goog.inherits(devtools.profiler.Processor, devtools.profiler.LogReader);
+
+
+/**
+ * @override
+ */
+devtools.profiler.Processor.prototype.printError = function(str) {
+ debugPrint(str);
+};
+
+
+/**
+ * @override
+ */
+devtools.profiler.Processor.prototype.skipDispatch = function(dispatch) {
+ return dispatch.needsProfile && this.currentProfile_ == null;
+};
+
+
+/**
+ * Sets profile processing callbacks.
+ *
+ * @param {function()} started Started processing callback.
+ * @param {function(devtools.profiler.ProfileView)} finished Finished
+ * processing callback.
+ */
+devtools.profiler.Processor.prototype.setCallbacks = function(
+ started, processing, finished) {
+ this.startedProfileProcessing_ = started;
+ this.profileProcessingStatus_ = processing;
+ this.finishedProfileProcessing_ = finished;
+};
+
+
+/**
+ * An address for the fake "(program)" entry. WebKit's visualisation
+ * has assumptions on how the top of the call tree should look like,
+ * and we need to add a fake entry as the topmost function. This
+ * address is chosen because it's the end address of the first memory
+ * page, which is never used for code or data, but only as a guard
+ * page for catching AV errors.
+ *
+ * @type {number}
+ */
+devtools.profiler.Processor.PROGRAM_ENTRY = 0xffff;
+/**
+ * @type {string}
+ */
+devtools.profiler.Processor.PROGRAM_ENTRY_STR = '0xffff';
+
+
+/**
+ * Sets new profile callback.
+ * @param {function(devtools.profiler.ProfileView)} callback Callback function.
+ */
+devtools.profiler.Processor.prototype.setNewProfileCallback = function(
+ callback) {
+ this.newProfileCallback_ = callback;
+};
+
+
+devtools.profiler.Processor.prototype.processProfiler_ = function(
+ state, params) {
+ var processingInterval = null;
+ switch (state) {
+ case 'resume':
+ if (this.currentProfile_ == null) {
+ this.currentProfile_ = new devtools.profiler.JsProfile();
+ // see the comment for devtools.profiler.Processor.PROGRAM_ENTRY
+ this.currentProfile_.addCode(
+ 'Function', '(program)',
+ devtools.profiler.Processor.PROGRAM_ENTRY, 1);
+ if (this.startedProfileProcessing_) {
+ this.startedProfileProcessing_();
+ }
+ this.ticksCount_ = 0;
+ var self = this;
+ if (this.profileProcessingStatus_) {
+ processingInterval = window.setInterval(
+ function() { self.profileProcessingStatus_(self.ticksCount_); },
+ 1000);
+ }
+ }
+ break;
+ case 'pause':
+ if (this.currentProfile_ != null) {
+ window.clearInterval(processingInterval);
+ if (this.finishedProfileProcessing_) {
+ this.finishedProfileProcessing_(this.createProfileForView());
+ }
+ this.currentProfile_ = null;
+ }
+ break;
+ case 'begin':
+ var samplingRate = NaN;
+ if (params.length > 0) {
+ samplingRate = parseInt(params[0]);
+ }
+ if (isNaN(samplingRate)) {
+ samplingRate = 1;
+ }
+ this.viewBuilder_ = new devtools.profiler.WebKitViewBuilder(samplingRate);
+ break;
+ // These events are valid but aren't used.
+ case 'compression':
+ case 'end': break;
+ default:
+ throw new Error('unknown profiler state: ' + state);
+ }
+};
+
+
+devtools.profiler.Processor.prototype.processCodeCreation_ = function(
+ type, start, size, name) {
+ this.currentProfile_.addCode(this.expandAlias(type), name, start, size);
+};
+
+
+devtools.profiler.Processor.prototype.processCodeMove_ = function(from, to) {
+ this.currentProfile_.moveCode(from, to);
+};
+
+
+devtools.profiler.Processor.prototype.processCodeDelete_ = function(start) {
+ this.currentProfile_.deleteCode(start);
+};
+
+
+devtools.profiler.Processor.prototype.processTick_ = function(
+ pc, sp, vmState, stack) {
+ // see the comment for devtools.profiler.Processor.PROGRAM_ENTRY
+ stack.push(devtools.profiler.Processor.PROGRAM_ENTRY_STR);
+ this.currentProfile_.recordTick(this.processStack(pc, stack));
+ this.ticksCount_++;
+};
+
+
+devtools.profiler.Processor.prototype.processHeapSampleBegin_ = function(
+ space, state, ticks) {
+ if (space != 'Heap') return;
+ this.currentHeapSnapshot_ = {
+ number: this.heapSnapshotId_++,
+ entries: {},
+ lowlevels: {},
+ ticks: ticks
+ };
+};
+
+
+devtools.profiler.Processor.prototype.processHeapSampleStats_ = function(
+ space, state, capacity, used) {
+ if (space != 'Heap') return;
+ this.currentHeapSnapshot_.capacity = capacity;
+ this.currentHeapSnapshot_.used = used;
+};
+
+
+devtools.profiler.Processor.prototype.processHeapSampleItem_ = function(
+ item, number, size) {
+ if (!this.currentHeapSnapshot_) return;
+ this.currentHeapSnapshot_.lowlevels[item] = {
+ type: item, count: number, size: size
+ };
+};
+
+
+devtools.profiler.Processor.prototype.processHeapJsConsItem_ = function(
+ item, number, size) {
+ if (!this.currentHeapSnapshot_) return;
+ this.currentHeapSnapshot_.entries[item] = {
+ cons: item, count: number, size: size
+ };
+};
+
+
+devtools.profiler.Processor.prototype.processHeapSampleEnd_ = function(
+ space, state) {
+ if (space != 'Heap') return;
+ var snapshot = this.currentHeapSnapshot_;
+ this.currentHeapSnapshot_ = null;
+ // For some reason, 'used' from 'heap-sample-stats' sometimes differ from
+ // the sum of objects sizes. To avoid discrepancy, we re-calculate 'used'.
+ snapshot.used = 0;
+ for (var item in snapshot.lowlevels) {
+ snapshot.used += snapshot.lowlevels[item].size;
+ }
+ WebInspector.panels.heap.addSnapshot(snapshot);
+};
+
+
+/**
+ * Creates a profile for further displaying in ProfileView.
+ */
+devtools.profiler.Processor.prototype.createProfileForView = function() {
+ var profile = this.viewBuilder_.buildView(
+ this.currentProfile_.getTopDownProfile());
+ profile.uid = this.profileId_++;
+ profile.title = UserInitiatedProfileName + '.' + profile.uid;
+ return profile;
+};
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/splaytree.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/splaytree.js
new file mode 100644
index 0000000..7b3af8b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/splaytree.js
@@ -0,0 +1,322 @@
+// Copyright 2009 the V8 project authors. All rights reserved.
+// Redistribution and use in source and binary forms, with or without
+// modification, are permitted provided that the following conditions are
+// met:
+//
+// * Redistributions of source code must retain the above copyright
+// notice, this list of conditions and the following disclaimer.
+// * Redistributions in binary form must reproduce the above
+// copyright notice, this list of conditions and the following
+// disclaimer in the documentation and/or other materials provided
+// with the distribution.
+// * Neither the name of Google Inc. nor the names of its
+// contributors may be used to endorse or promote products derived
+// from this software without specific prior written permission.
+//
+// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+// A namespace stub. It will become more clear how to declare it properly
+// during integration of this script into Dev Tools.
+var goog = goog || {};
+goog.structs = goog.structs || {};
+
+
+/**
+ * Constructs a Splay tree. A splay tree is a self-balancing binary
+ * search tree with the additional property that recently accessed
+ * elements are quick to access again. It performs basic operations
+ * such as insertion, look-up and removal in O(log(n)) amortized time.
+ *
+ * @constructor
+ */
+goog.structs.SplayTree = function() {
+};
+
+
+/**
+ * Pointer to the root node of the tree.
+ *
+ * @type {goog.structs.SplayTree.Node}
+ * @private
+ */
+goog.structs.SplayTree.prototype.root_ = null;
+
+
+/**
+ * @return {boolean} Whether the tree is empty.
+ */
+goog.structs.SplayTree.prototype.isEmpty = function() {
+ return !this.root_;
+};
+
+
+
+/**
+ * Inserts a node into the tree with the specified key and value if
+ * the tree does not already contain a node with the specified key. If
+ * the value is inserted, it becomes the root of the tree.
+ *
+ * @param {number} key Key to insert into the tree.
+ * @param {*} value Value to insert into the tree.
+ */
+goog.structs.SplayTree.prototype.insert = function(key, value) {
+ if (this.isEmpty()) {
+ this.root_ = new goog.structs.SplayTree.Node(key, value);
+ return;
+ }
+ // Splay on the key to move the last node on the search path for
+ // the key to the root of the tree.
+ this.splay_(key);
+ if (this.root_.key == key) {
+ return;
+ }
+ var node = new goog.structs.SplayTree.Node(key, value);
+ if (key > this.root_.key) {
+ node.left = this.root_;
+ node.right = this.root_.right;
+ this.root_.right = null;
+ } else {
+ node.right = this.root_;
+ node.left = this.root_.left;
+ this.root_.left = null;
+ }
+ this.root_ = node;
+};
+
+
+/**
+ * Removes a node with the specified key from the tree if the tree
+ * contains a node with this key. The removed node is returned. If the
+ * key is not found, an exception is thrown.
+ *
+ * @param {number} key Key to find and remove from the tree.
+ * @return {goog.structs.SplayTree.Node} The removed node.
+ */
+goog.structs.SplayTree.prototype.remove = function(key) {
+ if (this.isEmpty()) {
+ throw Error('Key not found: ' + key);
+ }
+ this.splay_(key);
+ if (this.root_.key != key) {
+ throw Error('Key not found: ' + key);
+ }
+ var removed = this.root_;
+ if (!this.root_.left) {
+ this.root_ = this.root_.right;
+ } else {
+ var right = this.root_.right;
+ this.root_ = this.root_.left;
+ // Splay to make sure that the new root has an empty right child.
+ this.splay_(key);
+ // Insert the original right child as the right child of the new
+ // root.
+ this.root_.right = right;
+ }
+ return removed;
+};
+
+
+/**
+ * Returns the node having the specified key or null if the tree doesn't contain
+ * a node with the specified key.
+ *
+ * @param {number} key Key to find in the tree.
+ * @return {goog.structs.SplayTree.Node} Node having the specified key.
+ */
+goog.structs.SplayTree.prototype.find = function(key) {
+ if (this.isEmpty()) {
+ return null;
+ }
+ this.splay_(key);
+ return this.root_.key == key ? this.root_ : null;
+};
+
+
+/**
+ * @return {goog.structs.SplayTree.Node} Node having the minimum key value.
+ */
+goog.structs.SplayTree.prototype.findMin = function() {
+ if (this.isEmpty()) {
+ return null;
+ }
+ var current = this.root_;
+ while (current.left) {
+ current = current.left;
+ }
+ return current;
+};
+
+
+/**
+ * @return {goog.structs.SplayTree.Node} Node having the maximum key value.
+ */
+goog.structs.SplayTree.prototype.findMax = function(opt_startNode) {
+ if (this.isEmpty()) {
+ return null;
+ }
+ var current = opt_startNode || this.root_;
+ while (current.right) {
+ current = current.right;
+ }
+ return current;
+};
+
+
+/**
+ * @return {goog.structs.SplayTree.Node} Node having the maximum key value that
+ * is less or equal to the specified key value.
+ */
+goog.structs.SplayTree.prototype.findGreatestLessThan = function(key) {
+ if (this.isEmpty()) {
+ return null;
+ }
+ // Splay on the key to move the node with the given key or the last
+ // node on the search path to the top of the tree.
+ this.splay_(key);
+ // Now the result is either the root node or the greatest node in
+ // the left subtree.
+ if (this.root_.key <= key) {
+ return this.root_;
+ } else if (this.root_.left) {
+ return this.findMax(this.root_.left);
+ } else {
+ return null;
+ }
+};
+
+
+/**
+ * @return {Array<*>} An array containing all the values of tree's nodes.
+ */
+goog.structs.SplayTree.prototype.exportValues = function() {
+ var result = [];
+ this.traverse_(function(node) { result.push(node.value); });
+ return result;
+};
+
+
+/**
+ * Perform the splay operation for the given key. Moves the node with
+ * the given key to the top of the tree. If no node has the given
+ * key, the last node on the search path is moved to the top of the
+ * tree. This is the simplified top-down splaying algorithm from:
+ * "Self-adjusting Binary Search Trees" by Sleator and Tarjan
+ *
+ * @param {number} key Key to splay the tree on.
+ * @private
+ */
+goog.structs.SplayTree.prototype.splay_ = function(key) {
+ if (this.isEmpty()) {
+ return;
+ }
+ // Create a dummy node. The use of the dummy node is a bit
+ // counter-intuitive: The right child of the dummy node will hold
+ // the L tree of the algorithm. The left child of the dummy node
+ // will hold the R tree of the algorithm. Using a dummy node, left
+ // and right will always be nodes and we avoid special cases.
+ var dummy, left, right;
+ dummy = left = right = new goog.structs.SplayTree.Node(null, null);
+ var current = this.root_;
+ while (true) {
+ if (key < current.key) {
+ if (!current.left) {
+ break;
+ }
+ if (key < current.left.key) {
+ // Rotate right.
+ var tmp = current.left;
+ current.left = tmp.right;
+ tmp.right = current;
+ current = tmp;
+ if (!current.left) {
+ break;
+ }
+ }
+ // Link right.
+ right.left = current;
+ right = current;
+ current = current.left;
+ } else if (key > current.key) {
+ if (!current.right) {
+ break;
+ }
+ if (key > current.right.key) {
+ // Rotate left.
+ var tmp = current.right;
+ current.right = tmp.left;
+ tmp.left = current;
+ current = tmp;
+ if (!current.right) {
+ break;
+ }
+ }
+ // Link left.
+ left.right = current;
+ left = current;
+ current = current.right;
+ } else {
+ break;
+ }
+ }
+ // Assemble.
+ left.right = current.left;
+ right.left = current.right;
+ current.left = dummy.right;
+ current.right = dummy.left;
+ this.root_ = current;
+};
+
+
+/**
+ * Performs a preorder traversal of the tree.
+ *
+ * @param {function(goog.structs.SplayTree.Node)} f Visitor function.
+ * @private
+ */
+goog.structs.SplayTree.prototype.traverse_ = function(f) {
+ var nodesToVisit = [this.root_];
+ while (nodesToVisit.length > 0) {
+ var node = nodesToVisit.shift();
+ if (node == null) {
+ continue;
+ }
+ f(node);
+ nodesToVisit.push(node.left);
+ nodesToVisit.push(node.right);
+ }
+};
+
+
+/**
+ * Constructs a Splay tree node.
+ *
+ * @param {number} key Key.
+ * @param {*} value Value.
+ */
+goog.structs.SplayTree.Node = function(key, value) {
+ this.key = key;
+ this.value = value;
+};
+
+
+/**
+ * @type {goog.structs.SplayTree.Node}
+ */
+goog.structs.SplayTree.Node.prototype.left = null;
+
+
+/**
+ * @type {goog.structs.SplayTree.Node}
+ */
+goog.structs.SplayTree.Node.prototype.right = null;
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/tests.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/tests.js
new file mode 100644
index 0000000..5691017
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/tests.js
@@ -0,0 +1,628 @@
+// Copyright (c) 2009 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.
+
+
+/**
+ * @fileoverview This file contains small testing framework along with the
+ * test suite for the frontend. These tests are a part of the continues build
+ * and are executed by the devtools_sanity_unittest.cc as a part of the
+ * Interactive UI Test suite.
+ */
+
+if (window.domAutomationController) {
+
+var ___interactiveUiTestsMode = true;
+
+/**
+ * Test suite for interactive UI tests.
+ * @constructor
+ */
+TestSuite = function() {
+ this.controlTaken_ = false;
+ this.timerId_ = -1;
+};
+
+
+/**
+ * Reports test failure.
+ * @param {string} message Failure description.
+ */
+TestSuite.prototype.fail = function(message) {
+ if (this.controlTaken_) {
+ this.reportFailure_(message);
+ } else {
+ throw message;
+ }
+};
+
+
+/**
+ * Equals assertion tests that expected == actual.
+ * @param {Object} expected Expected object.
+ * @param {Object} actual Actual object.
+ * @param {string} opt_message User message to print if the test fails.
+ */
+TestSuite.prototype.assertEquals = function(expected, actual, opt_message) {
+ if (expected != actual) {
+ var message = 'Expected: "' + expected + '", but was "' + actual + '"';
+ if (opt_message) {
+ message = opt_message + '(' + message + ')';
+ }
+ this.fail(message);
+ }
+};
+
+
+/**
+ * True assertion tests that value == true.
+ * @param {Object} value Actual object.
+ * @param {string} opt_message User message to print if the test fails.
+ */
+TestSuite.prototype.assertTrue = function(value, opt_message) {
+ this.assertEquals(true, value, opt_message);
+};
+
+
+/**
+ * Contains assertion tests that string contains substring.
+ * @param {string} string Outer.
+ * @param {string} substring Inner.
+ */
+TestSuite.prototype.assertContains = function(string, substring) {
+ if (string.indexOf(substring) == -1) {
+ this.fail('Expected to: "' + string + '" to contain "' + substring + '"');
+ }
+};
+
+
+/**
+ * Takes control over execution.
+ */
+TestSuite.prototype.takeControl = function() {
+ this.controlTaken_ = true;
+ // Set up guard timer.
+ var self = this;
+ this.timerId_ = setTimeout(function() {
+ self.reportFailure_('Timeout exceeded: 20 sec');
+ }, 20000);
+};
+
+
+/**
+ * Releases control over execution.
+ */
+TestSuite.prototype.releaseControl = function() {
+ if (this.timerId_ != -1) {
+ clearTimeout(this.timerId_);
+ this.timerId_ = -1;
+ }
+ this.reportOk_();
+};
+
+
+/**
+ * Async tests use this one to report that they are completed.
+ */
+TestSuite.prototype.reportOk_ = function() {
+ window.domAutomationController.send('[OK]');
+};
+
+
+/**
+ * Async tests use this one to report failures.
+ */
+TestSuite.prototype.reportFailure_ = function(error) {
+ if (this.timerId_ != -1) {
+ clearTimeout(this.timerId_);
+ this.timerId_ = -1;
+ }
+ window.domAutomationController.send('[FAILED] ' + error);
+};
+
+
+/**
+ * Runs all global functions starting with 'test' as unit tests.
+ */
+TestSuite.prototype.runTest = function(testName) {
+ try {
+ this[testName]();
+ if (!this.controlTaken_) {
+ this.reportOk_();
+ }
+ } catch (e) {
+ this.reportFailure_(e);
+ }
+};
+
+
+/**
+ * @param {string} panelName Name of the panel to show.
+ */
+TestSuite.prototype.showPanel = function(panelName) {
+ // Open Scripts panel.
+ var toolbar = document.getElementById('toolbar');
+ var button = toolbar.getElementsByClassName(panelName)[0];
+ button.click();
+ this.assertEquals(WebInspector.panels[panelName],
+ WebInspector.currentPanel);
+};
+
+
+/**
+ * Overrides the method with specified name until it's called first time.
+ * @param {Object} receiver An object whose method to override.
+ * @param {string} methodName Name of the method to override.
+ * @param {Function} override A function that should be called right after the
+ * overriden method returns.
+ * @param {boolean} opt_sticky Whether restore original method after first run
+ * or not.
+ */
+TestSuite.prototype.addSniffer = function(receiver, methodName, override,
+ opt_sticky) {
+ var orig = receiver[methodName];
+ if (typeof orig != 'function') {
+ this.fail('Cannot find method to override: ' + methodName);
+ }
+ var test = this;
+ receiver[methodName] = function(var_args) {
+ try {
+ var result = orig.apply(this, arguments);
+ } finally {
+ if (!opt_sticky) {
+ receiver[methodName] = orig;
+ }
+ }
+ // In case of exception the override won't be called.
+ try {
+ override.apply(this, arguments);
+ } catch (e) {
+ test.fail('Exception in overriden method "' + methodName + '": ' + e);
+ }
+ return result;
+ };
+};
+
+
+// UI Tests
+
+
+/**
+ * Tests that the real injected host is present in the context.
+ */
+TestSuite.prototype.testHostIsPresent = function() {
+ this.assertTrue(typeof DevToolsHost == 'object' && !DevToolsHost.isStub);
+};
+
+
+/**
+ * Tests elements tree has an 'HTML' root.
+ */
+TestSuite.prototype.testElementsTreeRoot = function() {
+ var doc = WebInspector.domAgent.document;
+ this.assertEquals('HTML', doc.documentElement.nodeName);
+ this.assertTrue(doc.documentElement.hasChildNodes());
+};
+
+
+/**
+ * Tests that main resource is present in the system and that it is
+ * the only resource.
+ */
+TestSuite.prototype.testMainResource = function() {
+ var tokens = [];
+ var resources = WebInspector.resources;
+ for (var id in resources) {
+ tokens.push(resources[id].lastPathComponent);
+ }
+ this.assertEquals('simple_page.html', tokens.join(','));
+};
+
+
+/**
+ * Tests that resources tab is enabled when corresponding item is selected.
+ */
+TestSuite.prototype.testEnableResourcesTab = function() {
+ this.showPanel('resources');
+
+ var test = this;
+ this.addSniffer(WebInspector, 'addResource',
+ function(identifier, payload) {
+ test.assertEquals('simple_page.html', payload.lastPathComponent);
+ WebInspector.panels.resources.refresh();
+ WebInspector.resources[identifier]._resourcesTreeElement.select();
+
+ test.releaseControl();
+ });
+
+ // Following call should lead to reload that we capture in the
+ // addResource override.
+ WebInspector.panels.resources._enableResourceTracking();
+
+ // We now have some time to report results to controller.
+ this.takeControl();
+};
+
+
+/**
+ * Tests resource headers.
+ */
+TestSuite.prototype.testResourceHeaders = function() {
+ this.showPanel('resources');
+
+ var test = this;
+
+ var requestOk = false;
+ var responseOk = false;
+ var timingOk = false;
+
+ this.addSniffer(WebInspector, 'addResource',
+ function(identifier, payload) {
+ var resource = this.resources[identifier];
+ if (resource.mainResource) {
+ // We are only interested in secondary resources in this test.
+ return;
+ }
+
+ var requestHeaders = JSON.stringify(resource.requestHeaders);
+ test.assertContains(requestHeaders, 'Accept');
+ requestOk = true;
+ }, true);
+
+ this.addSniffer(WebInspector, 'updateResource',
+ function(identifier, payload) {
+ var resource = this.resources[identifier];
+ if (resource.mainResource) {
+ // We are only interested in secondary resources in this test.
+ return;
+ }
+
+ if (payload.didResponseChange) {
+ var responseHeaders = JSON.stringify(resource.responseHeaders);
+ test.assertContains(responseHeaders, 'Content-type');
+ test.assertContains(responseHeaders, 'Content-Length');
+ test.assertTrue(typeof resource.responseReceivedTime != 'undefnied');
+ responseOk = true;
+ }
+
+ if (payload.didTimingChange) {
+ test.assertTrue(typeof resource.startTime != 'undefnied');
+ timingOk = true;
+ }
+
+ if (payload.didCompletionChange) {
+ test.assertTrue(requestOk);
+ test.assertTrue(responseOk);
+ test.assertTrue(timingOk);
+ test.assertTrue(typeof resource.endTime != 'undefnied');
+ test.releaseControl();
+ }
+ }, true);
+
+ WebInspector.panels.resources._enableResourceTracking();
+ this.takeControl();
+};
+
+
+/**
+ * Test that profiler works.
+ */
+TestSuite.prototype.testProfilerTab = function() {
+ this.showPanel('profiles');
+
+ var test = this;
+ this.addSniffer(WebInspector, 'addProfile',
+ function(profile) {
+ var panel = WebInspector.panels.profiles;
+ panel.showProfile(profile);
+ var node = panel.visibleView.profileDataGridTree.children[0];
+ // Iterate over displayed functions and search for a function
+ // that is called 'fib' or 'eternal_fib'. If found, it will mean
+ // that we actually have profiled page's code.
+ while (node) {
+ if (node.functionName.indexOf('fib') != -1) {
+ test.releaseControl();
+ }
+ node = node.traverseNextNode(true, null, true);
+ }
+
+ test.fail();
+ });
+ var ticksCount = 0;
+ var tickRecord = '\nt,';
+ this.addSniffer(RemoteDebuggerAgent, 'DidGetNextLogLines',
+ function(log) {
+ var pos = 0;
+ while ((pos = log.indexOf(tickRecord, pos)) != -1) {
+ pos += tickRecord.length;
+ ticksCount++;
+ }
+ if (ticksCount > 100) {
+ InspectorController.stopProfiling();
+ }
+ }, true);
+
+ InspectorController.startProfiling();
+ this.takeControl();
+};
+
+
+/**
+ * Tests that scripts tab can be open and populated with inspected scripts.
+ */
+TestSuite.prototype.testShowScriptsTab = function() {
+ var parsedDebuggerTestPageHtml = false;
+
+ // Intercept parsedScriptSource calls to check that all expected scripts are
+ // added to the debugger.
+ var test = this;
+ var receivedConsoleApiSource = false;
+ this.addSniffer(WebInspector, 'parsedScriptSource',
+ function(sourceID, sourceURL, source, startingLine) {
+ if (sourceURL == undefined) {
+ if (receivedConsoleApiSource) {
+ test.fail('Unexpected script without URL');
+ } else {
+ receivedConsoleApiSource = true;
+ }
+ } else if (sourceURL.search(/debugger_test_page.html$/) != -1) {
+ if (parsedDebuggerTestPageHtml) {
+ test.fail('Unexpected parse event: ' + sourceURL);
+ }
+ parsedDebuggerTestPageHtml = true;
+ } else {
+ test.fail('Unexpected script URL: ' + sourceURL);
+ }
+
+ if (!WebInspector.panels.scripts.visibleView) {
+ test.fail('No visible script view: ' + sourceURL);
+ }
+
+ // There should be two scripts: one for the main page and another
+ // one which is source of console API(see
+ // InjectedScript._ensureCommandLineAPIInstalled).
+ if (parsedDebuggerTestPageHtml && receivedConsoleApiSource) {
+ test.releaseControl();
+ }
+ }, true /* sticky */);
+
+ this.showPanel('scripts');
+
+ // Wait until all scripts are added to the debugger.
+ this.takeControl();
+};
+
+
+/**
+ * Tests that a breakpoint can be set.
+ */
+TestSuite.prototype.testSetBreakpoint = function() {
+ var parsedDebuggerTestPageHtml = false;
+ var parsedDebuggerTestJs = false;
+
+ this.showPanel('scripts');
+
+ var scriptUrl = null;
+ var breakpointLine = 12;
+
+ var test = this;
+ var orig = devtools.DebuggerAgent.prototype.handleScriptsResponse_;
+ this.addSniffer(devtools.DebuggerAgent.prototype, 'handleScriptsResponse_',
+ function(msg) {
+ var scriptSelect = document.getElementById('scripts-files');
+ var options = scriptSelect.options;
+
+ // There should be console API source (see
+ // InjectedScript._ensureCommandLineAPIInstalled) and the page script.
+ test.assertEquals(2, options.length, 'Unexpected number of scripts.');
+
+ // Select page's script if it's not current option.
+ var scriptResource;
+ if (options[scriptSelect.selectedIndex].text ==
+ 'debugger_test_page.html') {
+ scriptResource =
+ options[scriptSelect.selectedIndex].representedObject;
+ } else {
+ var pageScriptIndex = (1 - scriptSelect.selectedIndex);
+ test.assertEquals('debugger_test_page.html',
+ options[pageScriptIndex].text);
+ scriptResource = options[pageScriptIndex].representedObject;
+ // Current panel is 'Scripts'.
+ WebInspector.currentPanel._showScriptOrResource(scriptResource);
+ }
+
+ test.assertTrue(scriptResource instanceof WebInspector.Resource,
+ 'Unexpected resource class.');
+ test.assertTrue(!!scriptResource.url, 'Resource URL is null.');
+ test.assertTrue(
+ scriptResource.url.search(/debugger_test_page.html$/) != -1,
+ 'Main HTML resource should be selected.');
+
+ // Store for access from setbreakpoint handler.
+ scriptUrl = scriptResource.url;
+
+ var scriptsPanel = WebInspector.panels.scripts;
+
+ var view = scriptsPanel.visibleView;
+ test.assertTrue(view instanceof WebInspector.SourceView);
+
+ if (!view.sourceFrame._isContentLoaded()) {
+ test.addSniffer(view, '_sourceFrameSetupFinished', function(event) {
+ view._addBreakpoint(breakpointLine);
+ // Force v8 execution.
+ RemoteToolsAgent.ExecuteVoidJavaScript();
+ });
+ } else {
+ view._addBreakpoint(breakpointLine);
+ // Force v8 execution.
+ RemoteToolsAgent.ExecuteVoidJavaScript();
+ }
+ });
+
+ this.addSniffer(
+ devtools.DebuggerAgent.prototype,
+ 'handleSetBreakpointResponse_',
+ function(msg) {
+ var bps = this.urlToBreakpoints_[scriptUrl];
+ test.assertTrue(!!bps, 'No breakpoints for line ' + breakpointLine);
+ var line = devtools.DebuggerAgent.webkitToV8LineNumber_(breakpointLine);
+ test.assertTrue(!!bps[line].getV8Id(),
+ 'Breakpoint id was not assigned.');
+ test.releaseControl();
+ });
+
+ this.takeControl();
+};
+
+
+/**
+ * Tests 'Pause' button will pause debugger when a snippet is evaluated.
+ */
+TestSuite.prototype.testPauseInEval = function() {
+ this.showPanel('scripts');
+
+ var test = this;
+
+ var pauseButton = document.getElementById('scripts-pause');
+ pauseButton.click();
+
+ devtools.tools.evaluateJavaScript('fib(10)');
+
+ this.addSniffer(WebInspector, 'pausedScript',
+ function() {
+ test.releaseControl();
+ });
+
+ test.takeControl();
+};
+
+
+/**
+ * Key event with given key identifier.
+ */
+TestSuite.KeyEvent = function(key) {
+ this.keyIdentifier = key;
+};
+TestSuite.KeyEvent.prototype.preventDefault = function() {};
+TestSuite.KeyEvent.prototype.stopPropagation = function() {};
+
+
+/**
+ * Tests console eval.
+ */
+TestSuite.prototype.testConsoleEval = function() {
+ WebInspector.console.visible = true;
+ WebInspector.console.prompt.text = '123';
+ WebInspector.console.promptElement.handleKeyEvent(
+ new TestSuite.KeyEvent('Enter'));
+
+ var test = this;
+ this.addSniffer(WebInspector.ConsoleView.prototype, 'addMessage',
+ function(commandResult) {
+ test.assertEquals('123', commandResult.toMessageElement().textContent);
+ test.releaseControl();
+ });
+
+ this.takeControl();
+};
+
+
+/**
+ * Tests console log.
+ */
+TestSuite.prototype.testConsoleLog = function() {
+ WebInspector.console.visible = true;
+ var messages = WebInspector.console.messages;
+ var index = 0;
+
+ var test = this;
+ var assertNext = function(line, message, opt_class, opt_count, opt_substr) {
+ var elem = messages[index++].toMessageElement();
+ var clazz = elem.getAttribute('class');
+ var expectation = (opt_count || '') + 'console_test_page.html:' +
+ line + message;
+ if (opt_substr) {
+ test.assertContains(elem.textContent, expectation);
+ } else {
+ test.assertEquals(expectation, elem.textContent);
+ }
+ if (opt_class) {
+ test.assertContains(clazz, 'console-' + opt_class);
+ }
+ };
+
+ assertNext('5', 'log', 'log-level');
+ assertNext('7', 'debug', 'log-level');
+ assertNext('9', 'info', 'log-level');
+ assertNext('11', 'warn', 'warning-level');
+ assertNext('13', 'error', 'error-level');
+ assertNext('15', 'Message format number 1, 2 and 3.5');
+ assertNext('17', 'Message format for string');
+ assertNext('19', 'Object Object');
+ assertNext('22', 'repeated', 'log-level', 5);
+ assertNext('26', 'count: 1');
+ assertNext('26', 'count: 2');
+ assertNext('29', 'group', 'group-title');
+ index++;
+ assertNext('33', 'timer:', 'log-level', '', true);
+};
+
+
+/**
+ * Tests eval of global objects.
+ */
+TestSuite.prototype.testEvalGlobal = function() {
+ WebInspector.console.visible = true;
+ WebInspector.console.prompt.text = 'foo';
+ WebInspector.console.promptElement.handleKeyEvent(
+ new TestSuite.KeyEvent('Enter'));
+
+ var test = this;
+ this.addSniffer(WebInspector.ConsoleView.prototype, 'addMessage',
+ function(commandResult) {
+ test.assertEquals('fooValue',
+ commandResult.toMessageElement().textContent);
+ test.releaseControl();
+ });
+
+ this.takeControl();
+};
+
+
+/**
+ * Tests eval on call frame.
+ */
+TestSuite.prototype.testEvalCallFrame = function() {
+};
+
+
+/**
+ * Test runner for the test suite.
+ */
+var uiTests = {};
+
+
+/**
+ * Run each test from the test suit on a fresh instance of the suite.
+ */
+uiTests.runAllTests = function() {
+ // For debugging purposes.
+ for (var name in TestSuite.prototype) {
+ if (name.substring(0, 4) == 'test' &&
+ typeof TestSuite.prototype[name] == 'function') {
+ uiTests.runTest(name);
+ }
+ }
+};
+
+
+/**
+ * Run specified test on a fresh instance of the test suite.
+ * @param {string} name Name of a test method from TestSuite class.
+ */
+uiTests.runTest = function(name) {
+ new TestSuite().runTest(name);
+};
+
+
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/treeoutline.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/treeoutline.js
new file mode 100644
index 0000000..ecc322b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/treeoutline.js
@@ -0,0 +1,849 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+function TreeOutline(listNode)
+{
+ this.children = [];
+ this.selectedTreeElement = null;
+ this._childrenListNode = listNode;
+ this._childrenListNode.removeChildren();
+ this._knownTreeElements = [];
+ this._treeElementsExpandedState = [];
+ this.expandTreeElementsWhenArrowing = false;
+ this.root = true;
+ this.hasChildren = false;
+ this.expanded = true;
+ this.selected = false;
+ this.treeOutline = this;
+}
+
+TreeOutline._knownTreeElementNextIdentifier = 1;
+
+TreeOutline._appendChild = function(child)
+{
+ if (!child)
+ throw("child can't be undefined or null");
+
+ var lastChild = this.children[this.children.length - 1];
+ if (lastChild) {
+ lastChild.nextSibling = child;
+ child.previousSibling = lastChild;
+ } else {
+ child.previousSibling = null;
+ child.nextSibling = null;
+ }
+
+ this.children.push(child);
+ this.hasChildren = true;
+ child.parent = this;
+ child.treeOutline = this.treeOutline;
+ child.treeOutline._rememberTreeElement(child);
+
+ var current = child.children[0];
+ while (current) {
+ current.treeOutline = this.treeOutline;
+ current.treeOutline._rememberTreeElement(current);
+ current = current.traverseNextTreeElement(false, child, true);
+ }
+
+ if (child.hasChildren && child.treeOutline._treeElementsExpandedState[child.identifier] !== undefined)
+ child.expanded = child.treeOutline._treeElementsExpandedState[child.identifier];
+
+ if (!this._childrenListNode) {
+ this._childrenListNode = this.treeOutline._childrenListNode.ownerDocument.createElement("ol");
+ this._childrenListNode.parentTreeElement = this;
+ this._childrenListNode.addStyleClass("children");
+ if (this.hidden)
+ this._childrenListNode.addStyleClass("hidden");
+ }
+
+ child._attach();
+}
+
+TreeOutline._insertChild = function(child, index)
+{
+ if (!child)
+ throw("child can't be undefined or null");
+
+ var previousChild = (index > 0 ? this.children[index - 1] : null);
+ if (previousChild) {
+ previousChild.nextSibling = child;
+ child.previousSibling = previousChild;
+ } else {
+ child.previousSibling = null;
+ }
+
+ var nextChild = this.children[index];
+ if (nextChild) {
+ nextChild.previousSibling = child;
+ child.nextSibling = nextChild;
+ } else {
+ child.nextSibling = null;
+ }
+
+ this.children.splice(index, 0, child);
+ this.hasChildren = true;
+ child.parent = this;
+ child.treeOutline = this.treeOutline;
+ child.treeOutline._rememberTreeElement(child);
+
+ var current = child.children[0];
+ while (current) {
+ current.treeOutline = this.treeOutline;
+ current.treeOutline._rememberTreeElement(current);
+ current = current.traverseNextTreeElement(false, child, true);
+ }
+
+ if (child.hasChildren && child.treeOutline._treeElementsExpandedState[child.identifier] !== undefined)
+ child.expanded = child.treeOutline._treeElementsExpandedState[child.identifier];
+
+ if (!this._childrenListNode) {
+ this._childrenListNode = this.treeOutline._childrenListNode.ownerDocument.createElement("ol");
+ this._childrenListNode.parentTreeElement = this;
+ this._childrenListNode.addStyleClass("children");
+ if (this.hidden)
+ this._childrenListNode.addStyleClass("hidden");
+ }
+
+ child._attach();
+}
+
+TreeOutline._removeChildAtIndex = function(childIndex)
+{
+ if (childIndex < 0 || childIndex >= this.children.length)
+ throw("childIndex out of range");
+
+ var child = this.children[childIndex];
+ this.children.splice(childIndex, 1);
+
+ child.deselect();
+
+ if (child.previousSibling)
+ child.previousSibling.nextSibling = child.nextSibling;
+ if (child.nextSibling)
+ child.nextSibling.previousSibling = child.previousSibling;
+
+ if (child.treeOutline) {
+ child.treeOutline._forgetTreeElement(child);
+ child.treeOutline._forgetChildrenRecursive(child);
+ }
+
+ child._detach();
+ child.treeOutline = null;
+ child.parent = null;
+ child.nextSibling = null;
+ child.previousSibling = null;
+}
+
+TreeOutline._removeChild = function(child)
+{
+ if (!child)
+ throw("child can't be undefined or null");
+
+ var childIndex = this.children.indexOf(child);
+ if (childIndex === -1)
+ throw("child not found in this node's children");
+
+ TreeOutline._removeChildAtIndex.call(this, childIndex);
+}
+
+TreeOutline._removeChildren = function()
+{
+ for (var i = 0; i < this.children.length; ++i) {
+ var child = this.children[i];
+ child.deselect();
+
+ if (child.treeOutline) {
+ child.treeOutline._forgetTreeElement(child);
+ child.treeOutline._forgetChildrenRecursive(child);
+ }
+
+ child._detach();
+ child.treeOutline = null;
+ child.parent = null;
+ child.nextSibling = null;
+ child.previousSibling = null;
+ }
+
+ this.children = [];
+}
+
+TreeOutline._removeChildrenRecursive = function()
+{
+ var childrenToRemove = this.children;
+
+ var child = this.children[0];
+ while (child) {
+ if (child.children.length)
+ childrenToRemove = childrenToRemove.concat(child.children);
+ child = child.traverseNextTreeElement(false, this, true);
+ }
+
+ for (var i = 0; i < childrenToRemove.length; ++i) {
+ var child = childrenToRemove[i];
+ child.deselect();
+ if (child.treeOutline)
+ child.treeOutline._forgetTreeElement(child);
+ child._detach();
+ child.children = [];
+ child.treeOutline = null;
+ child.parent = null;
+ child.nextSibling = null;
+ child.previousSibling = null;
+ }
+
+ this.children = [];
+}
+
+TreeOutline.prototype._rememberTreeElement = function(element)
+{
+ if (!this._knownTreeElements[element.identifier])
+ this._knownTreeElements[element.identifier] = [];
+
+ // check if the element is already known
+ var elements = this._knownTreeElements[element.identifier];
+ if (elements.indexOf(element) !== -1)
+ return;
+
+ // add the element
+ elements.push(element);
+}
+
+TreeOutline.prototype._forgetTreeElement = function(element)
+{
+ if (this._knownTreeElements[element.identifier])
+ this._knownTreeElements[element.identifier].remove(element, true);
+}
+
+TreeOutline.prototype._forgetChildrenRecursive = function(parentElement)
+{
+ var child = parentElement.children[0];
+ while (child) {
+ this._forgetTreeElement(child);
+ child = child.traverseNextTreeElement(false, this, true);
+ }
+}
+
+TreeOutline.prototype.getCachedTreeElement = function(representedObject)
+{
+ if (!representedObject)
+ return null;
+
+ if ("__treeElementIdentifier" in representedObject) {
+ // If this representedObject has a tree element identifier, and it is a known TreeElement
+ // in our tree we can just return that tree element.
+ var elements = this._knownTreeElements[representedObject.__treeElementIdentifier];
+ if (elements) {
+ for (var i = 0; i < elements.length; ++i)
+ if (elements[i].representedObject === representedObject)
+ return elements[i];
+ }
+ }
+ return null;
+}
+
+TreeOutline.prototype.findTreeElement = function(representedObject, isAncestor, getParent)
+{
+ if (!representedObject)
+ return null;
+
+ var cachedElement = this.getCachedTreeElement(representedObject);
+ if (cachedElement)
+ return cachedElement;
+
+ // The representedObject isn't know, so we start at the top of the tree and work down to find the first
+ // tree element that represents representedObject or one of its ancestors.
+ var item;
+ var found = false;
+ for (var i = 0; i < this.children.length; ++i) {
+ item = this.children[i];
+ if (item.representedObject === representedObject || isAncestor(item.representedObject, representedObject)) {
+ found = true;
+ break;
+ }
+ }
+
+ if (!found)
+ return null;
+
+ // Make sure the item that we found is connected to the root of the tree.
+ // Build up a list of representedObject's ancestors that aren't already in our tree.
+ var ancestors = [];
+ var currentObject = representedObject;
+ while (currentObject) {
+ ancestors.unshift(currentObject);
+ if (currentObject === item.representedObject)
+ break;
+ currentObject = getParent(currentObject);
+ }
+
+ // For each of those ancestors we populate them to fill in the tree.
+ for (var i = 0; i < ancestors.length; ++i) {
+ // Make sure we don't call findTreeElement with the same representedObject
+ // again, to prevent infinite recursion.
+ if (ancestors[i] === representedObject)
+ continue;
+ // FIXME: we could do something faster than findTreeElement since we will know the next
+ // ancestor exists in the tree.
+ item = this.findTreeElement(ancestors[i], isAncestor, getParent);
+ if (item && item.onpopulate)
+ item.onpopulate(item);
+ }
+
+ return this.getCachedTreeElement(representedObject);
+}
+
+TreeOutline.prototype.treeElementFromPoint = function(x, y)
+{
+ var node = this._childrenListNode.ownerDocument.elementFromPoint(x, y);
+ var listNode = node.enclosingNodeOrSelfWithNodeNameInArray(["ol", "li"]);
+ if (listNode)
+ return listNode.parentTreeElement || listNode.treeElement;
+ return null;
+}
+
+TreeOutline.prototype.handleKeyEvent = function(event)
+{
+ if (!this.selectedTreeElement || event.shiftKey || event.metaKey || event.ctrlKey)
+ return false;
+
+ var handled = false;
+ var nextSelectedElement;
+ if (event.keyIdentifier === "Up" && !event.altKey) {
+ nextSelectedElement = this.selectedTreeElement.traversePreviousTreeElement(true);
+ while (nextSelectedElement && !nextSelectedElement.selectable)
+ nextSelectedElement = nextSelectedElement.traversePreviousTreeElement(!this.expandTreeElementsWhenArrowing);
+ handled = nextSelectedElement ? true : false;
+ } else if (event.keyIdentifier === "Down" && !event.altKey) {
+ nextSelectedElement = this.selectedTreeElement.traverseNextTreeElement(true);
+ while (nextSelectedElement && !nextSelectedElement.selectable)
+ nextSelectedElement = nextSelectedElement.traverseNextTreeElement(!this.expandTreeElementsWhenArrowing);
+ handled = nextSelectedElement ? true : false;
+ } else if (event.keyIdentifier === "Left") {
+ if (this.selectedTreeElement.expanded) {
+ if (event.altKey)
+ this.selectedTreeElement.collapseRecursively();
+ else
+ this.selectedTreeElement.collapse();
+ handled = true;
+ } else if (this.selectedTreeElement.parent && !this.selectedTreeElement.parent.root) {
+ handled = true;
+ if (this.selectedTreeElement.parent.selectable) {
+ nextSelectedElement = this.selectedTreeElement.parent;
+ handled = nextSelectedElement ? true : false;
+ } else if (this.selectedTreeElement.parent)
+ this.selectedTreeElement.parent.collapse();
+ }
+ } else if (event.keyIdentifier === "Right") {
+ if (!this.selectedTreeElement.revealed()) {
+ this.selectedTreeElement.reveal();
+ handled = true;
+ } else if (this.selectedTreeElement.hasChildren) {
+ handled = true;
+ if (this.selectedTreeElement.expanded) {
+ nextSelectedElement = this.selectedTreeElement.children[0];
+ handled = nextSelectedElement ? true : false;
+ } else {
+ if (event.altKey)
+ this.selectedTreeElement.expandRecursively();
+ else
+ this.selectedTreeElement.expand();
+ }
+ }
+ }
+
+ if (nextSelectedElement) {
+ nextSelectedElement.reveal();
+ nextSelectedElement.select();
+ }
+
+ if (handled) {
+ event.preventDefault();
+ event.stopPropagation();
+ }
+
+ return handled;
+}
+
+TreeOutline.prototype.expand = function()
+{
+ // this is the root, do nothing
+}
+
+TreeOutline.prototype.collapse = function()
+{
+ // this is the root, do nothing
+}
+
+TreeOutline.prototype.revealed = function()
+{
+ return true;
+}
+
+TreeOutline.prototype.reveal = function()
+{
+ // this is the root, do nothing
+}
+
+TreeOutline.prototype.appendChild = TreeOutline._appendChild;
+TreeOutline.prototype.insertChild = TreeOutline._insertChild;
+TreeOutline.prototype.removeChild = TreeOutline._removeChild;
+TreeOutline.prototype.removeChildAtIndex = TreeOutline._removeChildAtIndex;
+TreeOutline.prototype.removeChildren = TreeOutline._removeChildren;
+TreeOutline.prototype.removeChildrenRecursive = TreeOutline._removeChildrenRecursive;
+
+function TreeElement(title, representedObject, hasChildren)
+{
+ this._title = title;
+ this.representedObject = (representedObject || {});
+
+ if (this.representedObject.__treeElementIdentifier)
+ this.identifier = this.representedObject.__treeElementIdentifier;
+ else {
+ this.identifier = TreeOutline._knownTreeElementNextIdentifier++;
+ this.representedObject.__treeElementIdentifier = this.identifier;
+ }
+
+ this._hidden = false;
+ this.expanded = false;
+ this.selected = false;
+ this.hasChildren = hasChildren;
+ this.children = [];
+ this.treeOutline = null;
+ this.parent = null;
+ this.previousSibling = null;
+ this.nextSibling = null;
+ this._listItemNode = null;
+}
+
+TreeElement.prototype = {
+ selectable: true,
+ arrowToggleWidth: 10,
+
+ get listItemElement() {
+ return this._listItemNode;
+ },
+
+ get childrenListElement() {
+ return this._childrenListNode;
+ },
+
+ get title() {
+ return this._title;
+ },
+
+ set title(x) {
+ this._title = x;
+ if (this._listItemNode)
+ this._listItemNode.innerHTML = x;
+ },
+
+ get tooltip() {
+ return this._tooltip;
+ },
+
+ set tooltip(x) {
+ this._tooltip = x;
+ if (this._listItemNode)
+ this._listItemNode.title = x ? x : "";
+ },
+
+ get hasChildren() {
+ return this._hasChildren;
+ },
+
+ set hasChildren(x) {
+ if (this._hasChildren === x)
+ return;
+
+ this._hasChildren = x;
+
+ if (!this._listItemNode)
+ return;
+
+ if (x)
+ this._listItemNode.addStyleClass("parent");
+ else {
+ this._listItemNode.removeStyleClass("parent");
+ this.collapse();
+ }
+ },
+
+ get hidden() {
+ return this._hidden;
+ },
+
+ set hidden(x) {
+ if (this._hidden === x)
+ return;
+
+ this._hidden = x;
+
+ if (x) {
+ if (this._listItemNode)
+ this._listItemNode.addStyleClass("hidden");
+ if (this._childrenListNode)
+ this._childrenListNode.addStyleClass("hidden");
+ } else {
+ if (this._listItemNode)
+ this._listItemNode.removeStyleClass("hidden");
+ if (this._childrenListNode)
+ this._childrenListNode.removeStyleClass("hidden");
+ }
+ },
+
+ get shouldRefreshChildren() {
+ return this._shouldRefreshChildren;
+ },
+
+ set shouldRefreshChildren(x) {
+ this._shouldRefreshChildren = x;
+ if (x && this.expanded)
+ this.expand();
+ }
+}
+
+TreeElement.prototype.appendChild = TreeOutline._appendChild;
+TreeElement.prototype.insertChild = TreeOutline._insertChild;
+TreeElement.prototype.removeChild = TreeOutline._removeChild;
+TreeElement.prototype.removeChildAtIndex = TreeOutline._removeChildAtIndex;
+TreeElement.prototype.removeChildren = TreeOutline._removeChildren;
+TreeElement.prototype.removeChildrenRecursive = TreeOutline._removeChildrenRecursive;
+
+TreeElement.prototype._attach = function()
+{
+ if (!this._listItemNode || this.parent._shouldRefreshChildren) {
+ if (this._listItemNode && this._listItemNode.parentNode)
+ this._listItemNode.parentNode.removeChild(this._listItemNode);
+
+ this._listItemNode = this.treeOutline._childrenListNode.ownerDocument.createElement("li");
+ this._listItemNode.treeElement = this;
+ this._listItemNode.innerHTML = this._title;
+ this._listItemNode.title = this._tooltip ? this._tooltip : "";
+
+ if (this.hidden)
+ this._listItemNode.addStyleClass("hidden");
+ if (this.hasChildren)
+ this._listItemNode.addStyleClass("parent");
+ if (this.expanded)
+ this._listItemNode.addStyleClass("expanded");
+ if (this.selected)
+ this._listItemNode.addStyleClass("selected");
+
+ this._listItemNode.addEventListener("mousedown", TreeElement.treeElementSelected, false);
+ this._listItemNode.addEventListener("click", TreeElement.treeElementToggled, false);
+ this._listItemNode.addEventListener("dblclick", TreeElement.treeElementDoubleClicked, false);
+
+ if (this.onattach)
+ this.onattach(this);
+ }
+
+ var nextSibling = null;
+ if (this.nextSibling && this.nextSibling._listItemNode && this.nextSibling._listItemNode.parentNode === this.parent._childrenListNode)
+ nextSibling = this.nextSibling._listItemNode;
+ this.parent._childrenListNode.insertBefore(this._listItemNode, nextSibling);
+ if (this._childrenListNode)
+ this.parent._childrenListNode.insertBefore(this._childrenListNode, this._listItemNode.nextSibling);
+ if (this.selected)
+ this.select();
+ if (this.expanded)
+ this.expand();
+}
+
+TreeElement.prototype._detach = function()
+{
+ if (this._listItemNode && this._listItemNode.parentNode)
+ this._listItemNode.parentNode.removeChild(this._listItemNode);
+ if (this._childrenListNode && this._childrenListNode.parentNode)
+ this._childrenListNode.parentNode.removeChild(this._childrenListNode);
+}
+
+TreeElement.treeElementSelected = function(event)
+{
+ var element = event.currentTarget;
+ if (!element || !element.treeElement || !element.treeElement.selectable)
+ return;
+
+ if (element.treeElement.isEventWithinDisclosureTriangle(event))
+ return;
+
+ element.treeElement.select();
+}
+
+TreeElement.treeElementToggled = function(event)
+{
+ var element = event.currentTarget;
+ if (!element || !element.treeElement)
+ return;
+
+ if (!element.treeElement.isEventWithinDisclosureTriangle(event))
+ return;
+
+ if (element.treeElement.expanded) {
+ if (event.altKey)
+ element.treeElement.collapseRecursively();
+ else
+ element.treeElement.collapse();
+ } else {
+ if (event.altKey)
+ element.treeElement.expandRecursively();
+ else
+ element.treeElement.expand();
+ }
+}
+
+TreeElement.treeElementDoubleClicked = function(event)
+{
+ var element = event.currentTarget;
+ if (!element || !element.treeElement)
+ return;
+
+ if (element.treeElement.ondblclick)
+ element.treeElement.ondblclick(element.treeElement, event);
+ else if (element.treeElement.hasChildren && !element.treeElement.expanded)
+ element.treeElement.expand();
+}
+
+TreeElement.prototype.collapse = function()
+{
+ if (this._listItemNode)
+ this._listItemNode.removeStyleClass("expanded");
+ if (this._childrenListNode)
+ this._childrenListNode.removeStyleClass("expanded");
+
+ this.expanded = false;
+ if (this.treeOutline)
+ this.treeOutline._treeElementsExpandedState[this.identifier] = true;
+
+ if (this.oncollapse)
+ this.oncollapse(this);
+}
+
+TreeElement.prototype.collapseRecursively = function()
+{
+ var item = this;
+ while (item) {
+ if (item.expanded)
+ item.collapse();
+ item = item.traverseNextTreeElement(false, this, true);
+ }
+}
+
+TreeElement.prototype.expand = function()
+{
+ if (!this.hasChildren || (this.expanded && !this._shouldRefreshChildren && this._childrenListNode))
+ return;
+
+ if (this.treeOutline && (!this._childrenListNode || this._shouldRefreshChildren)) {
+ if (this._childrenListNode && this._childrenListNode.parentNode)
+ this._childrenListNode.parentNode.removeChild(this._childrenListNode);
+
+ this._childrenListNode = this.treeOutline._childrenListNode.ownerDocument.createElement("ol");
+ this._childrenListNode.parentTreeElement = this;
+ this._childrenListNode.addStyleClass("children");
+
+ if (this.hidden)
+ this._childrenListNode.addStyleClass("hidden");
+
+ if (this.onpopulate)
+ this.onpopulate(this);
+
+ for (var i = 0; i < this.children.length; ++i)
+ this.children[i]._attach();
+
+ delete this._shouldRefreshChildren;
+ }
+
+ if (this._listItemNode) {
+ this._listItemNode.addStyleClass("expanded");
+ if (this._childrenListNode && this._childrenListNode.parentNode != this._listItemNode.parentNode)
+ this.parent._childrenListNode.insertBefore(this._childrenListNode, this._listItemNode.nextSibling);
+ }
+
+ if (this._childrenListNode)
+ this._childrenListNode.addStyleClass("expanded");
+
+ this.expanded = true;
+ if (this.treeOutline)
+ this.treeOutline._treeElementsExpandedState[this.identifier] = true;
+
+ if (this.onexpand)
+ this.onexpand(this);
+}
+
+TreeElement.prototype.expandRecursively = function(maxDepth)
+{
+ var item = this;
+ var info = {};
+ var depth = 0;
+
+ // The Inspector uses TreeOutlines to represents object properties, so recursive expansion
+ // in some case can be infinite, since JavaScript objects can hold circular references.
+ // So default to a recursion cap of 3 levels, since that gives fairly good results.
+ if (typeof maxDepth === "undefined" || typeof maxDepth === "null")
+ maxDepth = 3;
+
+ while (item) {
+ if (depth < maxDepth)
+ item.expand();
+ item = item.traverseNextTreeElement(false, this, (depth >= maxDepth), info);
+ depth += info.depthChange;
+ }
+}
+
+TreeElement.prototype.hasAncestor = function(ancestor) {
+ if (!ancestor)
+ return false;
+
+ var currentNode = this.parent;
+ while (currentNode) {
+ if (ancestor === currentNode)
+ return true;
+ currentNode = currentNode.parent;
+ }
+
+ return false;
+}
+
+TreeElement.prototype.reveal = function()
+{
+ var currentAncestor = this.parent;
+ while (currentAncestor && !currentAncestor.root) {
+ if (!currentAncestor.expanded)
+ currentAncestor.expand();
+ currentAncestor = currentAncestor.parent;
+ }
+
+ if (this.onreveal)
+ this.onreveal(this);
+}
+
+TreeElement.prototype.revealed = function()
+{
+ var currentAncestor = this.parent;
+ while (currentAncestor && !currentAncestor.root) {
+ if (!currentAncestor.expanded)
+ return false;
+ currentAncestor = currentAncestor.parent;
+ }
+
+ return true;
+}
+
+TreeElement.prototype.select = function(supressOnSelect)
+{
+ if (!this.treeOutline || !this.selectable || this.selected)
+ return;
+
+ if (this.treeOutline.selectedTreeElement)
+ this.treeOutline.selectedTreeElement.deselect();
+
+ this.selected = true;
+ this.treeOutline.selectedTreeElement = this;
+ if (this._listItemNode)
+ this._listItemNode.addStyleClass("selected");
+
+ if (this.onselect && !supressOnSelect)
+ this.onselect(this);
+}
+
+TreeElement.prototype.deselect = function(supressOnDeselect)
+{
+ if (!this.treeOutline || this.treeOutline.selectedTreeElement !== this || !this.selected)
+ return;
+
+ this.selected = false;
+ this.treeOutline.selectedTreeElement = null;
+ if (this._listItemNode)
+ this._listItemNode.removeStyleClass("selected");
+
+ if (this.ondeselect && !supressOnDeselect)
+ this.ondeselect(this);
+}
+
+TreeElement.prototype.traverseNextTreeElement = function(skipHidden, stayWithin, dontPopulate, info)
+{
+ if (!dontPopulate && this.hasChildren && this.onpopulate)
+ this.onpopulate(this);
+
+ if (info)
+ info.depthChange = 0;
+
+ var element = skipHidden ? (this.revealed() ? this.children[0] : null) : this.children[0];
+ if (element && (!skipHidden || (skipHidden && this.expanded))) {
+ if (info)
+ info.depthChange = 1;
+ return element;
+ }
+
+ if (this === stayWithin)
+ return null;
+
+ element = skipHidden ? (this.revealed() ? this.nextSibling : null) : this.nextSibling;
+ if (element)
+ return element;
+
+ element = this;
+ while (element && !element.root && !(skipHidden ? (element.revealed() ? element.nextSibling : null) : element.nextSibling) && element.parent !== stayWithin) {
+ if (info)
+ info.depthChange -= 1;
+ element = element.parent;
+ }
+
+ if (!element)
+ return null;
+
+ return (skipHidden ? (element.revealed() ? element.nextSibling : null) : element.nextSibling);
+}
+
+TreeElement.prototype.traversePreviousTreeElement = function(skipHidden, dontPopulate)
+{
+ var element = skipHidden ? (this.revealed() ? this.previousSibling : null) : this.previousSibling;
+ if (!dontPopulate && element && element.hasChildren && element.onpopulate)
+ element.onpopulate(element);
+
+ while (element && (skipHidden ? (element.revealed() && element.expanded ? element.children[element.children.length - 1] : null) : element.children[element.children.length - 1])) {
+ if (!dontPopulate && element.hasChildren && element.onpopulate)
+ element.onpopulate(element);
+ element = (skipHidden ? (element.revealed() && element.expanded ? element.children[element.children.length - 1] : null) : element.children[element.children.length - 1]);
+ }
+
+ if (element)
+ return element;
+
+ if (!this.parent || this.parent.root)
+ return null;
+
+ return this.parent;
+}
+
+TreeElement.prototype.isEventWithinDisclosureTriangle = function(event)
+{
+ var left = this._listItemNode.totalOffsetLeft;
+ return event.pageX >= left && event.pageX <= left + this.arrowToggleWidth && this.hasChildren;
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/resources/inspector/utilities.js b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/utilities.js
new file mode 100644
index 0000000..e831abd
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/resources/inspector/utilities.js
@@ -0,0 +1,905 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+Object.proxyType = function(objectProxy)
+{
+ if (objectProxy === null)
+ return "null";
+
+ var type = typeof objectProxy;
+ if (type !== "object" && type !== "function")
+ return type;
+
+ return objectProxy.type;
+}
+
+Object.properties = function(obj)
+{
+ var properties = [];
+ for (var prop in obj)
+ properties.push(prop);
+ return properties;
+}
+
+Object.sortedProperties = function(obj, sortFunc)
+{
+ return Object.properties(obj).sort(sortFunc);
+}
+
+Function.prototype.bind = function(thisObject)
+{
+ var func = this;
+ var args = Array.prototype.slice.call(arguments, 1);
+ return function() { return func.apply(thisObject, args.concat(Array.prototype.slice.call(arguments, 0))) };
+}
+
+Node.prototype.rangeOfWord = function(offset, stopCharacters, stayWithinNode, direction)
+{
+ var startNode;
+ var startOffset = 0;
+ var endNode;
+ var endOffset = 0;
+
+ if (!stayWithinNode)
+ stayWithinNode = this;
+
+ if (!direction || direction === "backward" || direction === "both") {
+ var node = this;
+ while (node) {
+ if (node === stayWithinNode) {
+ if (!startNode)
+ startNode = stayWithinNode;
+ break;
+ }
+
+ if (node.nodeType === Node.TEXT_NODE) {
+ var start = (node === this ? (offset - 1) : (node.nodeValue.length - 1));
+ for (var i = start; i >= 0; --i) {
+ if (stopCharacters.indexOf(node.nodeValue[i]) !== -1) {
+ startNode = node;
+ startOffset = i + 1;
+ break;
+ }
+ }
+ }
+
+ if (startNode)
+ break;
+
+ node = node.traversePreviousNode(false, stayWithinNode);
+ }
+
+ if (!startNode) {
+ startNode = stayWithinNode;
+ startOffset = 0;
+ }
+ } else {
+ startNode = this;
+ startOffset = offset;
+ }
+
+ if (!direction || direction === "forward" || direction === "both") {
+ node = this;
+ while (node) {
+ if (node === stayWithinNode) {
+ if (!endNode)
+ endNode = stayWithinNode;
+ break;
+ }
+
+ if (node.nodeType === Node.TEXT_NODE) {
+ var start = (node === this ? offset : 0);
+ for (var i = start; i < node.nodeValue.length; ++i) {
+ if (stopCharacters.indexOf(node.nodeValue[i]) !== -1) {
+ endNode = node;
+ endOffset = i;
+ break;
+ }
+ }
+ }
+
+ if (endNode)
+ break;
+
+ node = node.traverseNextNode(false, stayWithinNode);
+ }
+
+ if (!endNode) {
+ endNode = stayWithinNode;
+ endOffset = stayWithinNode.nodeType === Node.TEXT_NODE ? stayWithinNode.nodeValue.length : stayWithinNode.childNodes.length;
+ }
+ } else {
+ endNode = this;
+ endOffset = offset;
+ }
+
+ var result = this.ownerDocument.createRange();
+ result.setStart(startNode, startOffset);
+ result.setEnd(endNode, endOffset);
+
+ return result;
+}
+
+Element.prototype.removeStyleClass = function(className)
+{
+ // Test for the simple case before using a RegExp.
+ if (this.className === className) {
+ this.className = "";
+ return;
+ }
+
+ this.removeMatchingStyleClasses(className.escapeForRegExp());
+}
+
+Element.prototype.removeMatchingStyleClasses = function(classNameRegex)
+{
+ var regex = new RegExp("(^|\\s+)" + classNameRegex + "($|\\s+)");
+ if (regex.test(this.className))
+ this.className = this.className.replace(regex, " ");
+}
+
+Element.prototype.addStyleClass = function(className)
+{
+ if (className && !this.hasStyleClass(className))
+ this.className += (this.className.length ? " " + className : className);
+}
+
+Element.prototype.hasStyleClass = function(className)
+{
+ if (!className)
+ return false;
+ // Test for the simple case before using a RegExp.
+ if (this.className === className)
+ return true;
+ var regex = new RegExp("(^|\\s)" + className.escapeForRegExp() + "($|\\s)");
+ return regex.test(this.className);
+}
+
+Element.prototype.positionAt = function(x, y)
+{
+ this.style.left = x + "px";
+ this.style.top = y + "px";
+}
+
+Node.prototype.enclosingNodeOrSelfWithNodeNameInArray = function(nameArray)
+{
+ for (var node = this; node && node !== this.ownerDocument; node = node.parentNode)
+ for (var i = 0; i < nameArray.length; ++i)
+ if (node.nodeName.toLowerCase() === nameArray[i].toLowerCase())
+ return node;
+ return null;
+}
+
+Node.prototype.enclosingNodeOrSelfWithNodeName = function(nodeName)
+{
+ return this.enclosingNodeOrSelfWithNodeNameInArray([nodeName]);
+}
+
+Node.prototype.enclosingNodeOrSelfWithClass = function(className)
+{
+ for (var node = this; node && node !== this.ownerDocument; node = node.parentNode)
+ if (node.nodeType === Node.ELEMENT_NODE && node.hasStyleClass(className))
+ return node;
+ return null;
+}
+
+Node.prototype.enclosingNodeWithClass = function(className)
+{
+ if (!this.parentNode)
+ return null;
+ return this.parentNode.enclosingNodeOrSelfWithClass(className);
+}
+
+Element.prototype.query = function(query)
+{
+ return this.ownerDocument.evaluate(query, this, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
+}
+
+Element.prototype.removeChildren = function()
+{
+ while (this.firstChild)
+ this.removeChild(this.firstChild);
+}
+
+Element.prototype.isInsertionCaretInside = function()
+{
+ var selection = window.getSelection();
+ if (!selection.rangeCount || !selection.isCollapsed)
+ return false;
+ var selectionRange = selection.getRangeAt(0);
+ return selectionRange.startContainer === this || selectionRange.startContainer.isDescendant(this);
+}
+
+Element.prototype.__defineGetter__("totalOffsetLeft", function()
+{
+ var total = 0;
+ for (var element = this; element; element = element.offsetParent)
+ total += element.offsetLeft;
+ return total;
+});
+
+Element.prototype.__defineGetter__("totalOffsetTop", function()
+{
+ var total = 0;
+ for (var element = this; element; element = element.offsetParent)
+ total += element.offsetTop;
+ return total;
+});
+
+Element.prototype.offsetRelativeToWindow = function(targetWindow)
+{
+ var elementOffset = {x: 0, y: 0};
+ var curElement = this;
+ var curWindow = this.ownerDocument.defaultView;
+ while (curWindow && curElement) {
+ elementOffset.x += curElement.totalOffsetLeft;
+ elementOffset.y += curElement.totalOffsetTop;
+ if (curWindow === targetWindow)
+ break;
+
+ curElement = curWindow.frameElement;
+ curWindow = curWindow.parent;
+ }
+
+ return elementOffset;
+}
+
+Element.prototype.firstChildSkippingWhitespace = firstChildSkippingWhitespace;
+Element.prototype.lastChildSkippingWhitespace = lastChildSkippingWhitespace;
+
+Node.prototype.isWhitespace = isNodeWhitespace;
+Node.prototype.displayName = nodeDisplayName;
+Node.prototype.isAncestor = function(node)
+{
+ return isAncestorNode(this, node);
+};
+Node.prototype.isDescendant = isDescendantNode;
+Node.prototype.nextSiblingSkippingWhitespace = nextSiblingSkippingWhitespace;
+Node.prototype.previousSiblingSkippingWhitespace = previousSiblingSkippingWhitespace;
+Node.prototype.traverseNextNode = traverseNextNode;
+Node.prototype.traversePreviousNode = traversePreviousNode;
+Node.prototype.onlyTextChild = onlyTextChild;
+
+String.prototype.hasSubstring = function(string, caseInsensitive)
+{
+ if (!caseInsensitive)
+ return this.indexOf(string) !== -1;
+ return this.match(new RegExp(string.escapeForRegExp(), "i"));
+}
+
+String.prototype.escapeCharacters = function(chars)
+{
+ var foundChar = false;
+ for (var i = 0; i < chars.length; ++i) {
+ if (this.indexOf(chars.charAt(i)) !== -1) {
+ foundChar = true;
+ break;
+ }
+ }
+
+ if (!foundChar)
+ return this;
+
+ var result = "";
+ for (var i = 0; i < this.length; ++i) {
+ if (chars.indexOf(this.charAt(i)) !== -1)
+ result += "\\";
+ result += this.charAt(i);
+ }
+
+ return result;
+}
+
+String.prototype.escapeForRegExp = function()
+{
+ return this.escapeCharacters("^[]{}()\\.$*+?|");
+}
+
+String.prototype.escapeHTML = function()
+{
+ return this.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;");
+}
+
+String.prototype.collapseWhitespace = function()
+{
+ return this.replace(/[\s\xA0]+/g, " ");
+}
+
+String.prototype.trimLeadingWhitespace = function()
+{
+ return this.replace(/^[\s\xA0]+/g, "");
+}
+
+String.prototype.trimTrailingWhitespace = function()
+{
+ return this.replace(/[\s\xA0]+$/g, "");
+}
+
+String.prototype.trimWhitespace = function()
+{
+ return this.replace(/^[\s\xA0]+|[\s\xA0]+$/g, "");
+}
+
+String.prototype.trimURL = function(baseURLDomain)
+{
+ var result = this.replace(new RegExp("^http[s]?:\/\/", "i"), "");
+ if (baseURLDomain)
+ result = result.replace(new RegExp("^" + baseURLDomain.escapeForRegExp(), "i"), "");
+ return result;
+}
+
+function isNodeWhitespace()
+{
+ if (!this || this.nodeType !== Node.TEXT_NODE)
+ return false;
+ if (!this.nodeValue.length)
+ return true;
+ return this.nodeValue.match(/^[\s\xA0]+$/);
+}
+
+function nodeDisplayName()
+{
+ if (!this)
+ return "";
+
+ switch (this.nodeType) {
+ case Node.DOCUMENT_NODE:
+ return "Document";
+
+ case Node.ELEMENT_NODE:
+ var name = "<" + this.nodeName.toLowerCase();
+
+ if (this.hasAttributes()) {
+ var value = this.getAttribute("id");
+ if (value)
+ name += " id=\"" + value + "\"";
+ value = this.getAttribute("class");
+ if (value)
+ name += " class=\"" + value + "\"";
+ if (this.nodeName.toLowerCase() === "a") {
+ value = this.getAttribute("name");
+ if (value)
+ name += " name=\"" + value + "\"";
+ value = this.getAttribute("href");
+ if (value)
+ name += " href=\"" + value + "\"";
+ } else if (this.nodeName.toLowerCase() === "img") {
+ value = this.getAttribute("src");
+ if (value)
+ name += " src=\"" + value + "\"";
+ } else if (this.nodeName.toLowerCase() === "iframe") {
+ value = this.getAttribute("src");
+ if (value)
+ name += " src=\"" + value + "\"";
+ } else if (this.nodeName.toLowerCase() === "input") {
+ value = this.getAttribute("name");
+ if (value)
+ name += " name=\"" + value + "\"";
+ value = this.getAttribute("type");
+ if (value)
+ name += " type=\"" + value + "\"";
+ } else if (this.nodeName.toLowerCase() === "form") {
+ value = this.getAttribute("action");
+ if (value)
+ name += " action=\"" + value + "\"";
+ }
+ }
+
+ return name + ">";
+
+ case Node.TEXT_NODE:
+ if (isNodeWhitespace.call(this))
+ return "(whitespace)";
+ return "\"" + this.nodeValue + "\"";
+
+ case Node.COMMENT_NODE:
+ return "<!--" + this.nodeValue + "-->";
+
+ case Node.DOCUMENT_TYPE_NODE:
+ var docType = "<!DOCTYPE " + this.nodeName;
+ if (this.publicId) {
+ docType += " PUBLIC \"" + this.publicId + "\"";
+ if (this.systemId)
+ docType += " \"" + this.systemId + "\"";
+ } else if (this.systemId)
+ docType += " SYSTEM \"" + this.systemId + "\"";
+ if (this.internalSubset)
+ docType += " [" + this.internalSubset + "]";
+ return docType + ">";
+ }
+
+ return this.nodeName.toLowerCase().collapseWhitespace();
+}
+
+function isAncestorNode(ancestor, node)
+{
+ if (!node || !ancestor)
+ return false;
+
+ var currentNode = node.parentNode;
+ while (currentNode) {
+ if (ancestor === currentNode)
+ return true;
+ currentNode = currentNode.parentNode;
+ }
+ return false;
+}
+
+function isDescendantNode(descendant)
+{
+ return isAncestorNode(descendant, this);
+}
+
+function nextSiblingSkippingWhitespace()
+{
+ if (!this)
+ return;
+ var node = this.nextSibling;
+ while (node && node.nodeType === Node.TEXT_NODE && isNodeWhitespace.call(node))
+ node = node.nextSibling;
+ return node;
+}
+
+function previousSiblingSkippingWhitespace()
+{
+ if (!this)
+ return;
+ var node = this.previousSibling;
+ while (node && node.nodeType === Node.TEXT_NODE && isNodeWhitespace.call(node))
+ node = node.previousSibling;
+ return node;
+}
+
+function firstChildSkippingWhitespace()
+{
+ if (!this)
+ return;
+ var node = this.firstChild;
+ while (node && node.nodeType === Node.TEXT_NODE && isNodeWhitespace.call(node))
+ node = nextSiblingSkippingWhitespace.call(node);
+ return node;
+}
+
+function lastChildSkippingWhitespace()
+{
+ if (!this)
+ return;
+ var node = this.lastChild;
+ while (node && node.nodeType === Node.TEXT_NODE && isNodeWhitespace.call(node))
+ node = previousSiblingSkippingWhitespace.call(node);
+ return node;
+}
+
+function traverseNextNode(skipWhitespace, stayWithin)
+{
+ if (!this)
+ return;
+
+ var node = skipWhitespace ? firstChildSkippingWhitespace.call(this) : this.firstChild;
+ if (node)
+ return node;
+
+ if (stayWithin && this === stayWithin)
+ return null;
+
+ node = skipWhitespace ? nextSiblingSkippingWhitespace.call(this) : this.nextSibling;
+ if (node)
+ return node;
+
+ node = this;
+ while (node && !(skipWhitespace ? nextSiblingSkippingWhitespace.call(node) : node.nextSibling) && (!stayWithin || !node.parentNode || node.parentNode !== stayWithin))
+ node = node.parentNode;
+ if (!node)
+ return null;
+
+ return skipWhitespace ? nextSiblingSkippingWhitespace.call(node) : node.nextSibling;
+}
+
+function traversePreviousNode(skipWhitespace, stayWithin)
+{
+ if (!this)
+ return;
+ if (stayWithin && this === stayWithin)
+ return null;
+ var node = skipWhitespace ? previousSiblingSkippingWhitespace.call(this) : this.previousSibling;
+ while (node && (skipWhitespace ? lastChildSkippingWhitespace.call(node) : node.lastChild) )
+ node = skipWhitespace ? lastChildSkippingWhitespace.call(node) : node.lastChild;
+ if (node)
+ return node;
+ return this.parentNode;
+}
+
+function onlyTextChild(ignoreWhitespace)
+{
+ if (!this)
+ return null;
+
+ var firstChild = ignoreWhitespace ? firstChildSkippingWhitespace.call(this) : this.firstChild;
+ if (!firstChild || firstChild.nodeType !== Node.TEXT_NODE)
+ return null;
+
+ var sibling = ignoreWhitespace ? nextSiblingSkippingWhitespace.call(firstChild) : firstChild.nextSibling;
+ return sibling ? null : firstChild;
+}
+
+function nodeTitleInfo(hasChildren, linkify)
+{
+ var info = {title: "", hasChildren: hasChildren};
+
+ switch (this.nodeType) {
+ case Node.DOCUMENT_NODE:
+ info.title = "Document";
+ break;
+
+ case Node.ELEMENT_NODE:
+ info.title = "<span class=\"webkit-html-tag\">&lt;" + this.nodeName.toLowerCase().escapeHTML();
+
+ if (this.hasAttributes()) {
+ for (var i = 0; i < this.attributes.length; ++i) {
+ var attr = this.attributes[i];
+ info.title += " <span class=\"webkit-html-attribute\"><span class=\"webkit-html-attribute-name\">" + attr.name.escapeHTML() + "</span>=&#8203;\"";
+
+ var value = attr.value;
+ if (linkify && (attr.name === "src" || attr.name === "href")) {
+ var value = value.replace(/([\/;:\)\]\}])/g, "$1\u200B");
+ info.title += linkify(attr.value, value, "webkit-html-attribute-value", this.nodeName.toLowerCase() == "a");
+ } else {
+ var value = value.escapeHTML();
+ value = value.replace(/([\/;:\)\]\}])/g, "$1&#8203;");
+ info.title += "<span class=\"webkit-html-attribute-value\">" + value + "</span>";
+ }
+ info.title += "\"</span>";
+ }
+ }
+ info.title += "&gt;</span>&#8203;";
+
+ // If this element only has a single child that is a text node,
+ // just show that text and the closing tag inline rather than
+ // create a subtree for them
+
+ var textChild = onlyTextChild.call(this, Preferences.ignoreWhitespace);
+ var showInlineText = textChild && textChild.textContent.length < Preferences.maxInlineTextChildLength;
+
+ if (showInlineText) {
+ info.title += "<span class=\"webkit-html-text-node\">" + textChild.nodeValue.escapeHTML() + "</span>&#8203;<span class=\"webkit-html-tag\">&lt;/" + this.nodeName.toLowerCase().escapeHTML() + "&gt;</span>";
+ info.hasChildren = false;
+ }
+ break;
+
+ case Node.TEXT_NODE:
+ if (isNodeWhitespace.call(this))
+ info.title = "(whitespace)";
+ else
+ info.title = "\"<span class=\"webkit-html-text-node\">" + this.nodeValue.escapeHTML() + "</span>\"";
+ break
+
+ case Node.COMMENT_NODE:
+ info.title = "<span class=\"webkit-html-comment\">&lt;!--" + this.nodeValue.escapeHTML() + "--&gt;</span>";
+ break;
+
+ case Node.DOCUMENT_TYPE_NODE:
+ info.title = "<span class=\"webkit-html-doctype\">&lt;!DOCTYPE " + this.nodeName;
+ if (this.publicId) {
+ info.title += " PUBLIC \"" + this.publicId + "\"";
+ if (this.systemId)
+ info.title += " \"" + this.systemId + "\"";
+ } else if (this.systemId)
+ info.title += " SYSTEM \"" + this.systemId + "\"";
+ if (this.internalSubset)
+ info.title += " [" + this.internalSubset + "]";
+ info.title += "&gt;</span>";
+ break;
+ default:
+ info.title = this.nodeName.toLowerCase().collapseWhitespace().escapeHTML();
+ }
+
+ return info;
+}
+
+function getDocumentForNode(node) {
+ return node.nodeType == Node.DOCUMENT_NODE ? node : node.ownerDocument;
+}
+
+function parentNode(node) {
+ return node.parentNode;
+}
+
+Number.secondsToString = function(seconds, formatterFunction, higherResolution)
+{
+ if (!formatterFunction)
+ formatterFunction = String.sprintf;
+
+ var ms = seconds * 1000;
+ if (higherResolution && ms < 1000)
+ return formatterFunction("%.3fms", ms);
+ else if (ms < 1000)
+ return formatterFunction("%.0fms", ms);
+
+ if (seconds < 60)
+ return formatterFunction("%.2fs", seconds);
+
+ var minutes = seconds / 60;
+ if (minutes < 60)
+ return formatterFunction("%.1fmin", minutes);
+
+ var hours = minutes / 60;
+ if (hours < 24)
+ return formatterFunction("%.1fhrs", hours);
+
+ var days = hours / 24;
+ return formatterFunction("%.1f days", days);
+}
+
+Number.bytesToString = function(bytes, formatterFunction, higherResolution)
+{
+ if (!formatterFunction)
+ formatterFunction = String.sprintf;
+ if (typeof higherResolution === "undefined")
+ higherResolution = true;
+
+ if (bytes < 1024)
+ return formatterFunction("%.0fB", bytes);
+
+ var kilobytes = bytes / 1024;
+ if (higherResolution && kilobytes < 1024)
+ return formatterFunction("%.2fKB", kilobytes);
+ else if (kilobytes < 1024)
+ return formatterFunction("%.0fKB", kilobytes);
+
+ var megabytes = kilobytes / 1024;
+ if (higherResolution)
+ return formatterFunction("%.3fMB", megabytes);
+ else
+ return formatterFunction("%.0fMB", megabytes);
+}
+
+Number.constrain = function(num, min, max)
+{
+ if (num < min)
+ num = min;
+ else if (num > max)
+ num = max;
+ return num;
+}
+
+HTMLTextAreaElement.prototype.moveCursorToEnd = function()
+{
+ var length = this.value.length;
+ this.setSelectionRange(length, length);
+}
+
+Array.prototype.remove = function(value, onlyFirst)
+{
+ if (onlyFirst) {
+ var index = this.indexOf(value);
+ if (index !== -1)
+ this.splice(index, 1);
+ return;
+ }
+
+ var length = this.length;
+ for (var i = 0; i < length; ++i) {
+ if (this[i] === value)
+ this.splice(i, 1);
+ }
+}
+
+function insertionIndexForObjectInListSortedByFunction(anObject, aList, aFunction)
+{
+ // indexOf returns (-lowerBound - 1). Taking (-result - 1) works out to lowerBound.
+ return (-indexOfObjectInListSortedByFunction(anObject, aList, aFunction) - 1);
+}
+
+function indexOfObjectInListSortedByFunction(anObject, aList, aFunction)
+{
+ var first = 0;
+ var last = aList.length - 1;
+ var floor = Math.floor;
+ var mid, c;
+
+ while (first <= last) {
+ mid = floor((first + last) / 2);
+ c = aFunction(anObject, aList[mid]);
+
+ if (c > 0)
+ first = mid + 1;
+ else if (c < 0)
+ last = mid - 1;
+ else {
+ // Return the first occurance of an item in the list.
+ while (mid > 0 && aFunction(anObject, aList[mid - 1]) === 0)
+ mid--;
+ first = mid;
+ break;
+ }
+ }
+
+ // By returning 1 less than the negative lower search bound, we can reuse this function
+ // for both indexOf and insertionIndexFor, with some simple arithmetic.
+ return (-first - 1);
+}
+
+String.sprintf = function(format)
+{
+ return String.vsprintf(format, Array.prototype.slice.call(arguments, 1));
+}
+
+String.tokenizeFormatString = function(format)
+{
+ var tokens = [];
+ var substitutionIndex = 0;
+
+ function addStringToken(str)
+ {
+ tokens.push({ type: "string", value: str });
+ }
+
+ function addSpecifierToken(specifier, precision, substitutionIndex)
+ {
+ tokens.push({ type: "specifier", specifier: specifier, precision: precision, substitutionIndex: substitutionIndex });
+ }
+
+ var index = 0;
+ for (var precentIndex = format.indexOf("%", index); precentIndex !== -1; precentIndex = format.indexOf("%", index)) {
+ addStringToken(format.substring(index, precentIndex));
+ index = precentIndex + 1;
+
+ if (format[index] === "%") {
+ addStringToken("%");
+ ++index;
+ continue;
+ }
+
+ if (!isNaN(format[index])) {
+ // The first character is a number, it might be a substitution index.
+ var number = parseInt(format.substring(index));
+ while (!isNaN(format[index]))
+ ++index;
+ // If the number is greater than zero and ends with a "$",
+ // then this is a substitution index.
+ if (number > 0 && format[index] === "$") {
+ substitutionIndex = (number - 1);
+ ++index;
+ }
+ }
+
+ var precision = -1;
+ if (format[index] === ".") {
+ // This is a precision specifier. If no digit follows the ".",
+ // then the precision should be zero.
+ ++index;
+ precision = parseInt(format.substring(index));
+ if (isNaN(precision))
+ precision = 0;
+ while (!isNaN(format[index]))
+ ++index;
+ }
+
+ addSpecifierToken(format[index], precision, substitutionIndex);
+
+ ++substitutionIndex;
+ ++index;
+ }
+
+ addStringToken(format.substring(index));
+
+ return tokens;
+}
+
+String.standardFormatters = {
+ d: function(substitution)
+ {
+ substitution = parseInt(substitution);
+ return !isNaN(substitution) ? substitution : 0;
+ },
+
+ f: function(substitution, token)
+ {
+ substitution = parseFloat(substitution);
+ if (substitution && token.precision > -1)
+ substitution = substitution.toFixed(token.precision);
+ return !isNaN(substitution) ? substitution : (token.precision > -1 ? Number(0).toFixed(token.precision) : 0);
+ },
+
+ s: function(substitution)
+ {
+ return substitution;
+ },
+};
+
+String.vsprintf = function(format, substitutions)
+{
+ return String.format(format, substitutions, String.standardFormatters, "", function(a, b) { return a + b; }).formattedResult;
+}
+
+String.format = function(format, substitutions, formatters, initialValue, append)
+{
+ if (!format || !substitutions || !substitutions.length)
+ return { formattedResult: append(initialValue, format), unusedSubstitutions: substitutions };
+
+ function prettyFunctionName()
+ {
+ return "String.format(\"" + format + "\", \"" + substitutions.join("\", \"") + "\")";
+ }
+
+ function warn(msg)
+ {
+ console.warn(prettyFunctionName() + ": " + msg);
+ }
+
+ function error(msg)
+ {
+ console.error(prettyFunctionName() + ": " + msg);
+ }
+
+ var result = initialValue;
+ var tokens = String.tokenizeFormatString(format);
+ var usedSubstitutionIndexes = {};
+
+ for (var i = 0; i < tokens.length; ++i) {
+ var token = tokens[i];
+
+ if (token.type === "string") {
+ result = append(result, token.value);
+ continue;
+ }
+
+ if (token.type !== "specifier") {
+ error("Unknown token type \"" + token.type + "\" found.");
+ continue;
+ }
+
+ if (token.substitutionIndex >= substitutions.length) {
+ // If there are not enough substitutions for the current substitutionIndex
+ // just output the format specifier literally and move on.
+ error("not enough substitution arguments. Had " + substitutions.length + " but needed " + (token.substitutionIndex + 1) + ", so substitution was skipped.");
+ result = append(result, "%" + (token.precision > -1 ? token.precision : "") + token.specifier);
+ continue;
+ }
+
+ usedSubstitutionIndexes[token.substitutionIndex] = true;
+
+ if (!(token.specifier in formatters)) {
+ // Encountered an unsupported format character, treat as a string.
+ warn("unsupported format character \u201C" + token.specifier + "\u201D. Treating as a string.");
+ result = append(result, substitutions[token.substitutionIndex]);
+ continue;
+ }
+
+ result = append(result, formatters[token.specifier](substitutions[token.substitutionIndex], token));
+ }
+
+ var unusedSubstitutions = [];
+ for (var i = 0; i < substitutions.length; ++i) {
+ if (i in usedSubstitutionIndexes)
+ continue;
+ unusedSubstitutions.push(substitutions[i]);
+ }
+
+ return { formattedResult: result, unusedSubstitutions: unusedSubstitutions };
+}
diff --git a/chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.exe b/chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.exe
new file mode 100644
index 0000000..a979e1b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.exe
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.pdb b/chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.pdb
new file mode 100644
index 0000000..aaa155a
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/servers/chrome_launcher.pdb
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.dll b/chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.dll
new file mode 100644
index 0000000..f7eca43
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.pdb b/chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.pdb
new file mode 100644
index 0000000..6326dce
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/servers/npchrome_tab.pdb
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/setup.pdb b/chrome_frame/tools/test/reference_build/chrome/setup.pdb
new file mode 100644
index 0000000..046e7a1
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/setup.pdb
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/syncapi.dll b/chrome_frame/tools/test/reference_build/chrome/syncapi.dll
new file mode 100644
index 0000000..e3f7e31
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/syncapi.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/themes/default.dll b/chrome_frame/tools/test/reference_build/chrome/themes/default.dll
new file mode 100644
index 0000000..0261aac
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/themes/default.dll
Binary files differ
diff --git a/chrome_frame/tools/test/reference_build/chrome/wow_helper.exe b/chrome_frame/tools/test/reference_build/chrome/wow_helper.exe
new file mode 100644
index 0000000..f9bfb4b
--- /dev/null
+++ b/chrome_frame/tools/test/reference_build/chrome/wow_helper.exe
Binary files differ
diff --git a/chrome_frame/unittest_precompile.cc b/chrome_frame/unittest_precompile.cc
new file mode 100644
index 0000000..9fb41a7
--- /dev/null
+++ b/chrome_frame/unittest_precompile.cc
@@ -0,0 +1,5 @@
+// Copyright (c) 2009 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 "chrome_frame/unittest_precompile.h"
diff --git a/chrome_frame/unittest_precompile.h b/chrome_frame/unittest_precompile.h
new file mode 100644
index 0000000..814f508
--- /dev/null
+++ b/chrome_frame/unittest_precompile.h
@@ -0,0 +1,18 @@
+// Copyright (c) 2009 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.
+
+// precompiled.h : include file for standard system include files,
+// or project specific include files that are used frequently,
+// but are changed infrequently
+
+#ifndef CHROME_FRAME_UNITTEST_PRECOMPILE_H_
+#define CHROME_FRAME_UNITTEST_PRECOMPILE_H_
+
+#include "resource.h"
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <atlctl.h>
+
+#endif // CHROME_FRAME_UNITTEST_PRECOMPILE_H_
diff --git a/chrome_frame/urlmon_upload_data_stream.cc b/chrome_frame/urlmon_upload_data_stream.cc
new file mode 100644
index 0000000..af7a1de
--- /dev/null
+++ b/chrome_frame/urlmon_upload_data_stream.cc
@@ -0,0 +1,99 @@
+// Copyright (c) 2009 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 "chrome_frame/urlmon_upload_data_stream.h"
+
+#include "net/base/io_buffer.h"
+
+void UrlmonUploadDataStream::Initialize(net::UploadData* upload_data) {
+ upload_data_ = upload_data;
+ request_body_stream_.reset(new net::UploadDataStream(upload_data));
+}
+
+STDMETHODIMP UrlmonUploadDataStream::Read(void* pv, ULONG cb, ULONG* read) {
+ if (pv == NULL) {
+ NOTREACHED();
+ return E_POINTER;
+ }
+
+ // Have we already read past the end of the stream?
+ if (request_body_stream_->position() >= request_body_stream_->size()) {
+ if (read) {
+ *read = 0;
+ }
+ return S_FALSE;
+ }
+
+ uint64 total_bytes_to_copy = std::min(static_cast<uint64>(cb),
+ request_body_stream_->size() - request_body_stream_->position());
+ uint64 initial_position = request_body_stream_->position();
+
+ uint64 bytes_copied = 0;
+
+ char* write_pointer = reinterpret_cast<char*>(pv);
+ while (bytes_copied < total_bytes_to_copy) {
+ net::IOBuffer* buf = request_body_stream_->buf();
+
+ // Make sure our length doesn't run past the end of the available data.
+ size_t bytes_to_copy_now = static_cast<size_t>(
+ std::min(static_cast<uint64>(request_body_stream_->buf_len()),
+ total_bytes_to_copy - bytes_copied));
+
+ memcpy(write_pointer, buf->data(), bytes_to_copy_now);
+
+ // Advance our copy tally
+ bytes_copied += bytes_to_copy_now;
+
+ // Advance our write pointer
+ write_pointer += bytes_to_copy_now;
+
+ // Advance the UploadDataStream read pointer:
+ request_body_stream_->DidConsume(bytes_to_copy_now);
+ }
+
+ DCHECK(bytes_copied == total_bytes_to_copy);
+ DCHECK(request_body_stream_->position() ==
+ initial_position + total_bytes_to_copy);
+
+ if (read) {
+ *read = static_cast<ULONG>(total_bytes_to_copy);
+ }
+
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUploadDataStream::Seek(LARGE_INTEGER move, DWORD origin,
+ ULARGE_INTEGER* new_pos) {
+ // UploadDataStream is really not very seek-able, so for now allow
+ // STREAM_SEEK_SETs to work with a 0 offset, but fail on everything else.
+ if (origin == STREAM_SEEK_SET && move.QuadPart == 0) {
+ if (request_body_stream_->position() != 0) {
+ request_body_stream_.reset(new net::UploadDataStream(upload_data_));
+ }
+ if (new_pos) {
+ new_pos->QuadPart = 0;
+ }
+ return S_OK;
+ }
+
+ DCHECK(false) << __FUNCTION__;
+ return STG_E_INVALIDFUNCTION;
+}
+
+STDMETHODIMP UrlmonUploadDataStream::Stat(STATSTG *stat_stg,
+ DWORD grf_stat_flag) {
+ if (stat_stg == NULL)
+ return E_POINTER;
+
+ memset(stat_stg, 0, sizeof(STATSTG));
+ if (0 == (grf_stat_flag & STATFLAG_NONAME)) {
+ const wchar_t kStreamBuffer[] = L"PostStream";
+ stat_stg->pwcsName =
+ static_cast<wchar_t*>(::CoTaskMemAlloc(sizeof(kStreamBuffer)));
+ lstrcpy(stat_stg->pwcsName, kStreamBuffer);
+ }
+ stat_stg->type = STGTY_STREAM;
+ stat_stg->cbSize.QuadPart = upload_data_->GetContentLength();
+ return S_OK;
+}
diff --git a/chrome_frame/urlmon_upload_data_stream.h b/chrome_frame/urlmon_upload_data_stream.h
new file mode 100644
index 0000000..eacb433
--- /dev/null
+++ b/chrome_frame/urlmon_upload_data_stream.h
@@ -0,0 +1,89 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_URLMON_UPLOAD_DATA_STREAM_H_
+#define CHROME_FRAME_URLMON_UPLOAD_DATA_STREAM_H_
+
+#include <urlmon.h>
+#include <atlbase.h>
+#include <atlcom.h>
+
+#include "base/logging.h"
+#include "base/ref_counted.h"
+
+#include "net/base/upload_data.h"
+#include "net/base/upload_data_stream.h"
+
+// Provides an IStream interface to the very different UploadDataStream
+// implementation.
+class UrlmonUploadDataStream : public CComObjectRoot,
+ public IStream {
+ public:
+ UrlmonUploadDataStream() {}
+
+ BEGIN_COM_MAP(UrlmonUploadDataStream)
+ COM_INTERFACE_ENTRY(ISequentialStream)
+ COM_INTERFACE_ENTRY(IStream)
+ END_COM_MAP()
+
+ void Initialize(net::UploadData* upload_data);
+
+ // Partial implementation of IStream.
+ STDMETHOD(Read)(void* pv, ULONG cb, ULONG* read);
+
+ // E_NOTIMPL the rest and DCHECK if they get called (could also use
+ // IStreamImpl but we'd lose the DCHECKS().
+ STDMETHOD(Write)(const void * buffer, ULONG size, ULONG* size_written) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(CopyTo)(IStream* stream, ULARGE_INTEGER cb, ULARGE_INTEGER* read,
+ ULARGE_INTEGER* written) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Seek)(LARGE_INTEGER move, DWORD origin, ULARGE_INTEGER* new_pos);
+
+ STDMETHOD(SetSize)(ULARGE_INTEGER new_size) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Commit)(DWORD flags) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Revert)() {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(LockRegion)(ULARGE_INTEGER offset, ULARGE_INTEGER cb,
+ DWORD type) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(UnlockRegion)(ULARGE_INTEGER offset, ULARGE_INTEGER cb,
+ DWORD type) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Stat)(STATSTG *pstatstg, DWORD grfStatFlag);
+
+ STDMETHOD(Clone)(IStream** stream) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ private:
+ scoped_refptr<net::UploadData> upload_data_;
+ scoped_ptr<net::UploadDataStream> request_body_stream_;
+};
+
+#endif // CHROME_FRAME_URLMON_UPLOAD_DATA_STREAM_H_
diff --git a/chrome_frame/urlmon_upload_data_stream_unittest.cc b/chrome_frame/urlmon_upload_data_stream_unittest.cc
new file mode 100644
index 0000000..a34ba2c
--- /dev/null
+++ b/chrome_frame/urlmon_upload_data_stream_unittest.cc
@@ -0,0 +1,162 @@
+// Copyright (c) 2009 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 "gtest/gtest.h"
+
+#include "base/ref_counted.h"
+#include "base/scoped_comptr_win.h"
+#include "chrome_frame/urlmon_upload_data_stream.h"
+
+TEST(UrlmonUploadDataStreamTest, TestBasicRead) {
+ char random_string[] = "some random data, no really this totally random";
+ int random_string_length = strlen(random_string);
+ scoped_refptr<net::UploadData> upload_data = new net::UploadData();
+ upload_data->AppendBytes(random_string, random_string_length);
+
+ CComObject<UrlmonUploadDataStream>* upload_stream = NULL;
+ HRESULT hr =
+ CComObject<UrlmonUploadDataStream>::CreateInstance(&upload_stream);
+ ASSERT_TRUE(SUCCEEDED(hr));
+
+ upload_stream->Initialize(upload_data.get());
+ ScopedComPtr<IStream> upload_istream(upload_stream);
+
+ char buffer[500];
+ memset(buffer, 0, 500);
+ ULONG bytes_read = 0;
+ hr = upload_istream->Read(buffer, 500, &bytes_read);
+
+ EXPECT_TRUE(SUCCEEDED(hr));
+ EXPECT_EQ(bytes_read, random_string_length);
+ EXPECT_TRUE(strcmp(buffer, random_string) == 0);
+
+ char buffer2[500];
+ memset(buffer2, 0, 500);
+ ULONG bytes_read2 = 0;
+ hr = upload_istream->Read(buffer2, 500, &bytes_read2);
+
+ EXPECT_EQ(S_FALSE, hr);
+ EXPECT_EQ(bytes_read2, 0);
+ EXPECT_FALSE(strcmp(buffer2, random_string) == 0);
+}
+
+TEST(UrlmonUploadDataStreamTest, TestBigRead) {
+ const size_t kBigBufferLength = 100000;
+ char big_buffer[kBigBufferLength];
+ memset(big_buffer, 'a', kBigBufferLength);
+
+ scoped_refptr<net::UploadData> upload_data = new net::UploadData();
+ upload_data->AppendBytes(big_buffer, kBigBufferLength);
+
+ CComObject<UrlmonUploadDataStream>* upload_stream = NULL;
+ HRESULT hr =
+ CComObject<UrlmonUploadDataStream>::CreateInstance(&upload_stream);
+ ASSERT_TRUE(SUCCEEDED(hr));
+
+ upload_stream->Initialize(upload_data.get());
+ ScopedComPtr<IStream> upload_istream(upload_stream);
+
+ char big_rcv_buffer[kBigBufferLength];
+ int write_pos = 0;
+ ULONG bytes_read = 0;
+ hr = E_UNEXPECTED;
+
+ while ((hr = upload_istream->Read(&big_rcv_buffer[write_pos],
+ kBigBufferLength,
+ &bytes_read)) != S_FALSE) {
+ EXPECT_TRUE(SUCCEEDED(hr));
+ EXPECT_GT(bytes_read, static_cast<ULONG>(0));
+
+ write_pos += bytes_read;
+ bytes_read = 0;
+ }
+
+ EXPECT_EQ(S_FALSE, hr);
+ EXPECT_TRUE((write_pos + bytes_read) == kBigBufferLength);
+ EXPECT_EQ(0, memcmp(big_buffer, big_rcv_buffer, kBigBufferLength));
+}
+
+TEST(UrlmonUploadDataStreamTest, TestStat) {
+ char random_string[] = "some random data, no really this totally random";
+ int random_string_length = strlen(random_string);
+ scoped_refptr<net::UploadData> upload_data = new net::UploadData();
+ upload_data->AppendBytes(random_string, random_string_length);
+
+ CComObject<UrlmonUploadDataStream>* upload_stream = NULL;
+ HRESULT hr =
+ CComObject<UrlmonUploadDataStream>::CreateInstance(&upload_stream);
+ ASSERT_TRUE(SUCCEEDED(hr));
+
+ upload_stream->Initialize(upload_data.get());
+ ScopedComPtr<IStream> upload_istream(upload_stream);
+
+ STATSTG statstg;
+ hr = upload_stream->Stat(&statstg, STATFLAG_NONAME);
+ EXPECT_TRUE(SUCCEEDED(hr));
+ EXPECT_EQ(static_cast<LONGLONG>(random_string_length),
+ statstg.cbSize.QuadPart);
+}
+
+TEST(UrlmonUploadDataStreamTest, TestRepeatedRead) {
+ char random_string[] = "some random data, no really this totally random";
+ int random_string_length = strlen(random_string);
+ scoped_refptr<net::UploadData> upload_data = new net::UploadData();
+ upload_data->AppendBytes(random_string, random_string_length);
+
+ CComObject<UrlmonUploadDataStream>* upload_stream = NULL;
+ HRESULT hr =
+ CComObject<UrlmonUploadDataStream>::CreateInstance(&upload_stream);
+ ASSERT_TRUE(SUCCEEDED(hr));
+
+ upload_stream->Initialize(upload_data.get());
+ ScopedComPtr<IStream> upload_istream(upload_stream);
+
+ char buffer[500];
+ memset(buffer, 0, 500);
+ ULONG bytes_read = 0;
+ hr = upload_istream->Read(buffer, 500, &bytes_read);
+
+ EXPECT_TRUE(SUCCEEDED(hr));
+ EXPECT_EQ(bytes_read, random_string_length);
+ EXPECT_EQ(0, strcmp(buffer, random_string));
+
+ char buffer2[500];
+ memset(buffer2, 0, 500);
+ ULONG bytes_read2 = 0;
+
+ for (int i = 0; i < 10; i++) {
+ hr = upload_istream->Read(buffer2, 500, &bytes_read2);
+ EXPECT_EQ(S_FALSE, hr);
+ EXPECT_EQ(bytes_read2, 0);
+ EXPECT_NE(0, strcmp(buffer2, random_string));
+ }
+}
+
+TEST(UrlmonUploadDataStreamTest, TestZeroRead) {
+ char random_string[] = "some random data, no really this totally random";
+ int random_string_length = strlen(random_string);
+ scoped_refptr<net::UploadData> upload_data = new net::UploadData();
+ upload_data->AppendBytes(random_string, random_string_length);
+
+ CComObject<UrlmonUploadDataStream>* upload_stream = NULL;
+ HRESULT hr =
+ CComObject<UrlmonUploadDataStream>::CreateInstance(&upload_stream);
+ ASSERT_TRUE(SUCCEEDED(hr));
+
+ upload_stream->Initialize(upload_data.get());
+ ScopedComPtr<IStream> upload_istream(upload_stream);
+
+ char buffer[500];
+ memset(buffer, 0, 500);
+ ULONG bytes_read = 42;
+ hr = upload_istream->Read(&buffer[0], 0, &bytes_read);
+
+ EXPECT_EQ(S_OK, hr);
+ EXPECT_EQ(0, bytes_read);
+
+ char buffer2[500];
+ memset(&buffer2[0], 0, 500);
+ EXPECT_EQ(0, memcmp(buffer, buffer2, 500));
+}
+
diff --git a/chrome_frame/urlmon_url_request.cc b/chrome_frame/urlmon_url_request.cc
new file mode 100644
index 0000000..f51da418
--- /dev/null
+++ b/chrome_frame/urlmon_url_request.cc
@@ -0,0 +1,751 @@
+// Copyright (c) 2009 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 "chrome_frame/urlmon_url_request.h"
+
+#include <wininet.h>
+
+#include "base/scoped_ptr.h"
+#include "base/string_util.h"
+#include "base/logging.h"
+#include "chrome_frame/urlmon_upload_data_stream.h"
+#include "net/http/http_util.h"
+#include "net/http/http_response_headers.h"
+
+static const LARGE_INTEGER kZero = {0};
+static const ULARGE_INTEGER kUnsignedZero = {0};
+int UrlmonUrlRequest::instance_count_ = 0;
+const char kXFrameOptionsHeader[] = "X-Frame-Options";
+
+UrlmonUrlRequest::UrlmonUrlRequest()
+ : pending_read_size_(0),
+ status_(URLRequestStatus::FAILED, net::ERR_FAILED),
+ thread_(PlatformThread::CurrentId()),
+ is_request_started_(false),
+ post_data_len_(0),
+ redirect_status_(0),
+ parent_window_(NULL) {
+ DLOG(INFO) << StringPrintf("Created request. Obj: %X", this)
+ << " Count: " << ++instance_count_;
+}
+
+UrlmonUrlRequest::~UrlmonUrlRequest() {
+ DLOG(INFO) << StringPrintf("Deleted request. Obj: %X", this)
+ << " Count: " << --instance_count_;
+}
+
+bool UrlmonUrlRequest::Start() {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+
+ status_.set_status(URLRequestStatus::IO_PENDING);
+ HRESULT hr = StartAsyncDownload();
+ if (FAILED(hr)) {
+ // Do not call EndRequest() here since it will attempt to free references
+ // that have not been established.
+ status_.set_os_error(HresultToNetError(hr));
+ status_.set_status(URLRequestStatus::FAILED);
+ DLOG(ERROR) << "StartAsyncDownload failed";
+ OnResponseEnd(status_);
+ return false;
+ }
+
+ // Take a self reference to maintain COM lifetime. This will be released
+ // in EndRequest. Set a flag indicating that we have an additional
+ // reference here
+ is_request_started_ = true;
+ AddRef();
+ request_handler()->AddRequest(this);
+ return true;
+}
+
+void UrlmonUrlRequest::Stop() {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+
+ if (binding_) {
+ binding_->Abort();
+ } else {
+ status_.set_status(URLRequestStatus::CANCELED);
+ status_.set_os_error(net::ERR_FAILED);
+ EndRequest();
+ }
+}
+
+bool UrlmonUrlRequest::Read(int bytes_to_read) {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X", url().c_str(), this);
+
+ // Send cached data if available.
+ CComObjectStackEx<SendStream> send_stream;
+ send_stream.Initialize(this);
+ size_t bytes_copied = 0;
+ if (cached_data_.is_valid() && cached_data_.Read(&send_stream, bytes_to_read,
+ &bytes_copied)) {
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X - bytes read from cache: %d",
+ url().c_str(), this, bytes_copied);
+ return true;
+ }
+
+ // if the request is finished or there's nothing more to read
+ // then end the request
+ if (!status_.is_io_pending() || !binding_) {
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X. Response finished. Status: %d",
+ url().c_str(), this, status_.status());
+ EndRequest();
+ return true;
+ }
+
+ pending_read_size_ = bytes_to_read;
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X", url().c_str(), this) <<
+ "- Read pending for: " << bytes_to_read;
+ return true;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnStartBinding(
+ DWORD reserved, IBinding *binding) {
+ binding_ = binding;
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::GetPriority(LONG *priority) {
+ if (!priority)
+ return E_POINTER;
+ *priority = THREAD_PRIORITY_NORMAL;
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnLowResource(DWORD reserved) {
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnProgress(ULONG progress, ULONG max_progress,
+ ULONG status_code, LPCWSTR status_text) {
+ switch (status_code) {
+ case BINDSTATUS_REDIRECTING:
+ DCHECK(status_text != NULL);
+ DLOG(INFO) << "URL: " << url() << " redirected to "
+ << status_text;
+ redirect_url_ = status_text;
+ // Fetch the redirect status as they aren't all equal (307 in particular
+ // retains the HTTP request verb).
+ redirect_status_ = GetHttpResponseStatus();
+ break;
+
+ default:
+ DLOG(INFO) << " Obj: " << std::hex << this << " OnProgress(" << url()
+ << StringPrintf(L") code: %i status: %ls", status_code, status_text);
+ break;
+ }
+
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnStopBinding(HRESULT result, LPCWSTR error) {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X", url().c_str(), this) <<
+ " - Request stopped, Result: " << std::hex << result <<
+ " Status: " << status_.status();
+ if (FAILED(result)) {
+ status_.set_status(URLRequestStatus::FAILED);
+ status_.set_os_error(HresultToNetError(result));
+ EndRequest();
+ } else {
+ status_.set_status(URLRequestStatus::SUCCESS);
+ status_.set_os_error(0);
+ }
+
+ // Release these variables after reporting EndRequest since we might need to
+ // access their state.
+ binding_ = NULL;
+ bind_context_ = NULL;
+
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::GetBindInfo(DWORD* bind_flags,
+ BINDINFO *bind_info) {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+
+ if ((bind_info == NULL) || (bind_info->cbSize == 0) || (bind_flags == NULL))
+ return E_INVALIDARG;
+
+ *bind_flags = BINDF_ASYNCHRONOUS | BINDF_ASYNCSTORAGE | BINDF_PULLDATA;
+ if (LowerCaseEqualsASCII(method(), "get")) {
+ bind_info->dwBindVerb = BINDVERB_GET;
+ } else if (LowerCaseEqualsASCII(method(), "post")) {
+ bind_info->dwBindVerb = BINDVERB_POST;
+
+ // Bypass caching proxies on POSTs and avoid writing responses to POST
+ // requests to the browser's cache.
+ *bind_flags |= BINDF_GETNEWESTVERSION | BINDF_NOWRITECACHE |
+ BINDF_PRAGMA_NO_CACHE;
+
+ // Initialize the STGMEDIUM.
+ memset(&bind_info->stgmedData, 0, sizeof(STGMEDIUM));
+ bind_info->grfBindInfoF = 0;
+ bind_info->szCustomVerb = NULL;
+
+ scoped_refptr<net::UploadData> upload_data(upload_data());
+ post_data_len_ = upload_data.get() ? upload_data->GetContentLength() : 0;
+ if (post_data_len_) {
+ DLOG(INFO) << " Obj: " << std::hex << this << " POST request with "
+ << Int64ToString(post_data_len_) << " bytes";
+ CComObject<UrlmonUploadDataStream>* upload_stream = NULL;
+ HRESULT hr =
+ CComObject<UrlmonUploadDataStream>::CreateInstance(&upload_stream);
+ if (FAILED(hr)) {
+ NOTREACHED();
+ return hr;
+ }
+ upload_stream->Initialize(upload_data.get());
+
+ // Fill the STGMEDIUM with the data to post
+ bind_info->stgmedData.tymed = TYMED_ISTREAM;
+ bind_info->stgmedData.pstm = static_cast<IStream*>(upload_stream);
+ bind_info->stgmedData.pstm->AddRef();
+ } else {
+ DLOG(INFO) << " Obj: " << std::hex << this
+ << "POST request with no data!";
+ }
+ } else {
+ NOTREACHED() << "Unknown HTTP method.";
+ status_.set_status(URLRequestStatus::FAILED);
+ status_.set_os_error(net::ERR_INVALID_URL);
+ EndRequest();
+ return E_FAIL;
+ }
+
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnDataAvailable(DWORD flags, DWORD size,
+ FORMATETC* formatetc,
+ STGMEDIUM* storage) {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X - Bytes available: %d",
+ url().c_str(), this, size);
+
+ if (!storage || (storage->tymed != TYMED_ISTREAM)) {
+ NOTREACHED();
+ return E_INVALIDARG;
+ }
+
+ IStream* read_stream = storage->pstm;
+ if (!read_stream) {
+ NOTREACHED();
+ return E_UNEXPECTED;
+ }
+
+ HRESULT hr = S_OK;
+ if (BSCF_FIRSTDATANOTIFICATION & flags) {
+ DCHECK(!cached_data_.is_valid());
+ cached_data_.Create();
+ }
+
+ // Always read data into cache. We have to read all the data here at this
+ // time or it won't be available later. Since the size of the data could
+ // be more than pending read size, it's not straightforward (or might even
+ // be impossible) to implement a true data pull model.
+ size_t bytes_available = 0;
+ cached_data_.Append(read_stream, &bytes_available);
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X", url().c_str(), this) <<
+ " - Bytes read into cache: " << bytes_available;
+
+ if (pending_read_size_) {
+ CComObjectStackEx<SendStream> send_stream;
+ send_stream.Initialize(this);
+ cached_data_.Read(&send_stream, pending_read_size_, &pending_read_size_);
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X", url().c_str(), this) <<
+ " - size read: " << pending_read_size_;
+ pending_read_size_ = 0;
+ } else {
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X", url().c_str(), this) <<
+ " - waiting for remote read";
+ }
+
+ if (BSCF_LASTDATANOTIFICATION & flags) {
+ status_.set_status(URLRequestStatus::SUCCESS);
+ DLOG(INFO) << StringPrintf("URL: %s Obj: %X", url().c_str(), this) <<
+ " - end of data.";
+ }
+
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnObjectAvailable(REFIID iid, IUnknown *object) {
+ // We are calling BindToStorage on the moniker we should always get called
+ // back on OnDataAvailable and should never get OnObjectAvailable
+ NOTREACHED();
+ return E_NOTIMPL;
+}
+
+STDMETHODIMP UrlmonUrlRequest::BeginningTransaction(const wchar_t* url,
+ const wchar_t* current_headers, DWORD reserved,
+ wchar_t** additional_headers) {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+ if (!additional_headers) {
+ NOTREACHED();
+ return E_POINTER;
+ }
+
+ DLOG(INFO) << "URL: " << url << " Obj: " << std::hex << this <<
+ " - Request headers: \n" << current_headers;
+
+ HRESULT hr = S_OK;
+
+ std::string new_headers;
+ if (post_data_len_ > 0) {
+ // Tack on the Content-Length header since when using an IStream type
+ // STGMEDIUM, it looks like it doesn't get set for us :(
+ new_headers = StringPrintf("Content-Length: %s\r\n",
+ Int64ToString(post_data_len_).c_str());
+ }
+
+ if (!extra_headers().empty()) {
+ // TODO(robertshield): We may need to sanitize headers on POST here.
+ new_headers += extra_headers();
+ }
+
+ if (!referrer().empty()) {
+ // Referrer is famously misspelled in HTTP:
+ new_headers += StringPrintf("Referer: %s\r\n", referrer().c_str());
+ }
+
+ if (!new_headers.empty()) {
+ *additional_headers = reinterpret_cast<wchar_t*>(
+ CoTaskMemAlloc((new_headers.size() + 1) * sizeof(wchar_t)));
+
+ if (*additional_headers == NULL) {
+ NOTREACHED();
+ hr = E_OUTOFMEMORY;
+ } else {
+ lstrcpynW(*additional_headers, ASCIIToWide(new_headers).c_str(),
+ new_headers.size());
+ }
+ }
+
+ return hr;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnResponse(DWORD dwResponseCode,
+ const wchar_t* response_headers, const wchar_t* request_headers,
+ wchar_t** additional_headers) {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_);
+
+ std::string raw_headers = WideToUTF8(response_headers);
+
+ // Security check for frame busting headers. We don't honor the headers
+ // as-such, but instead simply kill requests which we've been asked to
+ // look for. This puts the onus on the user of the UrlRequest to specify
+ // whether or not requests should be inspected. For ActiveDocuments, the
+ // answer is "no", since WebKit's detection/handling is sufficient and since
+ // ActiveDocuments cannot be hosted as iframes. For NPAPI and ActiveX
+ // documents, the Initialize() function of the PluginUrlRequest object
+ // allows them to specify how they'd like requests handled. Both should
+ // set enable_frame_busting_ to true to avoid CSRF attacks.
+ // Should WebKit's handling of this ever change, we will need to re-visit
+ // how and when frames are killed to better mirror a policy which may
+ // do something other than kill the sub-document outright.
+
+ // NOTE(slightlyoff): We don't use net::HttpResponseHeaders here because
+ // of lingering ICU/base_noicu issues.
+ if (frame_busting_enabled_ &&
+ net::HttpUtil::HasHeader(raw_headers, kXFrameOptionsHeader)) {
+ DLOG(ERROR) << "X-Frame-Options header detected, navigation canceled";
+ return E_FAIL;
+ }
+
+ std::wstring url_for_persistent_cookies =
+ redirect_url_.empty() ? UTF8ToWide(url()) : redirect_url_;
+
+ std::string persistent_cookies;
+
+ DWORD cookie_size = 0; // NOLINT
+ InternetGetCookie(url_for_persistent_cookies.c_str(), NULL, NULL,
+ &cookie_size);
+ if (cookie_size) {
+ scoped_ptr<wchar_t> cookies(new wchar_t[cookie_size + 1]);
+ if (!InternetGetCookie(url_for_persistent_cookies.c_str(), NULL,
+ cookies.get(), &cookie_size)) {
+ NOTREACHED() << "InternetGetCookie failed. Error: " << GetLastError();
+ } else {
+ persistent_cookies = WideToUTF8(cookies.get());
+ }
+ }
+
+ OnResponseStarted("",
+ raw_headers.c_str(),
+ 0,
+ base::Time(),
+ persistent_cookies,
+ redirect_url_.empty() ? std::string() :
+ WideToUTF8(redirect_url_),
+ redirect_status_);
+
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::GetWindow(const GUID& guid_reason,
+ HWND* parent_window) {
+ if (!parent_window) {
+ return E_INVALIDARG;
+ }
+
+#ifndef NDEBUG
+ wchar_t guid[40] = {0};
+ ::StringFromGUID2(guid_reason, guid, arraysize(guid));
+
+ DLOG(INFO) << " Obj: " << std::hex << this << " GetWindow: " <<
+ (guid_reason == IID_IAuthenticate ? L" - IAuthenticate" :
+ (guid_reason == IID_IHttpSecurity ? L"IHttpSecurity" :
+ (guid_reason == IID_IWindowForBindingUI ? L"IWindowForBindingUI" :
+ guid)));
+#endif
+
+ // TODO(iyengar): This hits when running the URL request tests.
+ DLOG_IF(ERROR, !::IsWindow(parent_window_))
+ << "UrlmonUrlRequest::GetWindow - no window!";
+ *parent_window = parent_window_;
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::Authenticate(HWND* parent_window,
+ LPWSTR* user_name,
+ LPWSTR* password) {
+ if (!parent_window) {
+ return E_INVALIDARG;
+ }
+
+ DCHECK(::IsWindow(parent_window_));
+ *parent_window = parent_window_;
+ return S_OK;
+}
+
+STDMETHODIMP UrlmonUrlRequest::OnSecurityProblem(DWORD problem) {
+ // Urlmon notifies the client of authentication problems, certificate
+ // errors, etc by querying the object implementing the IBindStatusCallback
+ // interface for the IHttpSecurity interface. If this interface is not
+ // implemented then Urlmon checks for the problem codes defined below
+ // and performs actions as defined below:-
+ // It invokes the ReportProgress method of the protocol sink with
+ // these problem codes and eventually invokes the ReportResult method
+ // on the protocol sink which ends up in a call to the OnStopBinding
+ // method of the IBindStatusCallBack interface.
+
+ // MSHTML's implementation of the IBindStatusCallback interface does not
+ // implement the IHttpSecurity interface. However it handles the
+ // OnStopBinding call with a HRESULT of 0x800c0019 and navigates to
+ // an interstitial page which presents the user with a choice of whether
+ // to abort the navigation.
+
+ // In our OnStopBinding implementation we stop the navigation and inform
+ // Chrome about the result. Ideally Chrome should behave in a manner similar
+ // to IE, i.e. display the SSL error interstitial page and if the user
+ // decides to proceed anyway we would turn off SSL warnings for that
+ // particular navigation and allow IE to download the content.
+ // We would need to return the certificate information to Chrome for display
+ // purposes. Currently we only return a dummy certificate to Chrome.
+ // At this point we decided that it is a lot of work at this point and
+ // decided to go with the easier option of implementing the IHttpSecurity
+ // interface and replicating the checks performed by Urlmon. This
+ // causes Urlmon to display a dialog box on the same lines as IE6.
+ DLOG(INFO) << __FUNCTION__ << " Security problem : " << problem;
+
+ HRESULT hr = E_ABORT;
+
+ switch (problem) {
+ case ERROR_INTERNET_SEC_CERT_REV_FAILED: {
+ hr = RPC_E_RETRY;
+ break;
+ }
+
+ case ERROR_INTERNET_SEC_CERT_DATE_INVALID:
+ case ERROR_INTERNET_SEC_CERT_CN_INVALID:
+ case ERROR_INTERNET_INVALID_CA: {
+ hr = S_FALSE;
+ break;
+ }
+
+ default: {
+ NOTREACHED() << "Unhandled security problem : " << problem;
+ break;
+ }
+ }
+ return hr;
+}
+
+HRESULT UrlmonUrlRequest::ConnectToExistingMoniker(IMoniker* moniker,
+ IBindCtx* context,
+ const std::wstring& url) {
+ if (!moniker || url.empty()) {
+ NOTREACHED() << "Invalid arguments";
+ return E_INVALIDARG;
+ }
+
+ DCHECK(moniker_.get() == NULL);
+ DCHECK(bind_context_.get() == NULL);
+
+ moniker_ = moniker;
+ bind_context_ = context;
+ set_url(WideToUTF8(url));
+ return S_OK;
+}
+
+HRESULT UrlmonUrlRequest::StartAsyncDownload() {
+ HRESULT hr = E_FAIL;
+ if (moniker_.get() == NULL) {
+ std::wstring wide_url = UTF8ToWide(url());
+ hr = CreateURLMonikerEx(NULL, wide_url.c_str(), moniker_.Receive(),
+ URL_MK_UNIFORM);
+ if (FAILED(hr)) {
+ NOTREACHED() << "CreateURLMonikerEx failed. Error: " << hr;
+ } else {
+ hr = CreateAsyncBindCtx(0, this, NULL, bind_context_.Receive());
+ DCHECK(SUCCEEDED(hr)) << "CreateAsyncBindCtx failed. Error: " << hr;
+ }
+ } else {
+ DCHECK(bind_context_.get() != NULL);
+ hr = RegisterBindStatusCallback(bind_context_, this, NULL, 0);
+ }
+
+ if (SUCCEEDED(hr)) {
+ ScopedComPtr<IStream> stream;
+ hr = moniker_->BindToStorage(bind_context_, NULL, __uuidof(IStream),
+ reinterpret_cast<void**>(stream.Receive()));
+ if (FAILED(hr)) {
+ // TODO(joshia): Look into. This currently fails for:
+ // http://user2:secret@localhost:1337/auth-basic?set-cookie-if-challenged
+ // when running the UrlRequest unit tests.
+ DLOG(ERROR) <<
+ StringPrintf("IUrlMoniker::BindToStorage failed. Error: 0x%08X.", hr)
+ << std::endl << url();
+ DCHECK(hr == MK_E_SYNTAX);
+ }
+ }
+
+ DLOG_IF(ERROR, FAILED(hr))
+ << StringPrintf(L"StartAsyncDownload failed: 0x%08X", hr);
+
+ return hr;
+}
+
+void UrlmonUrlRequest::EndRequest() {
+ DLOG(INFO) << __FUNCTION__;
+ // Special case. If the last request was a redirect and the current OS
+ // error value is E_ACCESSDENIED, that means an unsafe redirect was attempted.
+ // In that case, correct the OS error value to be the more specific
+ // ERR_UNSAFE_REDIRECT error value.
+ if (!status_.is_success() && status_.os_error() == net::ERR_ACCESS_DENIED) {
+ int status = GetHttpResponseStatus();
+ if (status >= 300 && status < 400) {
+ redirect_status_ = status; // store the latest redirect status value.
+ status_.set_os_error(net::ERR_UNSAFE_REDIRECT);
+ }
+ }
+ request_handler()->RemoveRequest(this);
+ OnResponseEnd(status_);
+
+ // If the request was started then we must have an additional reference on the
+ // request.
+ if (is_request_started_) {
+ is_request_started_ = false;
+ Release();
+ }
+}
+
+int UrlmonUrlRequest::GetHttpResponseStatus() const {
+ if (binding_ == NULL) {
+ DLOG(WARNING) << "GetHttpResponseStatus - no binding_";
+ return 0;
+ }
+
+ int http_status = 0;
+
+ ScopedComPtr<IWinInetHttpInfo> info;
+ if (SUCCEEDED(info.QueryFrom(binding_))) {
+ char status[10] = {0};
+ DWORD buf_size = sizeof(status);
+ if (SUCCEEDED(info->QueryInfo(HTTP_QUERY_STATUS_CODE, status, &buf_size,
+ 0, NULL))) {
+ http_status = StringToInt(status);
+ } else {
+ NOTREACHED() << "Failed to get HTTP status";
+ }
+ } else {
+ NOTREACHED() << "failed to get IWinInetHttpInfo from binding_";
+ }
+
+ return http_status;
+}
+
+//
+// UrlmonUrlRequest::Cache implementation.
+//
+
+size_t UrlmonUrlRequest::Cache::Size() {
+ size_t size = 0;
+ if (stream_) {
+ STATSTG cache_stat = {0};
+ stream_->Stat(&cache_stat, STATFLAG_NONAME);
+
+ DCHECK_EQ(0, cache_stat.cbSize.HighPart);
+ size = cache_stat.cbSize.LowPart;
+ }
+
+ return size;
+}
+
+size_t UrlmonUrlRequest::Cache::CurrentPos() {
+ size_t pos = 0;
+ if (stream_) {
+ ULARGE_INTEGER current_index = {0};
+ stream_->Seek(kZero, STREAM_SEEK_CUR, &current_index);
+
+ DCHECK_EQ(0, current_index.HighPart);
+ pos = current_index.LowPart;
+ }
+
+ return pos;
+}
+
+size_t UrlmonUrlRequest::Cache::SizeRemaining() {
+ size_t size = Size();
+ size_t pos = CurrentPos();
+ size_t size_remaining = 0;
+
+ if (size) {
+ DCHECK(pos <= size);
+ size_remaining = size - pos;
+ }
+ return size_remaining;
+}
+
+void UrlmonUrlRequest::Cache::Clear() {
+ if (!stream_) {
+ NOTREACHED();
+ return;
+ }
+
+ HRESULT hr = stream_->SetSize(kUnsignedZero);
+ DCHECK(SUCCEEDED(hr));
+}
+
+bool UrlmonUrlRequest::Cache::Read(IStream* dest, size_t size,
+ size_t* bytes_copied) {
+ if (!dest || !size) {
+ NOTREACHED();
+ return false;
+ }
+
+ // Copy the data and clear cache if there is no more data to copy.
+ ULARGE_INTEGER size_to_copy = {size, 0};
+ ULARGE_INTEGER size_written = {0};
+ stream_->CopyTo(dest, size_to_copy, NULL, &size_written);
+
+ if (size_written.LowPart && bytes_copied)
+ *bytes_copied = size_written.LowPart;
+
+ if (!SizeRemaining()) {
+ Clear();
+ stream_->Seek(kZero, STREAM_SEEK_SET, NULL);
+ }
+
+ return (size_written.LowPart != 0);
+}
+
+bool UrlmonUrlRequest::Cache::Append(IStream* source,
+ size_t* bytes_copied) {
+ if (!source) {
+ NOTREACHED();
+ return false;
+ }
+
+ size_t current_pos = CurrentPos();
+ stream_->Seek(kZero, STREAM_SEEK_END, NULL);
+
+ HRESULT hr = S_OK;
+ while (SUCCEEDED(hr)) {
+ DWORD chunk_read = 0; // NOLINT
+ hr = source->Read(read_buffer_, sizeof(read_buffer_), &chunk_read);
+ if (!chunk_read)
+ break;
+
+ DWORD chunk_written = 0; // NOLINT
+ stream_->Write(read_buffer_, chunk_read, &chunk_written);
+ DCHECK_EQ(chunk_read, chunk_written);
+
+ if (bytes_copied)
+ *bytes_copied += chunk_written;
+ }
+
+ LARGE_INTEGER last_read_position = {current_pos, 0};
+ stream_->Seek(last_read_position, STREAM_SEEK_SET, NULL);
+ return SUCCEEDED(hr);
+}
+
+bool UrlmonUrlRequest::Cache::Create() {
+ DCHECK(stream_ == NULL);
+ bool ret = SUCCEEDED(CreateStreamOnHGlobal(NULL, TRUE, stream_.Receive()));
+ DCHECK(ret && stream_);
+ return ret;
+}
+
+net::Error UrlmonUrlRequest::HresultToNetError(HRESULT hr) {
+ // Useful reference:
+ // http://msdn.microsoft.com/en-us/library/ms775145(VS.85).aspx
+
+ net::Error ret = net::ERR_UNEXPECTED;
+
+ switch (hr) {
+ case S_OK:
+ ret = net::OK;
+ break;
+
+ case MK_E_SYNTAX:
+ ret = net::ERR_INVALID_URL;
+ break;
+
+ case INET_E_CANNOT_CONNECT:
+ ret = net::ERR_CONNECTION_FAILED;
+ break;
+
+ case INET_E_DOWNLOAD_FAILURE:
+ case INET_E_CONNECTION_TIMEOUT:
+ case E_ABORT:
+ ret = net::ERR_CONNECTION_ABORTED;
+ break;
+
+ case INET_E_DATA_NOT_AVAILABLE:
+ ret = net::ERR_EMPTY_RESPONSE;
+ break;
+
+ case INET_E_RESOURCE_NOT_FOUND:
+ // To behave more closely to the chrome network stack, we translate this
+ // error value as tunnel connection failed. This error value is tested
+ // in the ProxyTunnelRedirectTest and UnexpectedServerAuthTest tests.
+ ret = net::ERR_TUNNEL_CONNECTION_FAILED;
+ break;
+
+ case INET_E_INVALID_URL:
+ case INET_E_UNKNOWN_PROTOCOL:
+ case INET_E_REDIRECT_FAILED:
+ ret = net::ERR_INVALID_URL;
+ break;
+
+ case INET_E_INVALID_CERTIFICATE:
+ ret = net::ERR_CERT_INVALID;
+ break;
+
+ case E_ACCESSDENIED:
+ ret = net::ERR_ACCESS_DENIED;
+ break;
+
+ default:
+ DLOG(WARNING)
+ << StringPrintf("TODO: translate HRESULT 0x%08X to net::Error", hr);
+ break;
+ }
+ return ret;
+}
diff --git a/chrome_frame/urlmon_url_request.h b/chrome_frame/urlmon_url_request.h
new file mode 100644
index 0000000..114ee6b
--- /dev/null
+++ b/chrome_frame/urlmon_url_request.h
@@ -0,0 +1,231 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_URLMON_URL_REQUEST_H_
+#define CHROME_FRAME_URLMON_URL_REQUEST_H_
+
+#include <urlmon.h>
+#include <atlbase.h>
+#include <atlcom.h>
+
+#include <algorithm>
+
+#include "base/lock.h"
+#include "base/platform_thread.h"
+#include "base/scoped_comptr_win.h"
+#include "chrome_frame/plugin_url_request.h"
+
+#include "net/base/net_errors.h"
+#include "net/base/upload_data.h"
+
+class UrlmonUrlRequest
+ : public CComObjectRootEx<CComSingleThreadModel>,
+ public PluginUrlRequest,
+ public IServiceProviderImpl<UrlmonUrlRequest>,
+ public IBindStatusCallback,
+ public IHttpNegotiate,
+ public IAuthenticate,
+ public IHttpSecurity {
+ public:
+ UrlmonUrlRequest();
+ ~UrlmonUrlRequest();
+
+BEGIN_COM_MAP(UrlmonUrlRequest)
+ COM_INTERFACE_ENTRY(IHttpNegotiate)
+ COM_INTERFACE_ENTRY(IServiceProvider)
+ COM_INTERFACE_ENTRY(IBindStatusCallback)
+ COM_INTERFACE_ENTRY(IWindowForBindingUI)
+ COM_INTERFACE_ENTRY(IAuthenticate)
+ COM_INTERFACE_ENTRY(IHttpSecurity)
+END_COM_MAP()
+
+BEGIN_SERVICE_MAP(UrlmonUrlRequest)
+ SERVICE_ENTRY(IID_IHttpNegotiate);
+END_SERVICE_MAP()
+
+ // PluginUrlRequest implementation
+ virtual bool Start();
+ virtual void Stop();
+ virtual bool Read(int bytes_to_read);
+
+ // IBindStatusCallback implementation
+ STDMETHOD(OnStartBinding)(DWORD reserved, IBinding* binding);
+ STDMETHOD(GetPriority)(LONG* priority);
+ STDMETHOD(OnLowResource)(DWORD reserved);
+ STDMETHOD(OnProgress)(ULONG progress, ULONG max_progress,
+ ULONG status_code, LPCWSTR status_text);
+ STDMETHOD(OnStopBinding)(HRESULT result, LPCWSTR error);
+ STDMETHOD(GetBindInfo)(DWORD* bind_flags, BINDINFO* bind_info);
+ STDMETHOD(OnDataAvailable)(DWORD flags, DWORD size, FORMATETC* formatetc,
+ STGMEDIUM* storage);
+ STDMETHOD(OnObjectAvailable)(REFIID iid, IUnknown* object);
+
+ // IHttpNegotiate implementation
+ STDMETHOD(BeginningTransaction)(const wchar_t* url,
+ const wchar_t* current_headers, DWORD reserved,
+ wchar_t** additional_headers);
+ STDMETHOD(OnResponse)(DWORD dwResponseCode, const wchar_t* response_headers,
+ const wchar_t* request_headers, wchar_t** additional_headers);
+
+ // IWindowForBindingUI implementation. This interface is used typically to
+ // query the window handle which URLMON uses as the parent of error dialogs.
+ STDMETHOD(GetWindow)(REFGUID guid_reason, HWND* parent_window);
+
+ // IAuthenticate implementation. Used to return the parent window for the
+ // dialog displayed by IE for authenticating with a proxy.
+ STDMETHOD(Authenticate)(HWND* parent_window, LPWSTR* user_name,
+ LPWSTR* password);
+
+ // IHttpSecurity implementation.
+ STDMETHOD(OnSecurityProblem)(DWORD problem);
+
+ HRESULT ConnectToExistingMoniker(IMoniker* moniker, IBindCtx* context,
+ const std::wstring& url);
+
+ void set_parent_window(HWND parent_window) {
+ parent_window_ = parent_window;
+ }
+
+ protected:
+ static const size_t kCopyChunkSize = 32 * 1024;
+
+ // A fake stream class to make it easier to copy received data using
+ // IStream::CopyTo instead of allocating temporary buffers and keeping
+ // track of data copied so far.
+ class SendStream
+ : public CComObjectRoot,
+ public IStream {
+ public:
+ SendStream() {
+ }
+
+ BEGIN_COM_MAP(SendStream)
+ COM_INTERFACE_ENTRY(IStream)
+ COM_INTERFACE_ENTRY(ISequentialStream)
+ END_COM_MAP()
+
+ void Initialize(UrlmonUrlRequest* request) {
+ request_ = request;
+ }
+
+ STDMETHOD(Read)(void* pv, ULONG cb, ULONG* read) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Write)(const void * buffer, ULONG size, ULONG* size_written) {
+ DCHECK(request_);
+ int size_to_write = static_cast<int>(
+ std::min(static_cast<ULONG>(MAXINT), size));
+ request_->OnReadComplete(buffer, size_to_write);
+ if (size_written)
+ *size_written = size_to_write;
+ return S_OK;
+ }
+
+ STDMETHOD(Seek)(LARGE_INTEGER move, DWORD origin, ULARGE_INTEGER* new_pos) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(SetSize)(ULARGE_INTEGER new_size) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(CopyTo)(IStream* stream, ULARGE_INTEGER cb, ULARGE_INTEGER* read,
+ ULARGE_INTEGER* written) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Commit)(DWORD flags) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Revert)() {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(LockRegion)(ULARGE_INTEGER offset, ULARGE_INTEGER cb,
+ DWORD type) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(UnlockRegion)(ULARGE_INTEGER offset, ULARGE_INTEGER cb,
+ DWORD type) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Stat)(STATSTG *pstatstg, DWORD grfStatFlag) {
+ return E_NOTIMPL;
+ }
+
+ STDMETHOD(Clone)(IStream** stream) {
+ DCHECK(false) << __FUNCTION__;
+ return E_NOTIMPL;
+ }
+
+ protected:
+ scoped_refptr<UrlmonUrlRequest> request_;
+ DISALLOW_COPY_AND_ASSIGN(SendStream);
+ };
+
+ // Manage data caching. Note: this class supports cache
+ // size less than 2GB
+ class Cache {
+ public:
+ bool Create();
+
+ // Adds data to the end of the cache.
+ bool Append(IStream* source, size_t* bytes_copied);
+
+ // Reads from the cache.
+ bool Read(IStream* dest, size_t size, size_t* bytes_copied);
+
+ size_t Size();
+ size_t CurrentPos();
+ size_t SizeRemaining();
+ void Clear();
+ bool is_valid() const {
+ return (stream_ != NULL);
+ }
+
+ protected:
+ ScopedComPtr<IStream> stream_;
+ char read_buffer_[kCopyChunkSize];
+ };
+
+ HRESULT StartAsyncDownload();
+ void EndRequest();
+
+ int GetHttpResponseStatus() const;
+
+ static net::Error HresultToNetError(HRESULT hr);
+
+ private:
+ std::wstring redirect_url_;
+ int redirect_status_;
+ ScopedComPtr<IBinding> binding_;
+ ScopedComPtr<IMoniker> moniker_;
+ ScopedComPtr<IBindCtx> bind_context_;
+ Cache cached_data_;
+ size_t pending_read_size_;
+ URLRequestStatus status_;
+
+ uint64 post_data_len_;
+
+ PlatformThreadId thread_;
+ bool is_request_started_;
+ static int instance_count_;
+ HWND parent_window_;
+ DISALLOW_COPY_AND_ASSIGN(UrlmonUrlRequest);
+};
+
+#endif // CHROME_FRAME_URLMON_URL_REQUEST_H_
+
diff --git a/chrome_frame/utils.cc b/chrome_frame/utils.cc
new file mode 100644
index 0000000..b445c26
--- /dev/null
+++ b/chrome_frame/utils.cc
@@ -0,0 +1,643 @@
+// Copyright (c) 2009 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 <shlobj.h>
+
+#include "chrome_frame/html_utils.h"
+#include "chrome_frame/utils.h"
+
+#include "base/file_util.h"
+#include "base/file_version_info.h"
+#include "base/logging.h"
+#include "base/path_service.h"
+#include "base/registry.h"
+#include "base/scoped_comptr_win.h"
+#include "base/string_util.h"
+#include "chrome/common/chrome_constants.h"
+#include "chrome/installer/util/google_update_constants.h"
+#include "googleurl/src/gurl.h"
+#include "grit/chrome_frame_resources.h"
+#include "chrome_frame/resource.h"
+
+// Note that these values are all lower case and are compared to
+// lower-case-transformed values.
+const wchar_t kMetaTag[] = L"meta";
+const wchar_t kHttpEquivAttribName[] = L"http-equiv";
+const wchar_t kContentAttribName[] = L"content";
+const wchar_t kXUACompatValue[] = L"x-ua-compatible";
+const wchar_t kBodyTag[] = L"body";
+const wchar_t kChromeContentPrefix[] = L"chrome=";
+const wchar_t kChromeProtocolPrefix[] = L"cf:";
+
+static const wchar_t kChromeFrameConfigKey[] =
+ L"Software\\Google\\ChromeFrame";
+static const wchar_t kChromeFrameOptinUrlsKey[] = L"OptinUrls";
+
+// Used to isolate chrome frame builds from google chrome release channels.
+const wchar_t kChromeFrameOmahaSuffix[] = L"-cf";
+const wchar_t kDevChannelName[] = L"-dev";
+
+const wchar_t kChromeAttachExternalTabPrefix[] = L"attach_external_tab";
+
+HRESULT UtilRegisterTypeLib(HINSTANCE tlb_instance,
+ LPCOLESTR index,
+ bool for_current_user_only) {
+ CComBSTR path;
+ CComPtr<ITypeLib> type_lib;
+ HRESULT hr = AtlLoadTypeLib(tlb_instance, index, &path, &type_lib);
+ if (SUCCEEDED(hr)) {
+ hr = UtilRegisterTypeLib(type_lib, path, NULL, for_current_user_only);
+ }
+ return hr;
+}
+
+HRESULT UtilUnRegisterTypeLib(HINSTANCE tlb_instance,
+ LPCOLESTR index,
+ bool for_current_user_only) {
+ CComBSTR path;
+ CComPtr<ITypeLib> type_lib;
+ HRESULT hr = AtlLoadTypeLib(tlb_instance, index, &path, &type_lib);
+ if (SUCCEEDED(hr)) {
+ hr = UtilUnRegisterTypeLib(type_lib, for_current_user_only);
+ }
+ return hr;
+}
+
+HRESULT UtilRegisterTypeLib(LPCWSTR typelib_path,
+ bool for_current_user_only) {
+ if (NULL == typelib_path) {
+ return E_INVALIDARG;
+ }
+ CComBSTR path;
+ CComPtr<ITypeLib> type_lib;
+ HRESULT hr = ::LoadTypeLib(typelib_path, &type_lib);
+ if (SUCCEEDED(hr)) {
+ hr = UtilRegisterTypeLib(type_lib,
+ typelib_path,
+ NULL,
+ for_current_user_only);
+ }
+ return hr;
+}
+
+HRESULT UtilUnRegisterTypeLib(LPCWSTR typelib_path,
+ bool for_current_user_only) {
+ CComPtr<ITypeLib> type_lib;
+ HRESULT hr = ::LoadTypeLib(typelib_path, &type_lib);
+ if (SUCCEEDED(hr)) {
+ hr = UtilUnRegisterTypeLib(type_lib, for_current_user_only);
+ }
+ return hr;
+}
+
+HRESULT UtilRegisterTypeLib(ITypeLib* typelib,
+ LPCWSTR typelib_path,
+ LPCWSTR help_dir,
+ bool for_current_user_only) {
+ typedef HRESULT(WINAPI *RegisterTypeLibPrototype)(ITypeLib FAR* type_lib,
+ OLECHAR FAR* full_path,
+ OLECHAR FAR* help_dir);
+ LPCSTR function_name =
+ for_current_user_only ? "RegisterTypeLibForUser" : "RegisterTypeLib";
+ RegisterTypeLibPrototype reg_tlb =
+ reinterpret_cast<RegisterTypeLibPrototype>(
+ GetProcAddress(GetModuleHandle(_T("oleaut32.dll")),
+ function_name));
+ if (NULL == reg_tlb) {
+ return E_FAIL;
+ }
+ return reg_tlb(typelib,
+ const_cast<OLECHAR*>(typelib_path),
+ const_cast<OLECHAR*>(help_dir));
+}
+
+HRESULT UtilUnRegisterTypeLib(ITypeLib* typelib,
+ bool for_current_user_only) {
+ if (NULL == typelib) {
+ return E_INVALIDARG;
+ }
+ typedef HRESULT(WINAPI *UnRegisterTypeLibPrototype)(
+ REFGUID libID,
+ unsigned short wVerMajor, // NOLINT
+ unsigned short wVerMinor, // NOLINT
+ LCID lcid,
+ SYSKIND syskind);
+ LPCSTR function_name =
+ for_current_user_only ? "UnRegisterTypeLibForUser" : "UnRegisterTypeLib";
+
+ UnRegisterTypeLibPrototype unreg_tlb =
+ reinterpret_cast<UnRegisterTypeLibPrototype>(
+ GetProcAddress(GetModuleHandle(_T("oleaut32.dll")),
+ function_name));
+ if (NULL == unreg_tlb) {
+ return E_FAIL;
+ }
+ TLIBATTR* tla = NULL;
+ HRESULT hr = typelib->GetLibAttr(&tla);
+ if (SUCCEEDED(hr)) {
+ hr = unreg_tlb(tla->guid,
+ tla->wMajorVerNum,
+ tla->wMinorVerNum,
+ tla->lcid,
+ tla->syskind);
+ typelib->ReleaseTLibAttr(tla);
+ }
+ return hr;
+}
+
+HRESULT UtilGetXUACompatContentValue(const std::wstring& html_string,
+ std::wstring* content_value) {
+ if (!content_value) {
+ return E_POINTER;
+ }
+
+ // Fail fast if the string X-UA-Compatible isn't in html_string
+ if (StringToLowerASCII(html_string).find(kXUACompatValue) ==
+ std::wstring::npos) {
+ return E_FAIL;
+ }
+
+ HTMLScanner scanner(html_string.c_str());
+
+ // Build the list of meta tags that occur before the body tag is hit.
+ HTMLScanner::StringRangeList tag_list;
+ scanner.GetTagsByName(kMetaTag, &tag_list, kBodyTag);
+
+ // Search the list of meta tags for one with an http-equiv="X-UA-Compatible"
+ // attribute.
+ HTMLScanner::StringRange attribute;
+ std::string search_attribute_ascii(WideToASCII(kXUACompatValue));
+ HTMLScanner::StringRangeList::const_iterator tag_list_iter(tag_list.begin());
+ for (; tag_list_iter != tag_list.end(); tag_list_iter++) {
+ if (!tag_list_iter->GetTagAttribute(kHttpEquivAttribName, &attribute)) {
+ continue;
+ }
+
+ // We found an http-equiv meta tag, check its value using the ascii
+ // case-insensitive comparison method.
+ if (!attribute.LowerCaseEqualsASCII(search_attribute_ascii.c_str())) {
+ continue;
+ }
+
+ // We found our X-UA-Compatible meta tag so look for and extract
+ // the value of the content attribute.
+ if (!tag_list_iter->GetTagAttribute(kContentAttribName, &attribute)) {
+ continue;
+ }
+
+ // Found the content string, copy and return.
+ content_value->assign(attribute.Copy());
+ return S_OK;
+ }
+
+ return E_FAIL;
+}
+
+bool AppendSuffixToChannelName(std::wstring* string,
+ const std::wstring& channel_name,
+ const std::wstring& suffix) {
+ size_t pos = string->find(channel_name);
+ // Append the suffix only if we find the channel name.
+ if (pos != std::wstring::npos) {
+ pos += channel_name.size();
+ // Append the suffix only to the channel name only if the name is not
+ // already followed by suffix.
+ if (string->find(suffix, pos) != pos) {
+ string->insert(pos, suffix);
+ return true;
+ }
+ }
+ return false;
+}
+
+bool RemoveSuffixFromChannelName(std::wstring* string,
+ const std::wstring& channel_name,
+ const std::wstring& suffix) {
+ std::wstring decorated_channel(channel_name + suffix);
+ size_t pos = string->find(decorated_channel);
+ // TODO(robertshield): Remove the suffix iff the suffix is the last thing in
+ // the string or is followed by another suffix that starts with '-'.
+ if (pos != std::wstring::npos) {
+ pos += channel_name.size();
+ string->erase(pos, suffix.size());
+ return true;
+ }
+ return false;
+}
+
+HRESULT UtilUpdateOmahaConfig(bool add_cf_suffix) {
+ HKEY reg_root = HKEY_LOCAL_MACHINE;
+
+ RegKey key;
+ std::wstring ap_key_value;
+ std::wstring reg_key(google_update::kRegPathClientState);
+ reg_key.append(L"\\");
+ reg_key.append(google_update::kChromeGuid);
+ if (!key.Open(reg_root, reg_key.c_str(), KEY_READ | KEY_WRITE) ||
+ !key.ReadValue(google_update::kRegApField, &ap_key_value)) {
+ // Can't read the Omaha config.
+ return REGDB_E_READREGDB;
+ }
+
+ HRESULT result = S_OK;
+ // We've read the key in, try and modify it then write it back.
+ if (add_cf_suffix && AppendSuffixToChannelName(&ap_key_value,
+ kDevChannelName,
+ kChromeFrameOmahaSuffix)) {
+ if (!key.WriteValue(google_update::kRegApField, ap_key_value.c_str())) {
+ DLOG(ERROR) << "Failed to add suffix to omaha ap key value.";
+ result = REGDB_E_WRITEREGDB;
+ }
+ } else if (!add_cf_suffix &&
+ RemoveSuffixFromChannelName(&ap_key_value,
+ kDevChannelName,
+ kChromeFrameOmahaSuffix)) {
+ if (!key.WriteValue(google_update::kRegApField, ap_key_value.c_str())) {
+ DLOG(ERROR) << "Failed to remove suffix from omaha ap key value.";
+ result = REGDB_E_WRITEREGDB;
+ }
+ } else {
+ // Getting here means that no modifications needed to be made.
+ result = S_FALSE;
+ }
+
+ return result;
+}
+
+std::wstring GetResourceString(int resource_id) {
+ std::wstring resource_string;
+ HMODULE this_module = reinterpret_cast<HMODULE>(&__ImageBase);
+ const ATLSTRINGRESOURCEIMAGE* image = AtlGetStringResourceImage(
+ this_module, resource_id);
+ if (image) {
+ resource_string.assign(image->achString, image->nLength);
+ } else {
+ NOTREACHED() << "Unable to find resource id " << resource_id;
+ }
+ return resource_string;
+}
+
+void DisplayVersionMismatchWarning(HWND parent,
+ const std::string& server_version) {
+ // Obtain the current module version.
+ FileVersionInfo* file_version_info =
+ FileVersionInfo::CreateFileVersionInfoForCurrentModule();
+ DCHECK(file_version_info);
+ std::wstring version_string(file_version_info->file_version());
+ std::wstring wide_server_version;
+ if (server_version.empty()) {
+ wide_server_version = GetResourceString(IDS_VERSIONUNKNOWN);
+ } else {
+ wide_server_version = ASCIIToWide(server_version);
+ }
+ std::wstring title = GetResourceString(IDS_VERSIONMISMATCH_HEADER);
+ std::wstring message;
+ SStringPrintf(&message, GetResourceString(IDS_VERSIONMISMATCH).c_str(),
+ wide_server_version.c_str(), version_string.c_str());
+
+ ::MessageBox(parent, message.c_str(), title.c_str(), MB_OK);
+}
+
+std::string CreateJavascript(const std::string& function_name,
+ const std::string args) {
+ std::string script_string = "javascript:";
+ script_string += function_name + "(";
+ if (!args.empty()) {
+ script_string += "'";
+ script_string += args;
+ script_string += "'";
+ }
+ script_string += ")";
+ return script_string;
+}
+
+AddRefModule::AddRefModule() {
+ // TODO(tommi): Override the module's Lock/Unlock methods to call
+ // npapi::SetValue(NPPVpluginKeepLibraryInMemory) and keep the dll loaded
+ // while the module's refcount is > 0. Only do this when we're being
+ // used as an NPAPI module.
+ _pAtlModule->Lock();
+}
+
+
+AddRefModule::~AddRefModule() {
+ _pAtlModule->Unlock();
+}
+
+namespace {
+const char kIEImageName[] = "iexplore.exe";
+const char kFirefoxImageName[] = "firefox.exe";
+const char kOperaImageName[] = "opera.exe";
+} // namespace
+
+std::wstring GetHostProcessName(bool include_extension) {
+ FilePath exe;
+ if (PathService::Get(base::FILE_EXE, &exe))
+ exe = exe.BaseName();
+ if (!include_extension) {
+ exe = exe.RemoveExtension();
+ }
+ return exe.ToWStringHack();
+}
+
+BrowserType GetBrowserType() {
+ static BrowserType browser_type = BROWSER_INVALID;
+
+ if (browser_type == BROWSER_INVALID) {
+ std::wstring exe(GetHostProcessName(true));
+ if (!exe.empty()) {
+ std::wstring::const_iterator begin = exe.begin();
+ std::wstring::const_iterator end = exe.end();
+ if (LowerCaseEqualsASCII(begin, end, kIEImageName)) {
+ browser_type = BROWSER_IE;
+ } else if (LowerCaseEqualsASCII(begin, end, kFirefoxImageName)) {
+ browser_type = BROWSER_FIREFOX;
+ } else if (LowerCaseEqualsASCII(begin, end, kOperaImageName)) {
+ browser_type = BROWSER_OPERA;
+ } else {
+ browser_type = BROWSER_UNKNOWN;
+ }
+ } else {
+ NOTREACHED();
+ }
+ }
+
+ return browser_type;
+}
+
+IEVersion GetIEVersion() {
+ static IEVersion ie_version = IE_INVALID;
+
+ if (ie_version == IE_INVALID) {
+ wchar_t exe_path[MAX_PATH];
+ HMODULE mod = GetModuleHandle(NULL);
+ GetModuleFileName(mod, exe_path, arraysize(exe_path) - 1);
+ std::wstring exe_name(file_util::GetFilenameFromPath(exe_path));
+ if (!LowerCaseEqualsASCII(exe_name, kIEImageName)) {
+ ie_version = NON_IE;
+ } else {
+ uint32 high = 0;
+ uint32 low = 0;
+ if (GetModuleVersion(mod, &high, &low)) {
+ switch (HIWORD(high)) {
+ case 6:
+ ie_version = IE_6;
+ break;
+ case 7:
+ ie_version = IE_7;
+ break;
+ default:
+ ie_version = HIWORD(high) >= 8 ? IE_8 : IE_UNSUPPORTED;
+ break;
+ }
+ } else {
+ NOTREACHED() << "Can't get IE version";
+ }
+ }
+ }
+
+ return ie_version;
+}
+
+bool IsIEInPrivate() {
+ typedef BOOL (WINAPI* IEIsInPrivateBrowsingPtr)();
+ bool incognito_mode = false;
+ HMODULE h = GetModuleHandle(L"ieframe.dll");
+ if (h) {
+ IEIsInPrivateBrowsingPtr IsInPrivate =
+ reinterpret_cast<IEIsInPrivateBrowsingPtr>(GetProcAddress(h,
+ "IEIsInPrivateBrowsing"));
+ if (IsInPrivate) {
+ incognito_mode = !!IsInPrivate();
+ }
+ }
+
+ return incognito_mode;
+}
+
+bool GetModuleVersion(HMODULE module, uint32* high, uint32* low) {
+ DCHECK(module != NULL)
+ << "Please use GetModuleHandle(NULL) to get the process name";
+ DCHECK(high);
+
+ bool ok = false;
+
+ HRSRC res = FindResource(module,
+ reinterpret_cast<const wchar_t*>(VS_VERSION_INFO), RT_VERSION);
+ if (res) {
+ HGLOBAL res_data = LoadResource(module, res);
+ DWORD version_resource_size = SizeofResource(module, res);
+ const void* readonly_resource_data = LockResource(res_data);
+ if (readonly_resource_data && version_resource_size) {
+ // Copy data as VerQueryValue tries to modify the data. This causes
+ // exceptions and heap corruption errors if debugger is attached.
+ scoped_ptr<char> data(new char[version_resource_size]);
+ memcpy(data.get(), readonly_resource_data, version_resource_size);
+ if (data.get()) {
+ VS_FIXEDFILEINFO* ver_info = NULL;
+ UINT info_size = 0;
+ if (VerQueryValue(data.get(), L"\\",
+ reinterpret_cast<void**>(&ver_info), &info_size)) {
+ *high = ver_info->dwFileVersionMS;
+ if (low != NULL)
+ *low = ver_info->dwFileVersionLS;
+ ok = true;
+ }
+
+ UnlockResource(res_data);
+ }
+ FreeResource(res_data);
+ }
+ }
+
+ return ok;
+}
+
+namespace {
+
+const int kMaxSubmenuDepth = 10;
+
+// Copies original_menu and returns the copy. The caller is responsible for
+// closing the returned HMENU. This does not currently copy over bitmaps
+// (e.g. hbmpChecked, hbmpUnchecked or hbmpItem), so checkmarks, radio buttons,
+// and custom icons won't work.
+// It also copies over submenus up to a maximum depth of kMaxSubMenuDepth.
+//
+// TODO(robertshield): Add support for the bitmap fields if need be.
+HMENU UtilCloneContextMenuImpl(HMENU original_menu, int depth) {
+ DCHECK(IsMenu(original_menu));
+
+ if (depth >= kMaxSubmenuDepth)
+ return NULL;
+
+ HMENU new_menu = CreatePopupMenu();
+ int item_count = GetMenuItemCount(original_menu);
+ if (item_count <= 0) {
+ NOTREACHED();
+ } else {
+ for (int i = 0; i < item_count; i++) {
+ MENUITEMINFO item_info = { 0 };
+ item_info.cbSize = sizeof(MENUITEMINFO);
+ item_info.fMask = MIIM_ID | MIIM_STRING | MIIM_FTYPE |
+ MIIM_STATE | MIIM_DATA | MIIM_SUBMENU |
+ MIIM_CHECKMARKS | MIIM_BITMAP;
+
+ // Call GetMenuItemInfo a first time to obtain the buffer size for
+ // the label.
+ if (GetMenuItemInfo(original_menu, i, TRUE, &item_info)) {
+ item_info.cch++; // Increment this as per MSDN
+ std::vector<wchar_t> buffer(item_info.cch, 0);
+ item_info.dwTypeData = &buffer[0];
+
+ // Call GetMenuItemInfo a second time with dwTypeData set to a buffer
+ // of a correct size to get the label.
+ GetMenuItemInfo(original_menu, i, TRUE, &item_info);
+
+ // Clone any submenus. Within reason.
+ if (item_info.hSubMenu) {
+ HMENU new_submenu = UtilCloneContextMenuImpl(item_info.hSubMenu,
+ depth + 1);
+ item_info.hSubMenu = new_submenu;
+ }
+
+ // Now insert the item into the new menu.
+ InsertMenuItem(new_menu, i, TRUE, &item_info);
+ }
+ }
+ }
+ return new_menu;
+}
+
+} // namespace
+
+HMENU UtilCloneContextMenu(HMENU original_menu) {
+ return UtilCloneContextMenuImpl(original_menu, 0);
+}
+
+std::string ResolveURL(const std::string& document,
+ const std::string& relative) {
+ if (document.empty()) {
+ return GURL(relative).spec();
+ } else {
+ return GURL(document).Resolve(relative).spec();
+ }
+}
+
+bool HaveSameOrigin(const std::string& url1, const std::string& url2) {
+ GURL a(url1), b(url2);
+ bool ret;
+ if (a.is_valid() != b.is_valid()) {
+ // Either (but not both) url is invalid, so they can't match.
+ ret = false;
+ } else if (!a.is_valid()) {
+ // Both URLs are invalid (see first check). Just check if the opaque
+ // strings match exactly.
+ ret = url1.compare(url2) == 0;
+ } else if (a.GetOrigin() != b.GetOrigin()) {
+ // The origins don't match.
+ ret = false;
+ } else {
+ // we have a match.
+ ret = true;
+ }
+
+ return ret;
+}
+
+int GetConfigInt(int default_value, const wchar_t* value_name) {
+ int ret = default_value;
+ RegKey config_key;
+ if (config_key.Open(HKEY_CURRENT_USER, kChromeFrameConfigKey,
+ KEY_QUERY_VALUE)) {
+ int value = FALSE;
+ if (config_key.ReadValueDW(value_name, reinterpret_cast<DWORD*>(&value))) {
+ ret = value;
+ }
+ }
+
+ return ret;
+}
+
+bool GetConfigBool(bool default_value, const wchar_t* value_name) {
+ DWORD value = GetConfigInt(default_value, value_name);
+ return (value != FALSE);
+}
+
+bool IsOptInUrl(const wchar_t* url) {
+ RegKey config_key;
+ if (!config_key.Open(HKEY_CURRENT_USER, kChromeFrameConfigKey, KEY_READ))
+ return false;
+
+ RegistryValueIterator optin_urls_list(config_key.Handle(),
+ kChromeFrameOptinUrlsKey);
+ while (optin_urls_list.Valid()) {
+ if (MatchPattern(url, optin_urls_list.Name()))
+ return true;
+ ++optin_urls_list;
+ }
+
+ return false;
+}
+
+HRESULT GetUrlFromMoniker(IMoniker* moniker, IBindCtx* bind_context,
+ std::wstring* url) {
+ if (!moniker || !url) {
+ NOTREACHED();
+ return E_INVALIDARG;
+ }
+
+ ScopedComPtr<IBindCtx> temp_bind_context;
+ if (!bind_context) {
+ CreateBindCtx(0, temp_bind_context.Receive());
+ bind_context = temp_bind_context;
+ }
+
+ CComHeapPtr<WCHAR> display_name;
+ HRESULT hr = moniker->GetDisplayName(bind_context, NULL, &display_name);
+ if (display_name)
+ *url = display_name;
+
+ return hr;
+}
+
+bool IsValidUrlScheme(const std::wstring& url) {
+ if (url.empty())
+ return false;
+
+ GURL crack_url(url);
+
+ if (crack_url.SchemeIs("http") || crack_url.SchemeIs("https") ||
+ crack_url.SchemeIs("about") || crack_url.SchemeIs("view-source"))
+ return true;
+
+ if (StartsWith(url, kChromeAttachExternalTabPrefix, false))
+ return true;
+
+ return false;
+}
+
+// TODO(robertshield): Register and use Chrome's PathProviders.
+// - Note that this function is used by unit tests as well to override
+// PathService paths, so please test when addressing todo.
+bool GetUserProfileBaseDirectory(std::wstring* path) {
+ DCHECK(path);
+ wchar_t path_buffer[MAX_PATH * 4];
+ path_buffer[0] = 0;
+ // TODO(robertshield): Ideally we should use SHGetFolderLocation and then
+ // get a path via PIDL.
+ HRESULT hr = SHGetFolderPath(NULL, CSIDL_LOCAL_APPDATA, NULL,
+ SHGFP_TYPE_CURRENT, path_buffer);
+
+ if (SUCCEEDED(hr)) {
+ *path = path_buffer;
+#if defined(GOOGLE_CHROME_BUILD)
+ file_util::AppendToPath(path, FILE_PATH_LITERAL("Google"));
+#endif
+ file_util::AppendToPath(path, chrome::kBrowserAppName);
+ file_util::AppendToPath(path, chrome::kUserDataDirname);
+ return true;
+ }
+
+ return false;
+}
diff --git a/chrome_frame/utils.h b/chrome_frame/utils.h
new file mode 100644
index 0000000..80b9a53
--- /dev/null
+++ b/chrome_frame/utils.h
@@ -0,0 +1,227 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_UTILS_H_
+#define CHROME_FRAME_UTILS_H_
+
+#include <atlbase.h>
+#include <string>
+
+#include "base/basictypes.h"
+
+// utils.h : Various utility functions and classes
+
+extern const wchar_t kChromeContentPrefix[];
+extern const wchar_t kChromeProtocolPrefix[];
+
+// This function is very similar to the AtlRegisterTypeLib function except
+// that it takes a parameter that specifies whether to register the typelib
+// for the current user only or on a machine-wide basis
+// Refer to the MSDN documentation for AtlRegisterTypeLib for a description of
+// the arguments
+HRESULT UtilRegisterTypeLib(HINSTANCE tlb_instance,
+ LPCOLESTR index,
+ bool for_current_user_only);
+
+// This function is very similar to the AtlUnRegisterTypeLib function except
+// that it takes a parameter that specifies whether to unregister the typelib
+// for the current user only or on a machine-wide basis
+// Refer to the MSDN documentation for AtlUnRegisterTypeLib for a description
+// of the arguments
+HRESULT UtilUnRegisterTypeLib(HINSTANCE tlb_instance,
+ LPCOLESTR index,
+ bool for_current_user_only);
+
+HRESULT UtilRegisterTypeLib(LPCWSTR typelib_path, bool for_current_user_only);
+
+HRESULT UtilUnRegisterTypeLib(LPCWSTR typelib_path, bool for_current_user_only);
+
+HRESULT UtilRegisterTypeLib(ITypeLib* typelib,
+ LPCWSTR typelib_path,
+ LPCWSTR help_dir,
+ bool for_current_user_only);
+
+HRESULT UtilUnRegisterTypeLib(ITypeLib* typelib,
+ bool for_current_user_only);
+
+// Given an HTML fragment, this function looks for the
+// <meta http-equiv="X-UA-Compatible"> tag and extracts the value of the
+// "content" attribute
+// This method will currently return a false positive if the tag appears
+// inside a string in a <SCRIPT> block.
+HRESULT UtilGetXUACompatContentValue(const std::wstring& html_string,
+ std::wstring* content_value);
+
+
+// Appends |suffix| to the substring |channel_name| of |string| iff
+// the first instance of |channel_name| in |string| is not already followed by
+// |suffix|.
+// Returns true if |string| was modified.
+bool AppendSuffixToChannelName(std::wstring* string,
+ const std::wstring& channel_name,
+ const std::wstring& suffix);
+
+// Removes |suffix| from |string| if |string| contains |channel_name| followed
+// by |suffix|.
+// Returns true if |string| was modified.
+bool RemoveSuffixFromChannelName(std::wstring* string,
+ const std::wstring& channel_name,
+ const std::wstring& suffix);
+
+// Looks for and alters if found the Omaha configuration for Chrome in the
+// registry. This changes the auto-update release channel to prevent installed
+// builds of Chrome that include Chrome Frame from getting replaced by
+// Chrome updates without it.
+// Adds the Chrome Frame suffix if add_cf_suffix is true, removes it
+// otherwise.
+// Returns S_OK if the Chrome Omaha configuration was found and updated.
+// Returns S_FALSE if the configuration was found but didn't need updating.
+// Returns REGDB_E_READREGDB if the Chrome Omaha key could not be read.
+// Returns REGDB_E_WRITEREGDB if the Chrome Omaha key could not be written.
+HRESULT UtilUpdateOmahaConfig(bool add_cf_suffix);
+
+// Returns a string from ChromeFrame's string table by resource. Must be
+// provided with a valid resource id.
+std::wstring GetResourceString(int resource_id);
+
+// Displays a message box indicating that there was a version mismatch between
+// ChromeFrame and the running instance of Chrome.
+// server_version is the version of the running instance of Chrome.
+void DisplayVersionMismatchWarning(HWND parent,
+ const std::string& server_version);
+
+// This class provides a base implementation for ATL modules which want to
+// perform all their registration under HKCU. This class overrides the
+// RegisterServer and UnregisterServer methods and registers the type libraries
+// under HKCU (the rest of the registation is made under HKCU by changing the
+// appropriate .RGS files)
+template < class BaseAtlModule >
+class AtlPerUserModule : public BaseAtlModule {
+ public:
+ HRESULT RegisterServer(BOOL reg_typelib = FALSE,
+ const CLSID* clsid = NULL) throw() {
+ HRESULT hr = BaseAtlModule::RegisterServer(FALSE, clsid);
+ if (FAILED(hr)) {
+ return hr;
+ }
+ if (reg_typelib) {
+ hr = UtilRegisterTypeLib(_AtlComModule.m_hInstTypeLib, NULL, false);
+ }
+ return hr;
+ }
+
+ HRESULT UnregisterServer(BOOL unreg_typelib,
+ const CLSID* clsid = NULL) throw() {
+ HRESULT hr = BaseAtlModule::UnregisterServer(FALSE, clsid);
+ if (FAILED(hr)) {
+ return hr;
+ }
+ if (unreg_typelib) {
+ hr = UtilUnRegisterTypeLib(_AtlComModule.m_hInstTypeLib, NULL, false);
+ }
+ return hr;
+ }
+};
+
+// Creates a javascript statement for execution from the function name and
+// arguments passed in.
+std::string CreateJavascript(const std::string& function_name,
+ const std::string args);
+
+// Use to prevent the DLL from being unloaded while there are still living
+// objects with outstanding references.
+class AddRefModule {
+ public:
+ AddRefModule();
+ ~AddRefModule();
+};
+
+// Retrieves the executable name of the process hosting us. If
+// |include_extension| is false, then we strip the extension from the name.
+std::wstring GetHostProcessName(bool include_extension);
+
+typedef enum BrowserType {
+ BROWSER_INVALID = -1,
+ BROWSER_UNKNOWN,
+ BROWSER_IE,
+ BROWSER_FIREFOX,
+ BROWSER_OPERA,
+};
+
+BrowserType GetBrowserType();
+
+typedef enum IEVersion {
+ IE_INVALID,
+ NON_IE,
+ IE_UNSUPPORTED,
+ IE_6,
+ IE_7,
+ IE_8,
+};
+
+// To get the IE version when Chrome Frame is hosted in IE. Make sure that
+// the hosting browser is IE before calling this function, otherwise NON_IE
+// will be returned.
+IEVersion GetIEVersion();
+
+// Retrieves the file version from a module handle without extra round trips
+// to the disk (as happens with the regular GetFileVersionInfo API).
+//
+// @param module A handle to the module for which to retrieve the version info.
+// @param high On successful return holds the most significant part of the
+// file version. Must be non-null.
+// @param low On successful return holds the least significant part of the
+// file version. May be NULL.
+// @returns true if the version info was successfully retrieved.
+bool GetModuleVersion(HMODULE module, uint32* high, uint32* low);
+
+// Return if the IEXPLORE is in private mode. The IEIsInPrivateBrowsing() checks
+// whether current process is IEXPLORE.
+bool IsIEInPrivate();
+
+// Creates a copy of a menu. We need this when original menu comes from
+// a process with higher integrity.
+HMENU UtilCloneContextMenu(HMENU original_menu);
+
+// Uses GURL internally to append 'relative' to 'document'
+std::string ResolveURL(const std::string& document,
+ const std::string& relative);
+
+// Returns true iff the two urls have the same scheme, same host and same port.
+bool HaveSameOrigin(const std::string& url1, const std::string& url2);
+
+// Get a boolean configuration value from registry.
+bool GetConfigBool(bool default_value, const wchar_t* value_name);
+
+// Gets an integer configuration value from the registry.
+int GetConfigInt(int default_value, const wchar_t* value_name);
+
+// Check if this url is opting into Chrome Frame based on static settings.
+bool IsOptInUrl(const wchar_t* url);
+
+// A shortcut for QueryService
+template <typename T>
+HRESULT DoQueryService(const CLSID& class_id, IUnknown* unk, T** service) {
+ if (!unk)
+ return E_INVALIDARG;
+ ScopedComPtr<IServiceProvider> service_provider;
+ HRESULT hr = service_provider.QueryFrom(unk);
+ if (!service_provider)
+ return hr;
+
+ return service_provider->QueryService(class_id, service);
+}
+
+// Get url (display name) from a moniker, |bind_context| is optional
+HRESULT GetUrlFromMoniker(IMoniker* moniker, IBindCtx* bind_context,
+ std::wstring* url);
+
+// Returns true if the URL passed in is something which can be handled by
+// Chrome. If this function returns false then we should fail the navigation.
+bool IsValidUrlScheme(const std::wstring& url);
+
+// This returns the base directory in which to store user profiles.
+bool GetUserProfileBaseDirectory(std::wstring* path);
+
+#endif // CHROME_FRAME_UTILS_H_
diff --git a/chrome_frame/vectored_handler-impl.h b/chrome_frame/vectored_handler-impl.h
new file mode 100644
index 0000000..f850641
--- /dev/null
+++ b/chrome_frame/vectored_handler-impl.h
@@ -0,0 +1,106 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_VECTORED_HANDLER_IMPL_H_
+#define CHROME_FRAME_VECTORED_HANDLER_IMPL_H_
+#include "chrome_frame/vectored_handler.h"
+
+#if defined(_M_IX86)
+typedef struct _EXCEPTION_REGISTRATION_RECORD {
+ struct _EXCEPTION_REGISTRATION_RECORD* Next;
+ PVOID Handler;
+} EXCEPTION_REGISTRATION_RECORD;
+#define EXCEPTION_CHAIN_END ((struct _EXCEPTION_REGISTRATION_RECORD*)-1)
+#else
+#error only x86 is supported for now.
+#endif
+
+
+// VEH handler flags settings.
+// These are grabbed from winnt.h for PocketPC.
+// Only EXCEPTION_NONCONTINUABLE in defined in "regular" winnt.h
+// #define EXCEPTION_NONCONTINUABLE 0x1 // Noncontinuable exception
+#define EXCEPTION_UNWINDING 0x2 // Unwind is in progress
+#define EXCEPTION_EXIT_UNWIND 0x4 // Exit unwind is in progress
+#define EXCEPTION_STACK_INVALID 0x8 // Stack out of limits or unaligned
+#define EXCEPTION_NESTED_CALL 0x10 // Nested exception handler call
+#define EXCEPTION_TARGET_UNWIND 0x20 // Target unwind in progress
+#define EXCEPTION_COLLIDED_UNWIND 0x40 // Collided exception handler call
+
+#define EXCEPTION_UNWIND (EXCEPTION_UNWINDING | EXCEPTION_EXIT_UNWIND | \
+ EXCEPTION_TARGET_UNWIND | EXCEPTION_COLLIDED_UNWIND)
+
+#define IS_UNWINDING(Flag) (((Flag) & EXCEPTION_UNWIND) != 0)
+#define IS_DISPATCHING(Flag) (((Flag) & EXCEPTION_UNWIND) == 0)
+#define IS_TARGET_UNWIND(Flag) ((Flag) & EXCEPTION_TARGET_UNWIND)
+// End of grabbed section
+
+template <class E>
+LONG WINAPI VectoredHandlerT<E>::VectoredHandler(
+ EXCEPTION_POINTERS* exceptionInfo) {
+ // TODO(stoyan): Consider reentrancy
+ const DWORD exceptionCode = exceptionInfo->ExceptionRecord->ExceptionCode;
+
+ // Not interested in non-error exceptions. In this category falls exceptions
+ // like:
+ // 0x40010006 - OutputDebugStringA. Seen when no debugger is attached
+ // (otherwise debugger swallows the exception and prints
+ // the string).
+ // 0x406D1388 - DebuggerProbe. Used by debug CRT - for example see source
+ // code of isatty(). Used to name a thread as well.
+ // RPC_E_DISCONNECTED and Co. - COM IPC non-fatal warnings
+ // STATUS_BREAKPOINT and Co. - Debugger related breakpoints
+
+ if ((exceptionCode & ERROR_SEVERITY_ERROR) != ERROR_SEVERITY_ERROR) {
+ return ExceptionContinueSearch;
+ }
+
+ ++VectoredHandlerT<E>::g_exceptions_seen;
+
+ // TODO(stoyan): Check whether exception address is inbetween
+ // [IsBadReadPtr, IsBadReadPtr + 0xXX]
+
+ const DWORD exceptionFlags = exceptionInfo->ExceptionRecord->ExceptionFlags;
+ // I don't think VEH is called on unwind. Just to be safe.
+ if (IS_DISPATCHING(exceptionFlags)) {
+ if (ModuleHasInstalledSEHFilter())
+ return ExceptionContinueSearch;
+
+ if (E::IsOurModule(exceptionInfo->ExceptionRecord->ExceptionAddress)) {
+ E::WriteDump(exceptionInfo);
+ return ExceptionContinueSearch;
+ }
+
+ // See whether our module is somewhere in the call stack.
+ void* back_trace[max_back_trace] = {0};
+ // Skip RtlCaptureStackBackTrace and VectoredHandler itself.
+ DWORD captured = E::RtlCaptureStackBackTrace(2, max_back_trace - 2,
+ &back_trace[0], NULL);
+ for (DWORD i = 0; i < captured; ++i) {
+ if (E::IsOurModule(back_trace[i])) {
+ E::WriteDump(exceptionInfo);
+ return ExceptionContinueSearch;
+ }
+ }
+ }
+
+ return ExceptionContinueSearch;
+}
+
+template <class E>
+BOOL VectoredHandlerT<E>::ModuleHasInstalledSEHFilter() {
+ EXCEPTION_REGISTRATION_RECORD* RegistrationFrame = E::RtlpGetExceptionList();
+ // TODO(stoyan): Add the stack limits check and some sanity checks like
+ // decreasing addresses of registration records
+ while (RegistrationFrame != EXCEPTION_CHAIN_END) {
+ if (E::IsOurModule(RegistrationFrame->Handler)) {
+ return TRUE;
+ }
+
+ RegistrationFrame = RegistrationFrame->Next;
+ }
+
+ return FALSE;
+}
+#endif // CHROME_FRAME_VECTORED_HANDLER_IMPL_H_
diff --git a/chrome_frame/vectored_handler.h b/chrome_frame/vectored_handler.h
new file mode 100644
index 0000000..7351b6c
--- /dev/null
+++ b/chrome_frame/vectored_handler.h
@@ -0,0 +1,87 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_VECTORED_HANDLER_H_
+#define CHROME_FRAME_VECTORED_HANDLER_H_
+
+// Base class for VectoredHandlerT just to hold some members (independent of
+// template parameter)
+class VectoredHandlerBase {
+ public:
+ // For RtlCaptureStackBackTrace MSDN says:
+ // Windows Server 2003 and Windows XP: The sum of the FramesToSkip and
+ // FramesToCapture parameters must be less than 64.
+ // In practice (on XPSP2) it has to be less than 63, hence leaving us with
+ // max back trace of 62.
+ static const DWORD max_back_trace = 62;
+ static unsigned long g_exceptions_seen;
+ protected:
+ static void* g_handler;
+};
+
+DECLSPEC_SELECTANY void* VectoredHandlerBase::g_handler;
+DECLSPEC_SELECTANY unsigned long VectoredHandlerBase::g_exceptions_seen;
+
+// The E class is supposed to provide external/API functions. Using template
+// make testability easier. It shall confirm the following concept/archetype:
+// void* Register(PVECTORED_EXCEPTION_HANDLER,
+// const void* module_start, const void* module_end)
+// Registers Vectored Exception Handler, non-unittest implementation shall call
+// ::AddVectoredExceptionHandler Win32 API
+// ULONG Unregister(void*) - ::RemoveVectoredExceptionHandler Win32 API
+// int IsOurModule(const void* address) -
+// void WriteDump(EXCEPTION_POINTERS*) -
+// WORD RtlCaptureStackBackTrace(..) - same as Win32 API
+// EXCEPTION_REGISTRATION_RECORD* RtlpGetExceptionList() - same as Win32 API
+// You may want to derive own External class by deriving from
+// VEHExternalBase helper (see below).
+// Create dump policy:
+// 1. Scan SEH chain, if there is a handler/filter that belongs to our
+// module - assume we expect this one and hence do nothing here.
+// 2. If the address of the exception is in our module - create dump.
+// 3. If our module is in somewhere in callstack - create dump.
+template <class E>
+class VectoredHandlerT : public VectoredHandlerBase {
+ public:
+ static void* Register(const void* module_start, const void* module_end) {
+ g_exceptions_seen = 0;
+ g_handler = E::Register(&VectoredHandler, module_start, module_end);
+ return g_handler;
+ }
+
+ static ULONG Unregister() {
+ if (g_handler)
+ return E::Unregister(g_handler);
+ return 0;
+ }
+
+ static LONG WINAPI VectoredHandler(EXCEPTION_POINTERS* exceptionInfo);
+ private:
+ static BOOL ModuleHasInstalledSEHFilter();
+};
+
+// Handy class supposed to act as a base class for classes used as template
+// parameter of VectoredHandlerT<E>
+class VEHTraitsBase {
+ public:
+ static const void* g_module_start;
+ static const void* g_module_end;
+
+ static inline int IsOurModule(const void* address) {
+ return (g_module_start <= address && address < g_module_end);
+ }
+
+ static inline void SetModule(const void* module_start,
+ const void* module_end) {
+ g_module_start = module_start;
+ g_module_end = module_end;
+ }
+};
+
+DECLSPEC_SELECTANY const void* VEHTraitsBase::g_module_start;
+DECLSPEC_SELECTANY const void* VEHTraitsBase::g_module_end;
+
+class Win32VEHTraits;
+typedef class VectoredHandlerT<Win32VEHTraits> VectoredHandler;
+#endif // CHROME_FRAME_VECTORED_HANDLER_H_
diff --git a/chrome_frame/vtable_patch_manager.cc b/chrome_frame/vtable_patch_manager.cc
new file mode 100644
index 0000000..5f15158
--- /dev/null
+++ b/chrome_frame/vtable_patch_manager.cc
@@ -0,0 +1,82 @@
+// Copyright (c) 2009 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 "chrome_frame/vtable_patch_manager.h"
+
+#include "base/logging.h"
+
+#include "chrome_frame/function_stub.h"
+
+namespace vtable_patch {
+
+// Convenient definition of a VTABLE
+typedef PROC* Vtable;
+
+// Returns a pointer to the VTable of a COM interface.
+// @param unknown [in] The pointer of the COM interface.
+inline Vtable GetIFVTable(void* unknown) {
+ return reinterpret_cast<Vtable>(*reinterpret_cast<void**>(unknown));
+}
+
+HRESULT PatchInterfaceMethods(void* unknown, MethodPatchInfo* patches) {
+ // Do some sanity checking of the input arguments.
+ if (NULL == unknown || NULL == patches) {
+ NOTREACHED();
+ return E_INVALIDARG;
+ }
+
+ Vtable vtable = GetIFVTable(unknown);
+ DCHECK(vtable);
+
+ for (MethodPatchInfo* it = patches; it->index_ != -1; ++it) {
+ PROC original_fn = vtable[it->index_];
+ FunctionStub* stub = FunctionStub::FromCode(original_fn);
+ if (stub != NULL) {
+ DLOG(ERROR) << "attempt to patch a function that's already patched";
+ DCHECK(stub->absolute_target() ==
+ reinterpret_cast<uintptr_t>(it->method_)) <<
+ "patching the same method multiple times with different hooks?";
+ continue;
+ }
+
+ stub = FunctionStub::Create(reinterpret_cast<uintptr_t>(original_fn),
+ it->method_);
+ if (!stub) {
+ NOTREACHED();
+ return E_OUTOFMEMORY;
+ } else {
+ DWORD protect = 0;
+ if (::VirtualProtect(&vtable[it->index_], sizeof(PROC),
+ PAGE_EXECUTE_READWRITE, &protect)) {
+ it->stub_ = stub; // save the stub
+ vtable[it->index_] = stub->code();
+ ::VirtualProtect(&vtable[it->index_], sizeof(PROC), protect,
+ &protect);
+ } else {
+ NOTREACHED();
+ }
+ }
+ }
+
+ return S_OK;
+}
+
+HRESULT UnpatchInterfaceMethods(MethodPatchInfo* patches) {
+ for (MethodPatchInfo* it = patches; it->index_ != -1; ++it) {
+ if (it->stub_) {
+ DCHECK(it->stub_->absolute_target() ==
+ reinterpret_cast<uintptr_t>(it->method_));
+ // Modify the stub to just jump directly to the original function.
+ it->stub_->BypassStub(reinterpret_cast<void*>(it->stub_->argument()));
+ it->stub_ = NULL;
+ // Leave the stub in memory so that we won't break any possible chains.
+ } else {
+ DLOG(WARNING) << "attempt to unpatch a function that wasn't patched";
+ }
+ }
+
+ return S_OK;
+}
+
+} // namespace vtable_patch
diff --git a/chrome_frame/vtable_patch_manager.h b/chrome_frame/vtable_patch_manager.h
new file mode 100644
index 0000000..944b9ef
--- /dev/null
+++ b/chrome_frame/vtable_patch_manager.h
@@ -0,0 +1,64 @@
+// Copyright (c) 2009 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.
+
+#ifndef CHROME_FRAME_COMMON_VTABLE_PATCH_MANAGER_H_
+#define CHROME_FRAME_COMMON_VTABLE_PATCH_MANAGER_H_
+
+#include <windows.h>
+
+struct FunctionStub;
+// This namespace provides methods to patch VTable methods of COM interfaces.
+namespace vtable_patch {
+
+// This structure represents information about one VTable method.
+// We allocate an array of these structures per VTable that we patch to
+// remember the original method. We also use this structure to actually
+// describe the VTable patch functions
+struct MethodPatchInfo {
+ int index_;
+ PROC method_;
+ FunctionStub* stub_;
+};
+
+// Patches methods in the passed in COM interface. The indexes of the
+// methods to patch and the actual patch functions are described in the
+// array pointed to by patches.
+// @param[in] unknown The pointer of the COM interface to patch
+// @param[in] patches An array of MethodPatchInfo structures describing
+// the methods to patch and the patch functions.
+// The last entry of patches must have index_ set to -1.
+HRESULT PatchInterfaceMethods(void* unknown, MethodPatchInfo* patches);
+
+// Using the patch info provided in |patches| the function goes through the
+// list of patched methods and modifies thunks so that they no longer point
+// to a hook method but rather go straight through to the original target.
+// The thunk itself is not destroyed to support chaining.
+// @param[in] patches An array of MethodPatchInfo structures describing
+// the methods to patch and the patch functions.
+// The last entry of patches must have index_ set to -1.
+HRESULT UnpatchInterfaceMethods(MethodPatchInfo* patches);
+
+} // namespace vtable_patch
+
+// Begins the declaration of a VTable patch
+// @param IFName The name of the interface to patch
+#define BEGIN_VTABLE_PATCHES(IFName) \
+ vtable_patch::MethodPatchInfo IFName##_PatchInfo[] = {
+
+// Defines a single method patch in a VTable
+// @param index The index of the method to patch
+// @param PatchFunction The patch function
+#define VTABLE_PATCH_ENTRY(index, PatchFunction) {\
+ index, \
+ reinterpret_cast<PROC>(PatchFunction), \
+ NULL, \
+ },
+
+// Ends the declaration of a VTable patch by adding an entry with
+// index set to -1.
+#define END_VTABLE_PATCHES() \
+ -1, NULL, NULL \
+ };
+
+#endif // CHROME_FRAME_COMMON_VTABLE_PATCH_MANAGER_H_