blob: 6910f56b82fcacc615cc160d526de76200ff3e36 [file] [log] [blame]
// Copyright 2013 The Flutter 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:io';
import 'dart:typed_data';
import 'dart:ui' as ui;
import 'package:litetest/litetest.dart';
import 'package:path/path.dart' as path;
import 'impeller_enabled.dart';
void main() {
test('Animation metadata', () async {
Uint8List data = await _getSkiaResource('alphabetAnim.gif').readAsBytes();
ui.Codec codec = await ui.instantiateImageCodec(data);
expect(codec, isNotNull);
expect(codec.frameCount, 13);
expect(codec.repetitionCount, 0);
codec.dispose();
data = await _getSkiaResource('test640x479.gif').readAsBytes();
codec = await ui.instantiateImageCodec(data);
expect(codec.frameCount, 4);
expect(codec.repetitionCount, -1);
});
test('Fails with invalid data', () async {
final Uint8List data = Uint8List.fromList(<int>[1, 2, 3]);
try {
await ui.instantiateImageCodec(data);
fail('exception not thrown');
} on Exception catch (e) {
expect(e.toString(), contains('Invalid image data'));
}
});
test('getNextFrame fails with invalid data', () async {
Uint8List data = await _getSkiaResource('flutter_logo.jpg').readAsBytes();
data = Uint8List.view(data.buffer, 0, 4000);
final ui.Codec codec = await ui.instantiateImageCodec(data);
try {
await codec.getNextFrame();
fail('exception not thrown');
} on Exception catch (e) {
if (impellerEnabled) {
expect(e.toString(), contains('Could not decompress image.'));
} else {
expect(e.toString(), contains('Codec failed'));
}
}
});
test('nextFrame', () async {
final Uint8List data = await _getSkiaResource('test640x479.gif').readAsBytes();
final ui.Codec codec = await ui.instantiateImageCodec(data);
final List<List<int>> decodedFrameInfos = <List<int>>[];
for (int i = 0; i < 5; i++) {
final ui.FrameInfo frameInfo = await codec.getNextFrame();
decodedFrameInfos.add(<int>[
frameInfo.duration.inMilliseconds,
frameInfo.image.width,
frameInfo.image.height,
]);
}
expect(decodedFrameInfos, equals(<List<int>>[
<int>[200, 640, 479],
<int>[200, 640, 479],
<int>[200, 640, 479],
<int>[200, 640, 479],
<int>[200, 640, 479],
]));
});
test('non animated image', () async {
final Uint8List data = await _getSkiaResource('baby_tux.png').readAsBytes();
final ui.Codec codec = await ui.instantiateImageCodec(data);
final List<List<int>> decodedFrameInfos = <List<int>>[];
for (int i = 0; i < 2; i++) {
final ui.FrameInfo frameInfo = await codec.getNextFrame();
decodedFrameInfos.add(<int>[
frameInfo.duration.inMilliseconds,
frameInfo.image.width,
frameInfo.image.height,
]);
}
expect(decodedFrameInfos, equals(<List<int>>[
<int>[0, 240, 246],
<int>[0, 240, 246],
]));
});
test('with size', () async {
final Uint8List data = await _getSkiaResource('baby_tux.png').readAsBytes();
final ui.ImmutableBuffer buffer = await ui.ImmutableBuffer.fromUint8List(data);
final ui.Codec codec = await ui.instantiateImageCodecWithSize(
buffer,
getTargetSize: (int intrinsicWidth, int intrinsicHeight) {
return ui.TargetImageSize(
width: intrinsicWidth ~/ 2,
height: intrinsicHeight ~/ 2,
);
},
);
final List<List<int>> decodedFrameInfos = <List<int>>[];
for (int i = 0; i < 2; i++) {
final ui.FrameInfo frameInfo = await codec.getNextFrame();
decodedFrameInfos.add(<int>[
frameInfo.duration.inMilliseconds,
frameInfo.image.width,
frameInfo.image.height,
]);
}
expect(decodedFrameInfos, equals(<List<int>>[
<int>[0, 120, 123],
<int>[0, 120, 123],
]));
});
test('disposed decoded image', () async {
final Uint8List data = await _getSkiaResource('flutter_logo.jpg').readAsBytes();
final ui.Codec codec = await ui.instantiateImageCodec(data);
final ui.FrameInfo frameInfo = await codec.getNextFrame();
expect(frameInfo.image, isNotNull);
frameInfo.image.dispose();
try {
await codec.getNextFrame();
fail('exception not thrown');
} on Exception catch (e) {
expect(e.toString(), contains('Decoded image has been disposed'));
}
});
test('Animated gif can reuse across multiple frames', () async {
// Regression test for b/271947267 and https://github.com/flutter/flutter/issues/122134
final Uint8List data = File(
path.join('flutter', 'lib', 'ui', 'fixtures', 'four_frame_with_reuse.gif'),
).readAsBytesSync();
final ui.Codec codec = await ui.instantiateImageCodec(data);
// Capture the final frame of animation. If we have not composited
// correctly, it will be clipped strangely.
late ui.FrameInfo frameInfo;
for (int i = 0; i < 4; i++) {
frameInfo = await codec.getNextFrame();
}
final ui.Image image = frameInfo.image;
final ByteData imageData = (await image.toByteData(format: ui.ImageByteFormat.png))!;
final String fileName = impellerEnabled ? 'impeller_four_frame_with_reuse_end.png' : 'four_frame_with_reuse_end.png';
final Uint8List goldenData = File(
path.join('flutter', 'lib', 'ui', 'fixtures', fileName),
).readAsBytesSync();
expect(imageData.buffer.asUint8List(), goldenData);
});
test('Animated webp can reuse across multiple frames', () async {
// Regression test for https://github.com/flutter/flutter/issues/61150#issuecomment-679055858
final Uint8List data = File(
path.join('flutter', 'lib', 'ui', 'fixtures', 'heart.webp'),
).readAsBytesSync();
final ui.Codec codec = await ui.instantiateImageCodec(data);
// Capture the final frame of animation. If we have not composited
// correctly, the hearts will be incorrectly repeated in the image.
late ui.FrameInfo frameInfo;
for (int i = 0; i < 69; i++) {
frameInfo = await codec.getNextFrame();
}
final ui.Image image = frameInfo.image;
final ByteData imageData = (await image.toByteData(format: ui.ImageByteFormat.png))!;
final String fileName = impellerEnabled ? 'impeller_heart_end.png' : 'heart_end.png';
final Uint8List goldenData = File(
path.join('flutter', 'lib', 'ui', 'fixtures', fileName),
).readAsBytesSync();
expect(imageData.buffer.asUint8List(), goldenData);
});
test('Animated apng can reuse pre-pre-frame', () async {
// https://github.com/flutter/engine/pull/42153
final Uint8List data = File(
path.join('flutter', 'lib', 'ui', 'fixtures', '2_dispose_op_restore_previous.apng'),
).readAsBytesSync();
final ui.Codec codec = await ui.instantiateImageCodec(data);
// Capture the 67,68,69 frames of animation and then compare the pixels.
late ui.FrameInfo frameInfo;
for (int i = 0; i < 70; i++) {
frameInfo = await codec.getNextFrame();
if (i >= 67) {
final ui.Image image = frameInfo.image;
final ByteData imageData = (await image.toByteData(format: ui.ImageByteFormat.png))!;
final String fileName = impellerEnabled ? 'impeller_2_dispose_op_restore_previous.apng.$i.png' : '2_dispose_op_restore_previous.apng.$i.png';
final Uint8List goldenData = File(
path.join('flutter', 'lib', 'ui', 'fixtures', fileName),
).readAsBytesSync();
expect(imageData.buffer.asUint8List(), goldenData);
}
}
});
test('Animated apng alpha type handling', () async {
final Uint8List data = File(
path.join('flutter', 'lib', 'ui', 'fixtures', 'alpha_animated.apng'),
).readAsBytesSync();
final ui.Codec codec = await ui.instantiateImageCodec(data);
// The test image contains two frames of solid red. The first has
// alpha=0.2, and the second has alpha=0.6.
ui.Image image = (await codec.getNextFrame()).image;
ByteData imageData = (await image.toByteData())!;
expect(imageData.getUint32(0), 0x33000033);
image = (await codec.getNextFrame()).image;
imageData = (await image.toByteData())!;
expect(imageData.getUint32(0), 0x99000099);
});
test('Animated apng background color restore', () async {
final Uint8List data = File(
path.join('flutter', 'lib', 'ui', 'fixtures', 'dispose_op_background.apng'),
).readAsBytesSync();
final ui.Codec codec = await ui.instantiateImageCodec(data);
// First frame is solid red
ui.Image image = (await codec.getNextFrame()).image;
ByteData imageData = (await image.toByteData())!;
expect(imageData.getUint32(0), 0xFF0000FF);
// Third frame is blue in the lower right corner.
await codec.getNextFrame();
image = (await codec.getNextFrame()).image;
imageData = (await image.toByteData())!;
expect(imageData.getUint32(imageData.lengthInBytes - 4), 0x0000FFFF);
// Fourth frame is transparent in the lower right corner
image = (await codec.getNextFrame()).image;
imageData = (await image.toByteData())!;
expect(imageData.getUint32(imageData.lengthInBytes - 4), 0x00000000);
});
}
/// Returns a File handle to a file in the skia/resources directory.
File _getSkiaResource(String fileName) {
// As Platform.script is not working for flutter_tester
// (https://github.com/flutter/flutter/issues/12847), this is currently
// assuming the curent working directory is engine/src.
// This is fragile and should be changed once the Platform.script issue is
// resolved.
final String assetPath = path.join(
'flutter', 'third_party', 'skia', 'resources', 'images', fileName,
);
return File(assetPath);
}