mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
Ideally the scrollbar would animate out when you stop scrolling, but we're not quite that good yet. R=ojan@chromium.org Review URL: https://codereview.chromium.org/882583002
121 lines
3.1 KiB
Plaintext
121 lines
3.1 KiB
Plaintext
<!--
|
|
// 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 src="/sky/framework/sky-element/sky-element.sky" as="SkyElement" />
|
|
<import src="/sky/framework/fling-curve.sky" as="FlingCurve" />
|
|
|
|
<sky-element
|
|
name="sky-scrollable"
|
|
on-gesturescrollstart="handleScrollStart_"
|
|
on-gesturescrollend="handleScrollEnd_"
|
|
on-gesturescrollupdate="handleScrollUpdate_"
|
|
on-gestureflingstart="handleFlingStart_"
|
|
on-gestureflingcancel="handleFlingCancel_">
|
|
<template>
|
|
<style>
|
|
:host {
|
|
overflow: hidden;
|
|
position: relative;
|
|
}
|
|
#scrollable {
|
|
transform: translateY(0);
|
|
}
|
|
#vbar {
|
|
position: absolute;
|
|
right: 0;
|
|
width: 3px;
|
|
background-color: lightgray;
|
|
pointer-events: none;
|
|
top: 0;
|
|
height: 0;
|
|
}
|
|
</style>
|
|
<div id="vbar" />
|
|
<div id="scrollable">
|
|
<content />
|
|
</div>
|
|
</template>
|
|
<script>
|
|
module.exports = class extends SkyElement {
|
|
created() {
|
|
this.scrollable_ = null;
|
|
this.vbar_ = null;
|
|
this.scrollOffset_ = 0;
|
|
this.flingCurve_ = null;
|
|
this.flingAnimationId_ = null;
|
|
}
|
|
|
|
shadowRootReady() {
|
|
this.scrollable_ = this.shadowRoot.getElementById('scrollable');
|
|
this.vbar_ = this.shadowRoot.getElementById('vbar');
|
|
}
|
|
|
|
get scrollOffset() {
|
|
return this.scrollOffset_;
|
|
}
|
|
|
|
set scrollOffset(value) {
|
|
// TODO(abarth): Can we get these values without forcing a synchronous layout?
|
|
var outerHeight = this.clientHeight;
|
|
var innerHeight = this.scrollable_.clientHeight;
|
|
var scrollRange = innerHeight - outerHeight;
|
|
var newScrollOffset = Math.max(0, Math.min(scrollRange, value));
|
|
if (newScrollOffset == this.scrollOffset_)
|
|
return;
|
|
this.scrollOffset_ = newScrollOffset;
|
|
var transform = 'translateY(' + -this.scrollOffset_.toFixed(2) + 'px)';
|
|
this.scrollable_.style.transform = transform;
|
|
|
|
var topPercent = newScrollOffset / innerHeight * 100;
|
|
var heightPercent = outerHeight / innerHeight * 100;
|
|
this.vbar_.style.top = topPercent + '%';
|
|
this.vbar_.style.height = heightPercent + '%';
|
|
}
|
|
|
|
scrollBy(scrollDelta) {
|
|
var oldScrollOffset = this.scrollOffset_;
|
|
this.scrollOffset += scrollDelta;
|
|
return this.scrollOffset_ != oldScrollOffset;
|
|
}
|
|
|
|
scheduleFlingUpdate_() {
|
|
this.flingAnimationId_ = requestAnimationFrame(this.updateFling_.bind(this));
|
|
}
|
|
|
|
stopFling_() {
|
|
cancelAnimationFrame(this.flingAnimationId_);
|
|
this.flingCurve_ = null;
|
|
this.flingAnimationId_ = null;
|
|
}
|
|
|
|
updateFling_(timeStamp) {
|
|
var scrollDelta = this.flingCurve_.update(timeStamp);
|
|
if (!scrollDelta || !this.scrollBy(scrollDelta))
|
|
return this.stopFling_();
|
|
this.scheduleFlingUpdate_();
|
|
}
|
|
|
|
handleScrollStart_(event) {
|
|
}
|
|
|
|
handleScrollEnd_(event) {
|
|
}
|
|
|
|
handleScrollUpdate_(event) {
|
|
this.scrollBy(-event.dy);
|
|
}
|
|
|
|
handleFlingStart_(event) {
|
|
this.flingCurve_ = new FlingCurve(-event.velocityY, event.timeStamp);
|
|
this.scheduleFlingUpdate_();
|
|
}
|
|
|
|
handleFlingCancel_(event) {
|
|
this.stopFling_();
|
|
}
|
|
}.register();
|
|
</script>
|
|
</sky-element>
|