- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我必须使用 Direct2D 绘制很多形状。我使用工厂来创建我用来绘制的渲染目标。在这些形状之上我需要在不改变之前的情况下添加其他形状(徒手画),但是如果我使用相同的渲染目标我必须刷新整个 Canvas (即重绘所有形状),这是不可行的,因为它太慢了。
我需要一种解决方案,允许我在静态 形状上绘制,而无需连续清除绘制所有 Canvas 。我想通过使用同一个工厂创建一个新的渲染目标,但这个解决方案对我不起作用(即新形状不显示在屏幕上)。
是否有可以解决此问题的解决方案?例如在位图上绘制静态形状?
最佳答案
好的.. 在这里。我现在没有安装 MS VS,所以我用 Delphi 写了一个简单的例子。函数名称都是一样的,所以我希望你能理解这个想法..
这里是我使用的变量:
FFactory: ID2D1Factory; //ID2D1Factory* FFactory;
FHWNDRT: ID2D1HwndRenderTarget; //ID2D1HwndRenderTarget* FHWNDRT;
FBitmapRT: ID2D1BitmapRenderTarget; //Etc..
FBrush: ID2D1SolidColorBrush;
例子如下:
function TForm1.InitializeD2D: HRESULT;
var
TargetRect: TRect;
BrushProps: D2D1_BRUSH_PROPERTIES;
begin
{ Create factory }
Result := D2D1CreateFactory( D2D1_FACTORY_TYPE_SINGLE_THREADED, ID2D1Factory, nil, FFactory );
If Failed(Result) then Exit;
{ Get form's client rect }
Winapi.Windows.GetClientRect( Self.Handle, targetRect );
{ Create hwnd render target }
Result := FFactory.CreateHwndRenderTarget(
D2D1RenderTargetProperties( D2D1_RENDER_TARGET_TYPE_HARDWARE, D2D1PixelFormat( DXGI_FORMAT_R8G8B8A8_UNORM, D2D1_ALPHA_MODE_PREMULTIPLIED ), 96, 96, D2D1_RENDER_TARGET_USAGE_NONE, D2D1_FEATURE_LEVEL_10 ),
D2D1HwndRenderTargetProperties( Self.Handle, D2D1SizeU( TargetRect.Width, TargetRect.Height ) ),
FHWNDRT
);
If Failed(Result) then Exit;
{ Create bitmap render target }
Result := FHWNDRT.CreateCompatibleRenderTarget(
nil,
nil,
nil,
D2D1_COMPATIBLE_RENDER_TARGET_OPTIONS_NONE,
FBitmapRT
);
If Failed(Result) then Exit;
With BrushProps do Begin
opacity := 1;
transform := D2D1_MATRIX_3X2_F.Identity;
End;
{ Create brush so we can draw }
Result := FBitmapRT.CreateSolidColorBrush( D2D1ColorF( 0, 0.25, 0.75, 1 ), @BrushProps, FBrush );
end;
function TForm1.UpdateStaticShapes: HRESULT;
begin
//Draw the static shapes on the offscreen bitmap
FBitmapRT.BeginDraw;
//Clear the offscreen bitmap
FBitmapRT.Clear( D2D1ColorF(0, 0, 0, 1) );
//Draw a line
FBrush.SetColor( D2D1ColorF( 1, 0, 0, 1 ) );
FBitmapRT.DrawLine( D2D1PointF( Random(10)+10, Random(10)+10 ), D2D1PointF( 50, 50 ), FBrush );
//Draw a filled rect
FBrush.SetColor( D2D1ColorF( 0, 0.25, 0.75, 1 ) );
FBitmapRT.FillRectangle( D2D1RectF( Random(50), Random(50), Random(250)+50, Random(300) + 50 ), FBrush );
//Etc.. (draw all your shapes)
Result := FBitmapRT.EndDraw();
end;
function TForm1.Render: HRESULT;
var
pBitmap: ID2D1Bitmap;
begin
FHWNDRT.BeginDraw;
FHWNDRT.Clear( D2D1ColorF( 0, 0, 0, 1 ) );
{ Draw the offscreen bitmap }
FBitmapRT.GetBitmap( pBitmap );
If pBitmap <> nil then Begin
FHWNDRT.DrawBitmap( pBitmap );
pBitmap := nil; //Equivalent to _Release()
End;
{ Draw the additional free hand drawing here }
FBrush.SetColor( D2D1ColorF( 1, 1, 1, 1 ) );
FHWNDRT.DrawRectangle( D2D1RectF( 100, 100, 200, 200 ), FBrush );
Result := FHWNDRT.EndDraw();
end;
屏幕外位图在 UpdateStaticShapes() 方法中重新绘制。最后一帧在 Render() 方法中渲染。
编辑:我尝试进行缩放并了解您的问题。我想解决方案是在每次更改缩放系数(以及每次调整窗口大小时)时重新创建位图渲染目标。离屏位图的大小应该是:
OffscreenBitmap.Width/Height = HwndRT.Width/Height * ScaleFactor;
当您绘制形状时,您必须使用相对于屏幕外位图大小的坐标。例如:不是画线(50,50, 100,100),你应该画它(50*K, 50*K, 100*K, 100*K)其中 K = 比例因子;
这是我创建位图 RT 的方法:
//In C++ this should look like:
//HRESULT TForm::CreateBitmapRT(float ScaleFactor) {}
function TForm1.CreateBitmapRT(ScaleFactor: Single): HRESULT;
var
DesiredSize: D2D1_POINT_2F;
begin
FBitmapRT := nil; //FBitmapRT->_Release(); FBitmapRT = NULL;
{ Decide offscreen bitmap's size }
DesiredSize := D2D1PointF( FWindowRect.Width * ScaleFactor, FWindowRect.Height * ScaleFactor );
{ Create bitmap render target }
Result := FHWNDRT.CreateCompatibleRenderTarget(
@DesiredSize,
nil,
nil,
D2D1_COMPATIBLE_RENDER_TARGET_OPTIONS_NONE,
FBitmapRT
);
end;
这是更新后的 Render() 方法:
function TForm1.Render: HRESULT;
var
pBitmap: ID2D1Bitmap;
SrcRect, DestRect: D2D1_RECT_F;
begin
FHWNDRT.BeginDraw;
FHWNDRT.Clear( D2D1ColorF( 0, 0, 0, 1 ) );
If FBitmapRT <> nil then Begin
{ Draw the offscreen bitmap }
FBitmapRT.GetBitmap( pBitmap );
If pBitmap <> nil then Begin
SrcRect := D2D1RectF( FZoomOffset.x, FZoomOffset.y, FZoomOffset.x + FWindowRect.Width, FZoomOffset.y + FWindowRect.Height );
DestRect := D2D1RectF( 0, 0, FWindowRect.Width, FWindowRect.Height );
FHWNDRT.DrawBitmap( pBitmap, @DestRect, 1, D2D1_BITMAP_INTERPOLATION_MODE_LINEAR, @SrcRect );
pBitmap := nil; //pBitmap->_Release(); pBitmap = NULL;
End;
End;
{ Draw the additional free hand drawing here }
FBrush.SetColor( D2D1ColorF( 1, 1, 1, 1 ) );
FHWNDRT.DrawRectangle( D2D1RectF( 100, 100, 200, 200 ), FBrush );
Result := FHWNDRT.EndDraw();
end;
如果我更改缩放系数(放大或缩小),会发生以下情况:
function TForm1.ApplyZoom(fScaleFactor: Single): HRESULT;
begin
If fScaleFactor = FZoomFactor then Exit(S_OK);
If fScaleFactor = 0 then fScaleFactor := 0.1;
{ Recreate the offscreen bitmap }
Result := CreateBitmapRT( fScaleFactor );
If Failed(Result) then Exit;
{ Here you have to redraw the shapes once again }
Result := UpdateStaticShapes;
If Failed(Result) then Exit;
{ I save the new Zoom Factor in a class field }
FZoomFactor := fScaleFactor;
end;
关于c++ - 给定同一个 Direct2D 工厂的多个渲染目标,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28793152/
我应该执行以下操作: 可能通过服务/工厂,使用 $q(异步)查询 API 以获取大型名称数据集 有另一个服务(也是异步的),它应该只返回上述工厂的元素,如果它们与某个字符串(搜索字段)匹配。目的是缩小
我有一个通用的基类。我有一个实现基类的具体类。 我将如何创建工厂类/方法来交付不同类型的具体类? 举个例子: public class ReceiverBase where T : IInte
我正在查看以下链接中的 Ninject Factory 扩展: http://www.planetgeek.ch/2011/12/31/ninject-extensions-factory-intro
工厂、提供商和服务这三个术语之间有什么区别? 刚刚了解 NHibernate 及其存储库模式(POCO 类等)。 最佳答案 工厂:通过将一堆位组合在一起或基于某种上下文选择类型来组装类 Provide
使用CGLIB我可以做到 final var enhancer = new Enhancer(); enhancer.setUseCache(false); enhancer.setSuperclas
我试图在 Kotlin 中使用伴随对象工厂方法(相当于 Java 中的静态工厂方法)创建一个嵌套内部类。这是我的代码的简化版本。 class OuterClass { var myData:L
我正在为我的大学做一个项目,但遇到了问题。 基本上,该项目由一个客户端-服务器应用程序组成,我想创建一个用于通信的 Packet 类。数据包由 header 和主体组成。现在问题来了。我可以有一些不同
这个问题在这里已经有了答案: Why doesn't polymorphism work without pointers/references? (6 个答案) What is object sl
我正在制作一个套接字工厂。我希望每个外部应用程序都使用 Socket 类的接口(interface),它是几个类(ServerSocketTCP、ClientSocketTCP、ServerSocke
我是 angularjs 的新手,我正在尝试创建一个小型电影数据库。这是我第一次使用工厂,我想确保这是正确的方法,以及如何在另一个功能中使用这个工厂,如下所示? 我希望这个工厂只运行一次,这样我就可以
这个问题在这里已经有了答案: Java inner class and static nested class (28 个答案) 关闭 5 年前。 public class DataFactory
我看过很多关于 C++ 工厂的帖子,但到目前为止我还没有看到解决我的问题的解决方案。 (虽然我可能遗漏了一些东西。) 示例控制台应用程序: #include #include #include
这是一个简单的 C++ 项目,有 2 种设计模式:单例和工厂,sigleton 也是一个模板化类,一个接口(interface) (IHash) 和一个类 (Hash1)。一个简单的工厂类 (Hash
这个问题类似于Factory and generics ,并且可能有相同的答案,但它是不同的。我有一个通用基类,它将由完全独立的 JAR 中的类进行扩展。所述 JAR 应该能够在不更改任何其他代码的情
问题是我需要为传递的类创建一个新实例 有没有办法重写这个函数,让它可以接受任意数量的参数? function createInstance(ofClass, arg1, arg2, arg3, ...
我想用简单的 C++ 语法创建一个简单的工厂方法: void *createObject(const char *str,...) { if(!strcmp("X",str)) retu
经过大约 10 个月的程序化 PHP 学习后,我现在正尝试着手研究基本的 OOP 原则和设计模式。这是一个爱好,我没有那么多时间去追求它,所以请原谅这个问题的水平很低。 我的网站(目前 100% 程序
我有一个简单的问题。 我如何编写一个工厂来定义使用 make() 或 create() 的关系,具体取决于原始调用 make() 还是 create()? 这是我的用例: 我有一个简单的工厂 /**
我正在尝试在延迟加载模块中提供 APP_BASE_HREF 注入(inject) token ,然而,工厂方法根本没有被调用。 在这里https://github.com/MaurizioCascia
我有以下 ast: import { factory as f } from 'typescript' const typeDeclaration = f.createTypeAliasDeclara
我是一名优秀的程序员,十分优秀!