15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/browser/android/touch_point.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/debug/debugger.h"
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/logging.h"
9eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch#include "base/time/time.h"
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "jni/TouchPoint_jni.h"
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using WebKit::WebTouchEvent;
145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using WebKit::WebTouchPoint;
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace {
175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void MaybeAddTouchPoint(JNIEnv* env,
192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                        jobject pt,
202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                        float dpi_scale,
212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                        WebKit::WebTouchEvent& event) {
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  WebTouchPoint::State state = static_cast<WebTouchPoint::State>(
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      Java_TouchPoint_getState(env, pt));
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (state == WebTouchPoint::StateUndefined)
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // When generating a cancel event from an event of a different type, the
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // touch points are out of sync, so we ensure the points are marked as
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // canceled as well.
305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (event.type == WebTouchEvent::TouchCancel)
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    state = WebTouchPoint::StateCancelled;
325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Record the current number of points in the WebTouchEvent
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const int idx = event.touchesLength;
355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK_LT(idx, WebKit::WebTouchEvent::touchesLengthCap);
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  WebTouchPoint wtp;
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wtp.id = Java_TouchPoint_getId(env, pt);
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wtp.state = state;
402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  wtp.position.x = Java_TouchPoint_getX(env, pt) / dpi_scale;
412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  wtp.position.y = Java_TouchPoint_getY(env, pt) / dpi_scale;
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TODO(joth): Raw event co-ordinates.
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wtp.screenPosition = wtp.position;
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wtp.force = Java_TouchPoint_getPressure(env, pt);
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TODO(djsollen): WebKit stores touch point size as a pair of radii, which
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // are integers.  We receive touch point size from Android as a float
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // between 0 and 1 and interpret 'size' as an elliptical area.  We convert
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // size to a radius and then scale up to avoid truncating away all of the
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // data. W3C spec is for the radii to be in units of screen pixels. Need to
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // change.
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const static double PI = 3.1415926;
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const static double SCALE_FACTOR = 1024.0;
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const int radius = static_cast<int>(
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      (sqrt(Java_TouchPoint_getSize(env, pt)) / PI) * SCALE_FACTOR);
562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  wtp.radiusX = radius / dpi_scale;
572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  wtp.radiusY = radius / dpi_scale;
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Since our radii are equal, a rotation angle doesn't mean anything.
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  wtp.rotationAngle = 0.0;
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Add the newly created WebTouchPoint to the event
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  event.touches[idx] = wtp;
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  ++(event.touchesLength);
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace content {
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void TouchPoint::BuildWebTouchEvent(JNIEnv* env,
712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                    jint type,
722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                    jlong time_ms,
732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                    float dpi_scale,
742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                    jobjectArray pts,
752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                    WebKit::WebTouchEvent& event) {
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  event.type = static_cast<WebTouchEvent::Type>(type);
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  event.timeStampSeconds =
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      static_cast<double>(time_ms) / base::Time::kMillisecondsPerSecond;
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int arrayLength = env->GetArrayLength(pts);
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Loop until either all of the input points have been consumed or the output
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // array has been filled
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (int i = 0; i < arrayLength; i++) {
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    jobject pt = env->GetObjectArrayElement(pts, i);
842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    MaybeAddTouchPoint(env, pt, dpi_scale, event);
855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (event.touchesLength >= event.touchesLengthCap)
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      break;
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK_GT(event.touchesLength, 0U);
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)static void RegisterConstants(JNIEnv* env) {
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   Java_TouchPoint_initializeConstants(
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       env,
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchEvent::TouchStart,
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchEvent::TouchMove,
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchEvent::TouchEnd,
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchEvent::TouchCancel,
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchPoint::StateUndefined,
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchPoint::StateReleased,
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchPoint::StatePressed,
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchPoint::StateMoved,
1025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchPoint::StateStationary,
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       WebKit::WebTouchPoint::StateCancelled);
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool RegisterTouchPoint(JNIEnv* env) {
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!RegisterNativesImpl(env))
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  RegisterConstants(env);
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace content
116