display_change_observer_chromeos.cc revision 1320f92c476a1ad9d19dba2a48c72b75566198e9
1// Copyright 2013 The Chromium Authors. All rights reserved. 2// Use of this source code is governed by a BSD-style license that can be 3// found in the LICENSE file. 4 5#include "ash/display/display_change_observer_chromeos.h" 6 7#include <algorithm> 8#include <map> 9#include <set> 10#include <vector> 11 12#include "ash/ash_switches.h" 13#include "ash/display/display_info.h" 14#include "ash/display/display_layout_store.h" 15#include "ash/display/display_manager.h" 16#include "ash/shell.h" 17#include "ash/touch/touchscreen_util.h" 18#include "base/command_line.h" 19#include "base/logging.h" 20#include "grit/ash_strings.h" 21#include "ui/base/l10n/l10n_util.h" 22#include "ui/compositor/dip_util.h" 23#include "ui/display/types/display_mode.h" 24#include "ui/display/types/display_snapshot.h" 25#include "ui/display/util/display_util.h" 26#include "ui/events/device_data_manager.h" 27#include "ui/events/touchscreen_device.h" 28#include "ui/gfx/display.h" 29#include "ui/wm/core/user_activity_detector.h" 30 31namespace ash { 32 33using ui::DisplayConfigurator; 34 35namespace { 36 37// The DPI threshold to determine the device scale factor. 38// DPI higher than |dpi| will use |device_scale_factor|. 39struct DeviceScaleFactorDPIThreshold { 40 float dpi; 41 float device_scale_factor; 42}; 43 44const DeviceScaleFactorDPIThreshold kThresholdTable[] = { 45 {180.0f, 2.0f}, 46 {150.0f, 1.25f}, 47 {0.0f, 1.0f}, 48}; 49 50// 1 inch in mm. 51const float kInchInMm = 25.4f; 52 53// The minimum pixel width whose monitor can be called as '4K'. 54const int kMinimumWidthFor4K = 3840; 55 56// The list of device scale factors (in addition to 1.0f) which is 57// available in extrenal large monitors. 58const float kAdditionalDeviceScaleFactorsFor4k[] = {1.25f, 2.0f}; 59 60// Display mode list is sorted by: 61// * the area in pixels in ascending order 62// * refresh rate in descending order 63struct DisplayModeSorter { 64 bool operator()(const DisplayMode& a, const DisplayMode& b) { 65 gfx::Size size_a_dip = a.GetSizeInDIP(); 66 gfx::Size size_b_dip = b.GetSizeInDIP(); 67 if (size_a_dip.GetArea() == size_b_dip.GetArea()) 68 return (a.refresh_rate > b.refresh_rate); 69 return (size_a_dip.GetArea() < size_b_dip.GetArea()); 70 } 71}; 72 73} // namespace 74 75// static 76std::vector<DisplayMode> DisplayChangeObserver::GetInternalDisplayModeList( 77 const DisplayInfo& display_info, 78 const DisplayConfigurator::DisplayState& output) { 79 std::vector<DisplayMode> display_mode_list; 80 const ui::DisplayMode* ui_native_mode = output.display->native_mode(); 81 DisplayMode native_mode(ui_native_mode->size(), 82 ui_native_mode->refresh_rate(), 83 ui_native_mode->is_interlaced(), 84 true); 85 native_mode.device_scale_factor = display_info.device_scale_factor(); 86 std::vector<float> ui_scales = 87 DisplayManager::GetScalesForDisplay(display_info); 88 for (size_t i = 0; i < ui_scales.size(); ++i) { 89 DisplayMode mode = native_mode; 90 mode.ui_scale = ui_scales[i]; 91 mode.native = (ui_scales[i] == display_info.device_scale_factor()); 92 display_mode_list.push_back(mode); 93 } 94 95 std::sort( 96 display_mode_list.begin(), display_mode_list.end(), DisplayModeSorter()); 97 return display_mode_list; 98} 99 100// static 101std::vector<DisplayMode> DisplayChangeObserver::GetExternalDisplayModeList( 102 const DisplayConfigurator::DisplayState& output) { 103 typedef std::map<std::pair<int, int>, DisplayMode> DisplayModeMap; 104 DisplayModeMap display_mode_map; 105 106 DisplayMode native_mode; 107 for (std::vector<const ui::DisplayMode*>::const_iterator it = 108 output.display->modes().begin(); 109 it != output.display->modes().end(); 110 ++it) { 111 const ui::DisplayMode& mode_info = **it; 112 const std::pair<int, int> size(mode_info.size().width(), 113 mode_info.size().height()); 114 const DisplayMode display_mode(mode_info.size(), 115 mode_info.refresh_rate(), 116 mode_info.is_interlaced(), 117 output.display->native_mode() == *it); 118 if (display_mode.native) 119 native_mode = display_mode; 120 121 // Add the display mode if it isn't already present and override interlaced 122 // display modes with non-interlaced ones. 123 DisplayModeMap::iterator display_mode_it = display_mode_map.find(size); 124 if (display_mode_it == display_mode_map.end()) 125 display_mode_map.insert(std::make_pair(size, display_mode)); 126 else if (display_mode_it->second.interlaced && !display_mode.interlaced) 127 display_mode_it->second = display_mode; 128 } 129 130 std::vector<DisplayMode> display_mode_list; 131 for (DisplayModeMap::const_iterator iter = display_mode_map.begin(); 132 iter != display_mode_map.end(); 133 ++iter) { 134 display_mode_list.push_back(iter->second); 135 } 136 137 if (native_mode.size.width() >= kMinimumWidthFor4K) { 138 for (size_t i = 0; i < arraysize(kAdditionalDeviceScaleFactorsFor4k); 139 ++i) { 140 DisplayMode mode = native_mode; 141 mode.device_scale_factor = kAdditionalDeviceScaleFactorsFor4k[i]; 142 mode.native = false; 143 display_mode_list.push_back(mode); 144 } 145 } 146 147 std::sort( 148 display_mode_list.begin(), display_mode_list.end(), DisplayModeSorter()); 149 return display_mode_list; 150} 151 152DisplayChangeObserver::DisplayChangeObserver() { 153 Shell::GetInstance()->AddShellObserver(this); 154 ui::DeviceDataManager::GetInstance()->AddObserver(this); 155} 156 157DisplayChangeObserver::~DisplayChangeObserver() { 158 ui::DeviceDataManager::GetInstance()->RemoveObserver(this); 159 Shell::GetInstance()->RemoveShellObserver(this); 160} 161 162ui::MultipleDisplayState DisplayChangeObserver::GetStateForDisplayIds( 163 const std::vector<int64>& display_ids) const { 164 CHECK_EQ(2U, display_ids.size()); 165 DisplayIdPair pair = std::make_pair(display_ids[0], display_ids[1]); 166 DisplayLayout layout = Shell::GetInstance()->display_manager()-> 167 layout_store()->GetRegisteredDisplayLayout(pair); 168 return layout.mirrored ? ui::MULTIPLE_DISPLAY_STATE_DUAL_MIRROR : 169 ui::MULTIPLE_DISPLAY_STATE_DUAL_EXTENDED; 170} 171 172bool DisplayChangeObserver::GetResolutionForDisplayId(int64 display_id, 173 gfx::Size* size) const { 174 DisplayMode mode; 175 if (!Shell::GetInstance()->display_manager()->GetSelectedModeForDisplayId( 176 display_id, &mode)) 177 return false; 178 179 *size = mode.size; 180 return true; 181} 182 183void DisplayChangeObserver::OnDisplayModeChanged( 184 const std::vector<DisplayConfigurator::DisplayState>& display_states) { 185 std::vector<DisplayInfo> displays; 186 std::set<int64> ids; 187 for (size_t i = 0; i < display_states.size(); ++i) { 188 const DisplayConfigurator::DisplayState& state = display_states[i]; 189 190 if (state.display->type() == ui::DISPLAY_CONNECTION_TYPE_INTERNAL && 191 gfx::Display::InternalDisplayId() == gfx::Display::kInvalidDisplayID) { 192 gfx::Display::SetInternalDisplayId(state.display->display_id()); 193 } 194 195 const ui::DisplayMode* mode_info = state.display->current_mode(); 196 if (!mode_info) 197 continue; 198 199 float device_scale_factor = 1.0f; 200 if (state.display->type() == ui::DISPLAY_CONNECTION_TYPE_INTERNAL) { 201 if (!ui::IsDisplaySizeBlackListed(state.display->physical_size())) { 202 device_scale_factor = 203 FindDeviceScaleFactor((kInchInMm * mode_info->size().width() / 204 state.display->physical_size().width())); 205 } 206 } else { 207 DisplayMode mode; 208 if (Shell::GetInstance()->display_manager()->GetSelectedModeForDisplayId( 209 state.display->display_id(), &mode)) { 210 device_scale_factor = mode.device_scale_factor; 211 } 212 } 213 gfx::Rect display_bounds(state.display->origin(), mode_info->size()); 214 215 std::string name = 216 state.display->type() == ui::DISPLAY_CONNECTION_TYPE_INTERNAL ? 217 l10n_util::GetStringUTF8(IDS_ASH_INTERNAL_DISPLAY_NAME) : 218 state.display->display_name(); 219 if (name.empty()) 220 name = l10n_util::GetStringUTF8(IDS_ASH_STATUS_TRAY_UNKNOWN_DISPLAY_NAME); 221 222 bool has_overscan = state.display->has_overscan(); 223 int64 id = state.display->display_id(); 224 ids.insert(id); 225 226 displays.push_back(DisplayInfo(id, name, has_overscan)); 227 DisplayInfo& new_info = displays.back(); 228 new_info.set_device_scale_factor(device_scale_factor); 229 new_info.SetBounds(display_bounds); 230 new_info.set_native(true); 231 new_info.set_is_aspect_preserving_scaling( 232 state.display->is_aspect_preserving_scaling()); 233 234 std::vector<DisplayMode> display_modes = 235 (state.display->type() == ui::DISPLAY_CONNECTION_TYPE_INTERNAL) ? 236 GetInternalDisplayModeList(new_info, state) : 237 GetExternalDisplayModeList(state); 238 new_info.set_display_modes(display_modes); 239 240 new_info.set_available_color_profiles( 241 Shell::GetInstance() 242 ->display_configurator() 243 ->GetAvailableColorCalibrationProfiles(id)); 244 } 245 246 AssociateTouchscreens( 247 &displays, ui::DeviceDataManager::GetInstance()->touchscreen_devices()); 248 // DisplayManager can be null during the boot. 249 Shell::GetInstance()->display_manager()->OnNativeDisplaysChanged(displays); 250 251 // For the purposes of user activity detection, ignore synthetic mouse events 252 // that are triggered by screen resizes: http://crbug.com/360634 253 ::wm::UserActivityDetector* user_activity_detector = 254 Shell::GetInstance()->user_activity_detector(); 255 if (user_activity_detector) 256 user_activity_detector->OnDisplayPowerChanging(); 257} 258 259void DisplayChangeObserver::OnAppTerminating() { 260#if defined(USE_ASH) 261 // Stop handling display configuration events once the shutdown 262 // process starts. crbug.com/177014. 263 Shell::GetInstance()->display_configurator()->PrepareForExit(); 264#endif 265} 266 267// static 268float DisplayChangeObserver::FindDeviceScaleFactor(float dpi) { 269 for (size_t i = 0; i < arraysize(kThresholdTable); ++i) { 270 if (dpi > kThresholdTable[i].dpi) 271 return kThresholdTable[i].device_scale_factor; 272 } 273 return 1.0f; 274} 275 276void DisplayChangeObserver::OnInputDeviceConfigurationChanged() { 277 std::vector<DisplayInfo> display_infos; 278 DisplayManager* display_manager = 279 ash::Shell::GetInstance()->display_manager(); 280 const std::vector<gfx::Display>& displays = display_manager->displays(); 281 // Reuse the current state in DisplayManager and re-associate the displays 282 // with the touchscreens. 283 for (size_t i = 0; i < displays.size(); ++i) { 284 DisplayInfo display = display_manager->GetDisplayInfo(displays[i].id()); 285 // Unset the touchscreen configuration since we'll be rematching them from 286 // scratch. 287 display.set_touch_device_id(ui::TouchscreenDevice::kInvalidId); 288 display.set_touch_support(gfx::Display::TOUCH_SUPPORT_UNKNOWN); 289 290 display_infos.push_back(display); 291 } 292 293 AssociateTouchscreens( 294 &display_infos, 295 ui::DeviceDataManager::GetInstance()->touchscreen_devices()); 296 display_manager->OnNativeDisplaysChanged(display_infos); 297} 298 299} // namespace ash 300