Chromium Code Reviews| Index: athena/system/orientation_controller.cc |
| diff --git a/athena/system/orientation_controller.cc b/athena/system/orientation_controller.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..64a1a9be7a6e08777128904a6d2f09980f65a11f |
| --- /dev/null |
| +++ b/athena/system/orientation_controller.cc |
| @@ -0,0 +1,133 @@ |
| +// Copyright 2014 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 "athena/screen/public/screen_manager.h" |
| +#include "athena/system/orientation_controller.h" |
| +#include "base/bind.h" |
| +#include "base/file_util.h" |
| +#include "base/files/file_path_watcher.h" |
| +#include "base/message_loop/message_loop.h" |
| +#include "base/task_runner.h" |
| + |
| +namespace athena { |
| + |
| +namespace { |
| + |
| +// Path of the socket which the sensor daemon creates. |
| +const char kSocketPath[] = "/dev/sensors/orientation"; |
| + |
| +// Threshold after which to rotate in a given direction. |
| +const int kGravityThreshold = 6.0f; |
| + |
| +// Minimum delay before triggering another orientation change. |
| +const int kOrientationChangeDelayNS = 500000000; |
| + |
| +enum { |
| + SENSOR_ACCELEROMETER, |
| + SENSOR_LIGHT, |
| + SENSOR_PROXIMITY |
| +}; |
| + |
| +// A sensor event from the device. |
| +struct DeviceSensorEvent { |
| + // The type of event from the enum above. |
| + int32_t type; |
| + |
| + // The time in nanoseconds at which the event happened. |
| + int64_t timestamp; |
| + |
| + union { |
| + // Accelerometer X,Y,Z values in SI units (m/s^2) including gravity. |
| + // The orientation is described at |
| + // http://www.html5rocks.com/en/tutorials/device/orientation/. |
| + float data[3]; |
| + |
| + // Ambient (room) temperature in degrees Celcius. |
| + float temperature; |
| + |
| + // Proximity sensor distance in centimeters. |
| + float distance; |
| + |
| + // Ambient light level in SI lux units. |
| + float light; |
| + }; |
| +}; |
| + |
| +} // namespace |
| + |
| +OrientationController::OrientationController() |
| + : DeviceSocketListener(kSocketPath, sizeof(DeviceSensorEvent)), |
| + last_orientation_change_time_(0), |
| + weak_factory_(this) { |
| + CHECK(base::MessageLoop::current()); |
|
oshima
2014/08/11 21:30:17
base::MessageLoopForUI::current() ?
flackr
2014/08/13 15:52:36
Done.
|
| + ui_task_runner_ = base::MessageLoop::current()->task_runner(); |
| + base::FilePath socket_path(kSocketPath); |
| + if (base::PathExists(socket_path)) { |
| + StartListening(); |
| + } else { |
| + // Watch for socket to be created if it doesn't already exist. |
| + watcher_.reset(new base::FilePathWatcher); |
| + watcher_->Watch(socket_path, false, |
| + base::Bind(&OrientationController::OnFilePathChanged, |
| + weak_factory_.GetWeakPtr())); |
|
oshima
2014/08/11 21:30:17
Looks like FilePathWacher::Watch has to be called
flackr
2014/08/13 15:52:36
Done.
|
| + } |
| +} |
| + |
| +OrientationController::~OrientationController() { |
| +} |
| + |
| +void OrientationController::OnDataAvailableOnIO(const void* data) { |
| + const DeviceSensorEvent* event = |
| + static_cast<const DeviceSensorEvent*>(data); |
| + if (event->type != SENSOR_ACCELEROMETER) |
| + return; |
| + |
| + float gravity_x = event->data[0]; |
| + float gravity_y = event->data[1]; |
| + gfx::Display::Rotation rotation; |
| + if (gravity_x < -kGravityThreshold) { |
| + rotation = gfx::Display::ROTATE_270; |
| + } else if (gravity_x > kGravityThreshold) { |
| + rotation = gfx::Display::ROTATE_90; |
| + } else if (gravity_y < -kGravityThreshold) { |
| + rotation = gfx::Display::ROTATE_180; |
| + } else if (gravity_y > kGravityThreshold) { |
| + rotation = gfx::Display::ROTATE_0; |
| + } else { |
| + // No rotation as gravity threshold was not hit. |
| + return; |
| + } |
| + |
| + if (rotation == current_rotation_ || |
| + event->timestamp - last_orientation_change_time_ < |
| + kOrientationChangeDelayNS) { |
| + return; |
| + } |
| + |
| + last_orientation_change_time_ = event->timestamp; |
| + current_rotation_ = rotation; |
| + |
| + // TODO(flackr): Avoid callbacks using unretained pointers. This could |
| + // technically call RotateOnUI after OrientationController has destructed. We |
| + // should instead use callbacks with weak pointers similar to |
| + // chromeos/accelerometer/accelerometer_reader.cc. |
| + ui_task_runner_->PostTask(FROM_HERE, |
| + base::Bind(&OrientationController::RotateOnUI, |
| + base::Unretained(this), rotation)); |
| +} |
| + |
| +void OrientationController::RotateOnUI(gfx::Display::Rotation rotation) { |
| + ScreenManager::Get()->SetRotation(rotation); |
| +} |
| + |
| +void OrientationController::OnFilePathChanged(const base::FilePath& path, |
| + bool error) { |
| + if (error) |
| + return; |
| + |
| + StartListening(); |
| + watcher_.reset(); |
| +} |
| + |
| +} // namespace athena |