Commit 4b60def2 authored by Jim Beveridge's avatar Jim Beveridge

Fix slow sound init in piano sample app

Fixes #1696. Also fixes handles leaking on error.
parent de395582
...@@ -2,6 +2,8 @@ ...@@ -2,6 +2,8 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
import 'dart:async';
import 'package:mojo/mojo/url_response.mojom.dart'; import 'package:mojo/mojo/url_response.mojom.dart';
import 'package:sky_services/media/media.mojom.dart'; import 'package:sky_services/media/media.mojom.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
...@@ -21,20 +23,31 @@ class Key { ...@@ -21,20 +23,31 @@ class Key {
final Color color; final Color color;
final String soundUrl; final String soundUrl;
MediaPlayerProxy player; final MediaPlayerProxy player = new MediaPlayerProxy.unbound();
bool get isPlayerOpen => player.impl.isOpen;
void down() { void down() {
if (player == null) if (!isPlayerOpen) return;
return;
player.ptr.seekTo(0); player.ptr.seekTo(0);
player.ptr.start(); player.ptr.start();
} }
void up() { void up() {
if (player == null) if (!isPlayerOpen) return;
return;
player.ptr.pause(); player.ptr.pause();
} }
Future load(MediaServiceProxy mediaService) async {
try {
mediaService.ptr.createPlayer(player);
UrlResponse response = await fetchUrl(soundUrl);
await player.ptr.prepare(response.body);
} catch (e) {
print("Error: failed to load sound file $soundUrl");
player.close();
}
}
} }
class PianoApp extends StatelessComponent { class PianoApp extends StatelessComponent {
...@@ -47,47 +60,70 @@ class PianoApp extends StatelessComponent { ...@@ -47,47 +60,70 @@ class PianoApp extends StatelessComponent {
new Key(Colors.purple[500], iLoveYou), new Key(Colors.purple[500], iLoveYou),
]; ];
PianoApp() { Future connect() {
loadSounds(); return _loadSounds();
} }
loadSounds() async { Future _loadSounds() async {
MediaServiceProxy mediaService = new MediaServiceProxy.unbound(); MediaServiceProxy mediaService = new MediaServiceProxy.unbound();
shell.requestService(null, mediaService); try {
shell.requestService(null, mediaService);
for (Key key in keys) { List<Future> pending = [];
MediaPlayerProxy player = new MediaPlayerProxy.unbound(); for (Key key in keys) {
mediaService.ptr.createPlayer(player); pending.add(key.load(mediaService));
}
UrlResponse response = await fetchUrl(key.soundUrl); await Future.wait(pending);
await player.ptr.prepare(response.body); } finally {
key.player = player; mediaService.close();
} }
mediaService.close();
// Are we leaking all the player connections?
} }
Widget build(BuildContext context) { Widget build(BuildContext context) {
List<Widget> children = []; List<Widget> children = [];
for (Key key in keys) { for (Key key in keys) {
children.add( children.add(new Flexible(
new Flexible(
child: new Listener( child: new Listener(
child: new Container( child: new Container(
decoration: new BoxDecoration(backgroundColor: key.color) decoration: new BoxDecoration(backgroundColor: key.color)),
), onPointerCancel: (_) => key.up(),
onPointerCancel: (_) => key.up(), onPointerDown: (_) => key.down(),
onPointerDown: (_) => key.down(), onPointerUp: (_) => key.up())));
onPointerUp: (_) => key.up()
)
)
);
} }
return new Column(children); return new Column(children);
} }
} }
void main() { Widget statusBox(Widget child) {
runApp(new PianoApp()); const mediumGray = const Color(0xff555555);
const darkGray = const Color(0xff222222);
return new Center(
child: new Container(
decoration: const BoxDecoration(boxShadow: const [
const BoxShadow(
color: mediumGray, offset: const Offset(6.0, 6.0), blur: 5.0)
], backgroundColor: darkGray),
height: 90.0,
padding: const EdgeDims.all(8.0),
margin: const EdgeDims.symmetric(horizontal: 50.0),
child: new Center(child: child)));
}
Widget splashScreen() {
return statusBox(
new Text('Loading sound files!', style: new TextStyle(fontSize: 18.0)));
}
main() async {
runApp(splashScreen());
PianoApp app = new PianoApp();
// use "await" to make sure the sound files are loaded before we show the ui.
await app.connect();
runApp(app);
// runApp() returns immediately so you can't put application cleanup code
// here. Android apps can be killed at any time, so there's also no way to
// catch a close event to do cleanup. Therefore, although we appear to be
// leaking the "player" handles, this is working as intended and the operating
// system will clean up when the activity is killed.
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment