- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我从未在编程中使用过 SkiaSharp 或处理过 SVG 和旋转点。我对矩阵的工作原理感到困惑;研究时看起来很复杂。我知道我可以将我的 pivotX/Y 点移动到中心,这样我就可以说我想将图像旋转 180 度,所有像素都是 1:1。
注意:我没有创建任何这些 SVG,它们是由公司内部的某个人制作的。
我正在使用 Xamarin 社区工具包来获取自定义弹出窗口。我绘制的 SVG 总是颠倒绘制的。所以我想将它旋转 180 度,但我不确定如何将 pivotX/Y 设置在 StarPath
或 Canvas center
的中心。
这个问题的全部要点让我可以学习如何在 SkiaSharp 中操作 Matrix 并能够通过指定 degrees
width/2
来创建旋转我的路径的方法高度/2
Xamarin 代码
<?xml version="1.0" encoding="utf-8" ?>
<xct:Popup xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
x:Class="uCue_Game.View.LostLifePopUp"
xmlns:xct="http://xamarin.com/schemas/2020/toolkit"
xmlns:sksh="clr-namespace:SkiaSharp.Views.Forms;assembly=SkiaSharp.Views.Forms"
Size="400, 400"
Color="Transparent">
<StackLayout BackgroundColor="Black"
Opacity="1">
<sksh:SKCanvasView x:Name="canvasView"
PaintSurface="canvasView_PaintSurface"
HorizontalOptions="FillAndExpand"
VerticalOptions="FillAndExpand"
HeightRequest="300"
Opacity="1"/>
<Label Text="Wrong Answer"
HorizontalOptions="FillAndExpand"
VerticalOptions="FillAndExpand"
FontSize="15"
TextColor="White"
HorizontalTextAlignment="Center"
VerticalTextAlignment="Center"
Opacity="1"/>
</StackLayout>
</xct:Popup>
C#代码
public partial class LostLifePopUp : Popup
{
SKPath StarPath = SKPath.ParseSvgPathData("M492" +
" 348 c16 -18 47 -67 69 -110 38 -76 40 -78 77 -78 55 -1 156 -24 186 -43 42" +
" -28 34 -61 -27 -120 -29 -29 -70 -63 -90 -77 -24 -16 -35 -30 -31 -40 18 -48" +
" 44 -170 44 -211 0 -44 -3 -49 -25 -55 -31 -8 -144 27 -209 64 l-47 27 -97 -47" +
" c-102 -50 -157 -65 -181 -49 -20 13 -16 97 9 195 l20 79 -58 61 c-69 73 -102" +
" 118 -102 141 0 22 66 50 177 75 l90 21 22 53 c12 29 35 74 51 100 38 59 76 63" +
" 122 14z");
SKPath StarFill = SKPath.ParseSvgPathData("M487" +
" 363 c21 -24 58 -71 83 -105 32 -46 51 -64 74 -69 65 -14 201 -56 213 -65 28" +
" -23 2 -34 -85 -35 -109 -1 -229 -31 -336 -85 -136 -69 -223 -82 -298 -43 -32" +
" 17 -98 73 -98 84 0 4 10 18 23 32 12 14 32 37 44 51 12 15 35 29 50 33 15 4" +
" 49 15 75 26 38 16 51 29 76 76 50 94 89 142 116 142 15 0 38 -16 63 -42z");
SKPath StarShine = SKPath.ParseSvgPathData(
"M0 1215 m0 -1215 m1260 0 m1260 0 m0 1215 m0 1215 m-1260 0 m-1260 0 m0" +
" -1215z m1395 838 c35 -35 95 -164 95 -204 0 -55 -64 -92" +
" -117 -69 -56 25 -68 54 -68 166 0 105 10 134 45 134 9 0 30 -12 45 -27z");
SKPaint StrokeRed = new SKPaint
{
Style = SKPaintStyle.StrokeAndFill,
Color = SKColors.Red,
StrokeWidth = 5,
StrokeCap = SKStrokeCap.Round,
StrokeJoin = SKStrokeJoin.Round
};
SKPaint FillYellow = new SKPaint
{
Style = SKPaintStyle.Fill,
Color = SKColors.Yellow,
StrokeWidth = 5,
StrokeCap = SKStrokeCap.Round,
StrokeJoin = SKStrokeJoin.Round
};
SKPaint GoldShine = new SKPaint
{
Style = SKPaintStyle.Stroke,
Color = SKColors.Green,
StrokeWidth = 40,
StrokeCap = SKStrokeCap.Round,
StrokeJoin = SKStrokeJoin.Round
};
public LostLifePopUp()
{
InitializeComponent();
}
private void canvasView_PaintSurface(object sender, SKPaintSurfaceEventArgs e)
{
SKRect bounds;
StarPath.GetBounds(out bounds);
var surface = e.Surface;
var surfaceWidth = e.RawInfo.Width;
var surfaceHeight = e.RawInfo.Height;
var canvas = surface.Canvas;
canvas.Flush();
canvas.Clear();
var rot = SKMatrix.CreateRotationDegrees(37, surfaceWidth / 2 - 50, surfaceHeight / 2);
canvas.SetMatrix(rot);
canvas.Translate(surfaceWidth / 2, surfaceHeight / 2);
canvas.Scale(0.9f * Math.Min(surfaceWidth / bounds.Width,
surfaceHeight / bounds.Height));
canvas.Translate(-bounds.MidX, -bounds.MidY);
canvas.DrawPath(StarShine, GoldShine);
canvas.DrawPath(StarPath, StrokeRed);
canvas.DrawPath(StarFill, FillYellow);
}
}
当前更新的代码 - 我的时间不多了,所以我现在不能再玩这个 SkiaSharp
了。我将开发这个游戏的第二个版本,这是一个更复杂的版本,所以我会在开始时尽快回到这个。我真的不满意 SKMatrix.PostConcat
使方法 过时
并且无法从任何地方访问。当有人失去等级时,我的明星有 2 个状态,我不得不采取非常错误的方法来实现它,但它是稍后修复的东西。此外,如果 Canvas 大小发生变化,整个绘图都会变得糟糕,其他地方都一样。
private void canvasView_PaintSurface(object sender, SKPaintSurfaceEventArgs e)
{
SKRect bounds;
StarPath.GetBounds(out bounds);
var surface = e.Surface;
var surfaceWidth = e.RawInfo.Width;
var surfaceHeight = e.RawInfo.Height;
var canvas = surface.Canvas;
var result = SKMatrix.CreateIdentity();
var translate = SKMatrix.CreateTranslation(-surfaceWidth / 2, -surfaceHeight / 2);
var rotate = SKMatrix.CreateRotationDegrees(180);
var translate2 = SKMatrix.CreateTranslation(surfaceWidth * 2, surfaceHeight);
SKMatrix.PostConcat(ref result, translate);
SKMatrix.PostConcat(ref result, rotate);
SKMatrix.PostConcat(ref result, translate2);
StarPath.Transform(result);
StarFill.Transform(result);
canvas.Scale(0.8f * Math.Min(surfaceWidth / bounds.Width, surfaceHeight / bounds.Height));
canvas.DrawPath(StarPath, WhiteOutline);
canvas.DrawPath(StarPath, Yellow);
canvas.DrawPath(StarFill, Shine);
if (DrawMain == true)
{
canvas.Scale(0.7f * Math.Min(surfaceWidth / bounds.Width, surfaceHeight / bounds.Height));
translate2 = SKMatrix.CreateTranslation(surfaceWidth * 2.5f, surfaceHeight * 2);
SKMatrix.PostConcat(ref result, translate2);
StarShine.Transform(result);
canvas.DrawPath(StarShine, White);
}
if (DrawSecond == true)
{
canvas.DrawPath(StarPath, WhiteOutline);
}
}
这是目前的样子。我看不到闪耀,填充实际上也没有填满整个星星……右图是它应该看起来的样子。
最佳答案
============更新答案======
在我看来,您可以更改 StarPath 以在中心绘制星星。
我在 300 X 300 canvasview 上制作这个
SKPath StarPath = SKPath.ParseSvgPathData("m 492" +
" 805.6 c 16 -18 47 -67 69 -110 c 38 -76 40 -78 77 -78 c 55 -1 156 -24 186 -43" +
" c 42 -28 34 -61 -27 -120 c -29 -29 -70 -63 -90 -77 c -24 -16 -35 -30 -31 -40" +
" c 18 -48 44 -170 44 -211 c 0 -44 -3 -49 -25 -55 c -31 -8 -144 27 -209 64" +
" l -47 27 l -97 -47 c -102 -50 -157 -65 -181 -49 c -20 13 -16 97 9 195 l 20 79" +
" l -58 61 c -69 73 -102 118 -102 141 c 0 22 66 50 177 75 l 90 21 l 22 53 c 12 29" +
" 35 74 51 100 c 38 59 76 63 122 14 z");
它在中心绘制星星。我使用具有@3x 比例因子的高分辨率的iphone13 模拟器。
============原始答案
感谢@ThisQRequiresASpecialist 的评论。我发现变换的顺序非常重要。如果旋转 Canvas ,坐标系也会旋转。也就是说如果你旋转90度,x轴变成y轴,y轴变成x轴,给后面的平移带来麻烦。所以先平移再旋转可能会更好。
在您的情况下,要旋转 180 度并平移到中心,只需尝试以下代码:
void canvasView_PaintSurface(System.Object sender, SkiaSharp.Views.Forms.SKPaintSurfaceEventArgs e)
{
SKImageInfo info = e.Info;
SKRect bounds;
var surface = e.Surface;
var surfaceWidth = info.Width;
var surfaceHeight = info.Height;
StarPath.GetTightBounds(out bounds);
canvas.Translate(surfaceWidth/2 - bounds.MidX, (surfaceHeight / 2 - bounds.MidY)); // Translate to the center
canvas.RotateDegrees(180, bounds.MidX, bounds.MidY); // rotate 180
...
=======在这里更新=======
@ThisQRequiresASpecialist 的评论也展示了一种使用 Matrix 的新方法。我还制作了一个演示,它运行良好。
var result = SKMatrix.CreateIdentity();
var rotate = SKMatrix.CreateRotationDegrees(180, bounds.MidX, bounds.MidY);
var translate = SKMatrix.CreateTranslation(surfaceWidth / 2 - bounds.MidX, -(surfaceHeight / 2 - bounds.MidY));
SKMatrix.PostConcat(ref result, translate);
SKMatrix.PostConcat(ref result, rotate);
StarPath.Transform(result);
canvas.DrawPath(StarPath, StrokeRed);
更多信息可以引用The Rotate Transform
希望我的回答能帮到你。
关于c# - SkiaSharp 在 Canvas 中心设置新的旋转点并围绕该点旋转,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/74347823/
我遇到了一个奇怪的问题。我的应用程序的 Release 版本似乎运行良好,但最近当我切换到 Debug 版本时,我在启动时立即遇到访问冲突。当释放分配的内存块时,就会发生访问冲突。所有这些都发生在静态
我在 C# 中偶然发现了这种奇怪的语法形式,并试图弄清楚它的含义以及如何使用它。网络上似乎没有关于此的任何文档。 object data = new { var1 = someValue, var2
我正在尝试使用浏览器的内置类型 CSSStyleDeclaration 以编程方式传递和修改样式(由于 .cssText 属性,这很方便)。 但是,new CSSStyleDeclaration()
我有现成的代码: internal bool firstAsSymbol(out Symbol s) { return (s = first as Symbol) !=
在新的 Eclipse 版本 2022-03 中,一些(但不是全部)java 项目在 Project Explorer View 中的外观发生了变化。尽管 Package Presentation 设
我正在尝试使用 FormData 通过获取 API 在 POST 请求中发送用户输入的数据。问题是,当我用我创建的表单创建一个新的 FormData 对象时,它一直在创建一个空对象——没有条目/键/值
我有一个用一些 intel-intrinsincs 编写的 C 代码。在我先用 avx 然后用 ssse3 标志编译后,我得到了两个完全不同的汇编代码。例如: AVX: vpunpckhbw %xm
最近,discord 为您自己的应用程序添加了对斜杠命令的支持。我通读了它的文档,并尝试搜索一些视频(但是该功能刚刚出现),但我不明白我实际上需要做什么才能使其正常工作。我正在使用 WebStorm(
我想使用 JRI 从 Java 调用 R。 我在 eclipse 下在主类中运行它: Rengine c = new Rengine(new String[] { "--vanilla" },
我正在使用新的 Place Autocomplete那是来自新的静态Google Places SDK 客户端库 (here)。所以它真的很容易使用,我刚得到this tutorial它按预期工作。
我刚刚更新到 flutter 版本 1.25.0-5.0.pre.92,我的代码中出现了很多与空安全相关的错误,这些错误以前运行良好。我没有以任何方式选择空安全,我所做的只是运行 flutter 升级
我已经使用 React Native 有一段时间了,但我想我会在网络上试用 React。所以我遵循了这个指南:https://reactjs.org/docs/create-a-new-react-a
周六早上在这里。尝试学习新的 Scala 编译器 dotty。 安装在我的 Mac 上使用 brew install lampepfl/brew/dotty 安装成功。我有版本 dotr -versi
我使用了谷歌地方的新依赖。单击自动完成 View 时应用程序崩溃。错误如下。, java.lang.NullPointerException: Place Fields must be set.
我关注了这个博客-> https://medium.com/@teyou21/training-your-object-detection-model-on-tensorflow-part-2-e9e
在哪里可以找到用于在此架构上进行组装的新寄存器的名称? 我指的是 X86 中的寄存器,如 EAX、ESP、EBX 等。但我希望它们是 64 位的。 我认为它们与我反汇编 C 代码时不同,我得到的是 r
新的服务总线库 Azure.Messaging.ServiceBus 使用 ServiceBusReceivedMessage 来接收消息 https://learn.microsoft.com/en
需要使用实时流媒体 channel 的实时编码类型在新的 Azure 门户中配置广告插入和石板图像。请帮忙解决这个问题,因为我找不到该功能。 最佳答案 此处描述了 Azure 媒体服务的广告插入选项
我正在使用新的 GitHub 操作,下面的工作流程的想法是在打开或同步 pr 时运行,它应该首先检查并安装依赖项,然后运行一些 yarn 脚本 name: PR to Master on: pul
我听说 DMD 2.058 中将有一个用于匿名函数的新语法,但我找不到任何相关信息。新语法是什么?旧语法是否会被弃用? 最佳答案 我相信它就像 C#'s . 以下内容是等效的: delegate(i,
我是一名优秀的程序员,十分优秀!