Создание масок изображения в флаттере
Flutter предоставляет несколько способов для масок, основанных на путях, т.е. путях обрезки. Я пытаюсь найти способ, где можно взять изображение со слоем прозрачности, как в примере ниже, и использовать это изображение для маскировки другого изображения / вида или в качестве общей маски.
Моим первым инстинктом было посмотреть на класс CustomPaint, но я не могу разобраться с этой первоначальной идеей.
3 ответа
Flutter имеет класс BoxDecoration, который принимает перечисление BlendMode. Используя их, вы можете добиться различных эффектов маски с помощью изображений, для моего конкретного случая выше dstIn
было решение.
Вот пример, который преобразует любой белый пиксель изображения маски в прозрачный на основном изображении.
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:image/image.dart' as img;
Image? _image;
Future<void> _loadImage(String maskPath, String imgPath) async {
ByteData data = await rootBundle.load(maskPath);
List<int> bytes = data.buffer.asUint8List();
img.Image mask = img.decodeImage(Uint8List.fromList(bytes))!;
data = await rootBundle.load(imgPath);
bytes = data.buffer.asUint8List();
img.Image image = img.decodeImage(Uint8List.fromList(bytes))!;
for (int y = 0; y < mask.height; y++) {
for (int x = 0; x < mask.width; x++) {
if (mask.getPixel(x, y) == img.getColor(255, 255, 255, 255)) {
image.setPixel(x, y, img.getColor(0, 0, 0, 0));
}
}
}
_image = Image.memory(Uint8List.fromList(img.encodePng(image!)));
}
dependencies:
image: ^3.0.1
Я отправил ответ на мою же проблему здесь, используя пользовательский класс художника с маской squircle и изображением.
@override
void paint(Canvas canvas, Size size) {
if (image != null && mask != null) {
var rect = Rect.fromLTRB(0, 0, 200, 200);
Size outputSize = rect.size;
Paint paint = new Paint();
//Mask
Size maskInputSize = Size(mask.width.toDouble(), mask.height.toDouble());
final FittedSizes maskFittedSizes =
applyBoxFit(BoxFit.cover, maskInputSize, outputSize);
final Size maskSourceSize = maskFittedSizes.source;
final Rect maskSourceRect = Alignment.center
.inscribe(maskSourceSize, Offset.zero & maskInputSize);
canvas.saveLayer(rect, paint);
canvas.drawImageRect(mask, maskSourceRect, rect, paint);
//Image
Size inputSize = Size(image.width.toDouble(), image.height.toDouble());
final FittedSizes fittedSizes =
applyBoxFit(BoxFit.cover, inputSize, outputSize);
final Size sourceSize = fittedSizes.source;
final Rect sourceRect =
Alignment.center.inscribe(sourceSize, Offset.zero & inputSize);
canvas.drawImageRect(
image, sourceRect, rect, paint..blendMode = BlendMode.srcIn);
canvas.restore();
}
}
Результат: