1
2
3
4
5
6
7
8
9
10
11
12
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
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
|
// Copyright (c) 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.
#include "ui/gfx/paint_throbber.h"
#include "base/time/time.h"
#include "ui/gfx/animation/tween.h"
#include "ui/gfx/canvas.h"
#include "ui/gfx/color_utils.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/gfx/skia_util.h"
namespace gfx {
namespace {
// The maximum size of the "spinning" state arc, in degrees.
const int64_t kMaxArcSize = 270;
// The amount of time it takes to grow the "spinning" arc from 0 to 270 degrees.
const int64_t kArcTimeMs = 666;
// The amount of time it takes for the "spinning" throbber to make a full
// rotation.
const int64_t kRotationTimeMs = 1568;
void PaintArc(Canvas* canvas,
const Rect& bounds,
SkColor color,
SkScalar start_angle,
SkScalar sweep) {
// Stroke width depends on size.
// . For size < 28: 3 - (28 - size) / 16
// . For 28 <= size: (8 + size) / 12
SkScalar stroke_width = bounds.width() < 28
? 3.0 - SkIntToScalar(28 - bounds.width()) / 16.0
: SkIntToScalar(bounds.width() + 8) / 12.0;
Rect oval = bounds;
// Inset by half the stroke width to make sure the whole arc is inside
// the visible rect.
int inset = SkScalarCeilToInt(stroke_width / 2.0);
oval.Inset(inset, inset);
SkPath path;
path.arcTo(RectToSkRect(oval), start_angle, sweep, true);
SkPaint paint;
paint.setColor(color);
paint.setStrokeCap(SkPaint::kRound_Cap);
paint.setStrokeWidth(stroke_width);
paint.setStyle(SkPaint::kStroke_Style);
paint.setAntiAlias(true);
canvas->DrawPath(path, paint);
}
void CalculateWaitingAngles(const base::TimeDelta& elapsed_time,
int64_t* start_angle,
int64_t* sweep) {
// Calculate start and end points. The angles are counter-clockwise because
// the throbber spins counter-clockwise. The finish angle starts at 12 o'clock
// (90 degrees) and rotates steadily. The start angle trails 180 degrees
// behind, except for the first half revolution, when it stays at 12 o'clock.
base::TimeDelta revolution_time = base::TimeDelta::FromMilliseconds(1320);
int64_t twelve_oclock = 90;
int64_t finish_angle_cc =
twelve_oclock + 360 * elapsed_time / revolution_time;
int64_t start_angle_cc = std::max(finish_angle_cc - 180, twelve_oclock);
// Negate the angles to convert to the clockwise numbers Skia expects.
if (start_angle)
*start_angle = -finish_angle_cc;
if (sweep)
*sweep = finish_angle_cc - start_angle_cc;
}
// This is a Skia port of the MD spinner SVG. The |start_angle| rotation
// here corresponds to the 'rotate' animation.
void PaintThrobberSpinningWithStartAngle(Canvas* canvas,
const Rect& bounds,
SkColor color,
const base::TimeDelta& elapsed_time,
int64_t start_angle) {
// The sweep angle ranges from -270 to 270 over 1333ms. CSS
// animation timing functions apply in between key frames, so we have to
// break up the 1333ms into two keyframes (-270 to 0, then 0 to 270).
base::TimeDelta arc_time = base::TimeDelta::FromMilliseconds(kArcTimeMs);
double arc_size_progress = static_cast<double>(elapsed_time.InMicroseconds() %
arc_time.InMicroseconds()) /
arc_time.InMicroseconds();
// This tween is equivalent to cubic-bezier(0.4, 0.0, 0.2, 1).
double sweep = kMaxArcSize * Tween::CalculateValue(Tween::FAST_OUT_SLOW_IN,
arc_size_progress);
int64_t sweep_keyframe = (elapsed_time / arc_time) % 2;
if (sweep_keyframe == 0)
sweep -= kMaxArcSize;
// This part makes sure the sweep is at least 5 degrees long. Roughly
// equivalent to the "magic constants" in SVG's fillunfill animation.
const double min_sweep_length = 5.0;
if (sweep >= 0.0 && sweep < min_sweep_length) {
start_angle -= (min_sweep_length - sweep);
sweep = min_sweep_length;
} else if (sweep <= 0.0 && sweep > -min_sweep_length) {
start_angle += (-min_sweep_length - sweep);
sweep = -min_sweep_length;
}
// To keep the sweep smooth, we have an additional rotation after each
// |arc_time| period has elapsed. See SVG's 'rot' animation.
int64_t rot_keyframe = (elapsed_time / (arc_time * 2)) % 4;
PaintArc(canvas, bounds, color, start_angle + rot_keyframe * kMaxArcSize,
sweep);
}
} // namespace
void PaintThrobberSpinning(Canvas* canvas,
const Rect& bounds,
SkColor color,
const base::TimeDelta& elapsed_time) {
base::TimeDelta rotation_time =
base::TimeDelta::FromMilliseconds(kRotationTimeMs);
int64_t start_angle = 270 + 360 * elapsed_time / rotation_time;
PaintThrobberSpinningWithStartAngle(canvas, bounds, color, elapsed_time,
start_angle);
}
void PaintThrobberWaiting(Canvas* canvas,
const Rect& bounds, SkColor color, const base::TimeDelta& elapsed_time) {
int64_t start_angle = 0, sweep = 0;
CalculateWaitingAngles(elapsed_time, &start_angle, &sweep);
PaintArc(canvas, bounds, color, start_angle, sweep);
}
void PaintThrobberSpinningAfterWaiting(Canvas* canvas,
const Rect& bounds,
SkColor color,
const base::TimeDelta& elapsed_time,
ThrobberWaitingState* waiting_state) {
int64_t waiting_start_angle = 0, waiting_sweep = 0;
CalculateWaitingAngles(waiting_state->elapsed_time, &waiting_start_angle,
&waiting_sweep);
// |arc_time_offset| is the effective amount of time one would have to wait
// for the "spinning" sweep to match |waiting_sweep|. Brute force calculation.
if (waiting_state->arc_time_offset == base::TimeDelta()) {
for (int64_t arc_time_it = 0; arc_time_it <= kArcTimeMs; ++arc_time_it) {
double arc_size_progress = static_cast<double>(arc_time_it) / kArcTimeMs;
if (kMaxArcSize * Tween::CalculateValue(Tween::FAST_OUT_SLOW_IN,
arc_size_progress) >=
waiting_sweep) {
// Add kArcTimeMs to sidestep the |sweep_keyframe == 0| offset below.
waiting_state->arc_time_offset =
base::TimeDelta::FromMilliseconds(arc_time_it + kArcTimeMs);
break;
}
}
}
// Blend the color between "waiting" and "spinning" states.
base::TimeDelta color_fade_time = base::TimeDelta::FromMilliseconds(900);
double color_progress = 1.0;
if (elapsed_time < color_fade_time) {
color_progress = Tween::CalculateValue(
Tween::LINEAR_OUT_SLOW_IN,
static_cast<double>(elapsed_time.InMicroseconds()) /
color_fade_time.InMicroseconds());
}
SkColor blend_color = color_utils::AlphaBlend(color, waiting_state->color,
color_progress * 255);
int64_t start_angle =
waiting_start_angle +
360 * elapsed_time / base::TimeDelta::FromMilliseconds(kRotationTimeMs);
base::TimeDelta effective_elapsed_time =
elapsed_time + waiting_state->arc_time_offset;
PaintThrobberSpinningWithStartAngle(canvas, bounds, blend_color,
effective_elapsed_time, start_angle);
}
} // namespace gfx
|