sky_binding.dart 4.7 KB
Newer Older
1 2 3 4 5 6
// Copyright 2015 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.

import 'dart:sky' as sky;

Adam Barth's avatar
Adam Barth committed
7
import 'package:sky/base/pointer_router.dart';
8
import 'package:sky/base/hit_test.dart';
9
import 'package:sky/base/scheduler.dart' as scheduler;
10 11
import 'package:sky/rendering/box.dart';
import 'package:sky/rendering/object.dart';
12
import 'package:sky/rendering/view.dart';
13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43

int _hammingWeight(int value) {
  if (value == 0)
    return 0;
  int weight = 0;
  for (int i = 0; i < value.bitLength; ++i) {
    if (value & (1 << i) != 0)
      ++weight;
  }
  return weight;
}

class PointerState {
  PointerState({ this.result, this.lastPosition });
  HitTestResult result;
  Point lastPosition;
}

typedef void EventListener(sky.Event event);

class SkyBinding {

  SkyBinding({ RenderBox root: null, RenderView renderViewOverride }) {
    assert(_instance == null);
    _instance = this;

    sky.view.setEventCallback(_handleEvent);

    sky.view.setMetricsChangedCallback(_handleMetricsChanged);
    scheduler.init();
    if (renderViewOverride == null) {
44
      _renderView = new RenderView(child: root);
45 46
      _renderView.attach();
      _renderView.rootConstraints = _createConstraints();
47
      _renderView.scheduleInitialFrame();
48 49 50 51
    } else {
      _renderView = renderViewOverride;
    }
    assert(_renderView != null);
52
    scheduler.addPersistentFrameCallback(beginFrame);
53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73

    assert(_instance == this);
  }

  static SkyBinding _instance; // used to enforce that we're a singleton
  static SkyBinding get instance => _instance;

  RenderView _renderView;
  RenderView get renderView => _renderView;

  ViewConstraints _createConstraints() {
    return new ViewConstraints(size: new Size(sky.view.width, sky.view.height));
  }
  void _handleMetricsChanged() {
    _renderView.rootConstraints = _createConstraints();
  }

  RenderBox get root => _renderView.child;
  void set root(RenderBox value) {
    _renderView.child = value;
  }
74
  void beginFrame(double timeStamp) {
75
    RenderObject.flushLayout();
Hixie's avatar
Hixie committed
76
    _renderView.updateCompositingBits();
77
    RenderObject.flushPaint();
78
    _renderView.compositeFrame();
79 80 81 82 83 84 85 86 87 88
  }

  final List<EventListener> _eventListeners = new List<EventListener>();
  void addEventListener(EventListener e) => _eventListeners.add(e);
  bool removeEventListener(EventListener e) => _eventListeners.remove(e);

  void _handleEvent(sky.Event event) {
    if (event is sky.PointerEvent) {
      _handlePointerEvent(event);
    } else if (event is sky.GestureEvent) {
89
      dispatchEvent(event, hitTest(new Point(event.x, event.y)));
90 91 92 93 94 95
    } else {
      for (EventListener e in _eventListeners)
        e(event);
    }
  }

Adam Barth's avatar
Adam Barth committed
96 97
  final PointerRouter pointerRouter = new PointerRouter();

98 99 100
  Map<int, PointerState> _stateForPointer = new Map<int, PointerState>();

  PointerState _createStateForPointer(sky.PointerEvent event, Point position) {
101
    HitTestResult result = hitTest(position);
102 103 104 105 106
    PointerState state = new PointerState(result: result, lastPosition: position);
    _stateForPointer[event.pointer] = state;
    return state;
  }

107 108 109 110 111 112 113
  PointerState _getOrCreateStateForPointer(event, position) {
    PointerState state = _stateForPointer[event.pointer];
    if (state == null)
      state = _createStateForPointer(event, position);
    return state;
  }

114
  EventDisposition _handlePointerEvent(sky.PointerEvent event) {
115 116
    Point position = new Point(event.x, event.y);

117 118 119 120 121
    PointerState state = _getOrCreateStateForPointer(event, position);

    if (event.type == 'pointerup' || event.type == 'pointercancel') {
      if (_hammingWeight(event.buttons) <= 1)
        _stateForPointer.remove(event.pointer);
122
    }
123

124 125 126 127
    event.dx = position.x - state.lastPosition.x;
    event.dy = position.y - state.lastPosition.y;
    state.lastPosition = position;

128
    return dispatchEvent(event, state.result);
129 130
  }

131 132
  HitTestResult hitTest(Point position) {
    HitTestResult result = new HitTestResult();
Adam Barth's avatar
Adam Barth committed
133
    result.add(new HitTestEntry(pointerRouter));
134 135 136 137
    _renderView.hitTest(result, position: position);
    return result;
  }

138
  EventDisposition dispatchEvent(sky.Event event, HitTestResult result) {
139
    assert(result != null);
140 141 142 143 144 145 146 147 148
    EventDisposition disposition = EventDisposition.ignored;
    for (HitTestEntry entry in result.path.reversed) {
      EventDisposition entryDisposition = entry.target.handleEvent(event, entry);
      if (entryDisposition == EventDisposition.consumed)
        return EventDisposition.consumed;
      else if (entryDisposition == EventDisposition.processed)
        disposition = EventDisposition.processed;
    }
    return disposition;
149 150 151 152 153 154 155 156 157
  }

  String toString() => 'Render Tree:\n${_renderView}';

  void debugDumpRenderTree() {
    toString().split('\n').forEach(print);
  }

}