您最多选择25个主题 主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
 
 
 
 
 
 

383 行
14 KiB

using System;
using System.Collections.Generic;
using Unity.UIWidgets.foundation;
using Unity.UIWidgets.ui;
using UnityEngine;
using Canvas = Unity.UIWidgets.ui.Canvas;
using Rect = Unity.UIWidgets.ui.Rect;
namespace Unity.UIWidgets.painting {
public enum ImageRepeat {
repeat,
repeatX,
repeatY,
noRepeat,
}
public class DecorationImage : IEquatable<DecorationImage> {
public DecorationImage(
ImageProvider image = null,
ImageErrorListener onError = null,
ColorFilter colorFilter = null,
BoxFit? fit = null,
AlignmentGeometry alignment = null,
Rect centerSlice = null,
ImageRepeat repeat = ImageRepeat.noRepeat,
bool matchTextDirection = false
) {
D.assert(image != null);
this.image = image;
this.onError = onError;
this.colorFilter = colorFilter;
this.fit = fit;
this.alignment = alignment ?? Alignment.center;
this.centerSlice = centerSlice;
this.repeat = repeat;
this.matchTextDirection = matchTextDirection;
}
public readonly ImageProvider image;
public readonly ImageErrorListener onError;
public readonly ColorFilter colorFilter;
public readonly BoxFit? fit;
public readonly AlignmentGeometry alignment;
public readonly Rect centerSlice;
public readonly ImageRepeat repeat;
public readonly bool matchTextDirection;
public DecorationImagePainter createPainter(VoidCallback onChanged) {
D.assert(onChanged != null);
return new DecorationImagePainter(this, onChanged);
}
public bool Equals(DecorationImage other) {
if (ReferenceEquals(null, other)) {
return false;
}
if (ReferenceEquals(this, other)) {
return true;
}
return Equals(image, other.image) && Equals(colorFilter, other.colorFilter) &&
fit == other.fit && Equals(alignment, other.alignment) &&
Equals(centerSlice, other.centerSlice) && repeat == other.repeat;
}
public override bool Equals(object obj) {
if (ReferenceEquals(null, obj)) {
return false;
}
if (ReferenceEquals(this, obj)) {
return true;
}
if (obj.GetType() != GetType()) {
return false;
}
return Equals((DecorationImage) obj);
}
public override int GetHashCode() {
unchecked {
var hashCode = (image != null ? image.GetHashCode() : 0);
hashCode = (hashCode * 397) ^ (colorFilter != null ? colorFilter.GetHashCode() : 0);
hashCode = (hashCode * 397) ^ fit.GetHashCode();
hashCode = (hashCode * 397) ^ (alignment != null ? alignment.GetHashCode() : 0);
hashCode = (hashCode * 397) ^ (centerSlice != null ? centerSlice.GetHashCode() : 0);
hashCode = (hashCode * 397) ^ (int) repeat;
return hashCode;
}
}
public static bool operator ==(DecorationImage left, DecorationImage right) {
return Equals(left, right);
}
public static bool operator !=(DecorationImage left, DecorationImage right) {
return !Equals(left, right);
}
public override string ToString() {
var properties = new List<string>();
properties.Add($"{image}");
if (colorFilter != null) {
properties.Add($"{colorFilter}");
}
if (fit != null &&
!(fit == BoxFit.fill && centerSlice != null) &&
!(fit == BoxFit.scaleDown && centerSlice == null)) {
properties.Add($"{fit}");
}
properties.Add($"{alignment}");
if (centerSlice != null) {
properties.Add($"centerSlice: {centerSlice}");
}
if (repeat != ImageRepeat.noRepeat) {
properties.Add($"{repeat}");
}
return $"{GetType()}({string.Join(", ", properties)})";
}
}
public class DecorationImagePainter : IDisposable {
internal DecorationImagePainter(DecorationImage details, VoidCallback onChanged) {
D.assert(details != null);
_details = details;
_onChanged = onChanged;
}
readonly DecorationImage _details;
readonly VoidCallback _onChanged;
ImageStream _imageStream;
ImageInfo _image;
public void paint(Canvas canvas, Rect rect, Path clipPath, ImageConfiguration configuration) {
D.assert(canvas != null);
D.assert(rect != null);
D.assert(configuration != null);
bool flipHorizontally = false;
if (_details.matchTextDirection) {
D.assert(() => {
// We check this first so that the assert will fire immediately, not just
// when the image is ready.
if (configuration?.textDirection == null) {
throw new UIWidgetsError(new List<DiagnosticsNode>() {
new ErrorSummary(
"DecorationImage.matchTextDirection can only be used when a TextDirection is available."),
new ErrorDescription(
"When DecorationImagePainter.paint() was called, there was no text direction provided " +
"in the ImageConfiguration object to match."
),
new DiagnosticsProperty<DecorationImage>("The DecorationImage was", _details,
style: DiagnosticsTreeStyle.errorProperty),
new DiagnosticsProperty<ImageConfiguration>("The ImageConfiguration was", configuration,
style: DiagnosticsTreeStyle.errorProperty)
});
}
return true;
});
if (configuration.textDirection == TextDirection.rtl)
flipHorizontally = true;
}
ImageStream newImageStream = _details.image.resolve(configuration);
if (newImageStream.key != _imageStream?.key) {
ImageStreamListener listener = new ImageStreamListener(
_handleImage,
onError: _details.onError
);
_imageStream?.removeListener(listener);
_imageStream = newImageStream;
_imageStream.addListener(listener);
}
if (_image == null) {
return;
}
if (clipPath != null) {
canvas.save();
canvas.clipPath(clipPath);
}
painting_.paintImage(
canvas: canvas,
rect: rect,
image: _image.image,
scale: _image.scale,
colorFilter: _details.colorFilter,
fit: _details.fit,
alignment: _details.alignment.resolve(configuration.textDirection),
centerSlice: _details.centerSlice,
repeat: _details.repeat,
flipHorizontally: flipHorizontally,
filterQuality: FilterQuality.low
);
if (clipPath != null) {
canvas.restore();
}
}
void _handleImage(ImageInfo value, bool synchronousCall) {
if (_image == value) {
return;
}
_image = value;
D.assert(_onChanged != null);
if (!synchronousCall) {
_onChanged();
}
}
public void Dispose() {
_imageStream?.removeListener(new ImageStreamListener(
_handleImage,
onError: _details.onError
));
}
public override string ToString() {
return $"{GetType()}(stream: {_imageStream}, image: {_image}) for {_details}";
}
}
public static partial class painting_ {
public static void paintImage(
Canvas canvas = null,
Rect rect = null,
Image image = null,
float scale = 1.0f,
ColorFilter colorFilter = null,
BoxFit? fit = null,
Alignment alignment = null,
Rect centerSlice = null,
ImageRepeat repeat = ImageRepeat.noRepeat,
bool flipHorizontally = false,
bool invertColors = false,
FilterQuality filterQuality = FilterQuality.low
) {
D.assert(canvas != null);
D.assert(rect != null);
D.assert(image != null);
alignment = alignment ?? Alignment.center;
if (rect.isEmpty) {
return;
}
Size outputSize = rect.size;
Size inputSize = new Size(image.width, image.height);
Offset sliceBorder = null;
if (centerSlice != null) {
sliceBorder = new Offset(
centerSlice.left + inputSize.width - centerSlice.right,
centerSlice.top + inputSize.height - centerSlice.bottom
);
outputSize -= sliceBorder;
inputSize -= sliceBorder;
}
fit = fit ?? (centerSlice == null ? BoxFit.scaleDown : BoxFit.fill);
D.assert(centerSlice == null || (fit != BoxFit.none && fit != BoxFit.cover),
() => $"centerSlice was used with a BoxFit {fit} that is not supported.");
FittedSizes fittedSizes = FittedSizes.applyBoxFit(fit.Value, inputSize / scale, outputSize);
Size sourceSize = fittedSizes.source * scale;
Size destinationSize = fittedSizes.destination;
if (centerSlice != null) {
outputSize += sliceBorder;
destinationSize += sliceBorder;
D.assert(sourceSize == inputSize,
() =>
$"centerSlice was used with a BoxFit {fit} that does not guarantee that the image is fully visible.");
}
if (repeat != ImageRepeat.noRepeat && destinationSize == outputSize) {
repeat = ImageRepeat.noRepeat;
}
Paint paint = new Paint();
if (colorFilter != null) {
paint.colorFilter = colorFilter;
}
if (sourceSize != destinationSize) {
paint.filterQuality = filterQuality;
}
paint.invertColors = invertColors;
float halfWidthDelta = (outputSize.width - destinationSize.width) / 2.0f;
float halfHeightDelta = (outputSize.height - destinationSize.height) / 2.0f;
float dx = halfWidthDelta + (flipHorizontally ? -alignment.x : alignment.x) * halfWidthDelta;
float dy = halfHeightDelta + alignment.y * halfHeightDelta;
Offset destinationPosition = rect.topLeft.translate(dx, dy);
Rect destinationRect = destinationPosition & destinationSize;
bool needSave = repeat != ImageRepeat.noRepeat || flipHorizontally;
if (needSave) {
canvas.save();
}
if (flipHorizontally) {
float dxInside = -(rect.left + rect.width / 2.0f);
canvas.translate(-dxInside, 0.0f);
canvas.scale(-1.0f, 1.0f);
canvas.translate(dxInside, 0.0f);
}
if (repeat != ImageRepeat.noRepeat) {
canvas.clipRect(rect);
}
if (centerSlice == null) {
Rect sourceRect = alignment.inscribe(
sourceSize, Offset.zero & inputSize
);
if (repeat == ImageRepeat.noRepeat) {
canvas.drawImageRect(image, sourceRect, destinationRect, paint);
}
else {
foreach (Rect tileRect in _generateImageTileRects(rect, destinationRect, repeat)) {
canvas.drawImageRect(image, sourceRect, tileRect, paint);
}
}
}
else {
if (repeat == ImageRepeat.noRepeat) {
canvas.drawImageNine(image, centerSlice, destinationRect, paint);
}
else {
foreach (Rect tileRect in _generateImageTileRects(rect, destinationRect, repeat)) {
canvas.drawImageNine(image, centerSlice, tileRect, paint);
}
}
}
if (needSave) {
canvas.restore();
}
}
static IEnumerable<Rect> _generateImageTileRects(Rect outputRect, Rect fundamentalRect,
ImageRepeat repeat) {
int startX = 0;
int startY = 0;
int stopX = 0;
int stopY = 0;
float strideX = fundamentalRect.width;
float strideY = fundamentalRect.height;
if (repeat == ImageRepeat.repeat || repeat == ImageRepeat.repeatX) {
startX = ((outputRect.left - fundamentalRect.left) / strideX).floor();
stopX = ((outputRect.right - fundamentalRect.right) / strideX).ceil();
}
if (repeat == ImageRepeat.repeat || repeat == ImageRepeat.repeatY) {
startY = ((outputRect.top - fundamentalRect.top) / strideY).floor();
stopY = ((outputRect.bottom - fundamentalRect.bottom) / strideY).ceil();
}
for (int i = startX; i <= stopX; ++i) {
for (int j = startY; j <= stopY; ++j) {
yield return fundamentalRect.shift(new Offset(i * strideX, j * strideY));
}
}
}
}
}