/////////////////////////////////////////////// void FillTransparentPixelsByNearPixels2(ref ImageUtility.Image2D image, Vector2I[,] opacityImageNearestCellTable) { for (int y = 0; y < image.Size.Y; y++) { for (int x = 0; x < image.Size.X; x++) { var takeFrom = opacityImageNearestCellTable[x, y]; var pixel = image.GetPixelByte(new Vector2I(takeFrom.X, takeFrom.Y)); image.SetPixel(new Vector2I(x, y), pixel); //var transparent = opacityImage.GetPixel( new Vector2I( x, y ) ).ToVector3F() == Vector3F.Zero; //ColorByte pixel; //if( transparent ) //{ // var takeFrom = opacityImageNearestCellTable[ x, y ]; // pixel = image.GetPixelByte( new Vector2I( takeFrom.X, takeFrom.Y ) ); //} //else // pixel = image.GetPixelByte( new Vector2I( x, y ) ); //image.SetPixel( new Vector2I( x, y ), pixel ); } } }
DistanceMapItem[] GetDistanceMap(ImageUtility.Image2D image) { if (distanceMap == null || distanceMapForSize != image.Size) { var map = new List <DistanceMapItem>(image.Size.X * 5); for (int y = -image.Size.Y; y <= image.Size.Y; y++) { for (int x = -image.Size.X; x <= image.Size.X; x++) { if (x == 0 && y == 0) { continue; } var item = new DistanceMapItem(); item.offset = new Vector2I(x, y); item.distance = MathEx.Sqrt(x * x + y * y); map.Add(item); } } distanceMap = map.ToArray(); distanceMapForSize = image.Size; CollectionUtility.MergeSort(distanceMap, delegate(DistanceMapItem item1, DistanceMapItem item2) { if (item1.distance < item2.distance) { return(-1); } else if (item1.distance > item2.distance) { return(1); } return(0); }, true); } return(distanceMap); }
public void Update() { if (Working) { try { if (RenderingSystem.LastFrameNumber >= demandedFrame) { //task is done var imageRender = new ImageUtility.Image2D(imageFormat, new Vector2I(imageSizeRender, imageSizeRender), imageData); //get clip rectangle RectangleI rectangle; if (generator.ClampImage) { rectangle = FindActualData(imageRender); //borders rectangle.Expand(imageEmptyBorder); //!!!!станет не квадратный ////clamp //if( rectangle.Left < 0 ) // rectangle.Left = 0; //if( rectangle.Right > imageSizeRender ) // rectangle.Right = imageSizeRender; //if( rectangle.Top < 0 ) // rectangle.Top = 0; //if( rectangle.Bottom > imageSizeRender ) // rectangle.Bottom = imageSizeRender; } else { rectangle = new RectangleI(0, 0, imageSizeRender, imageSizeRender); } var newSize = rectangle.Size; //make clamped image var image2 = new ImageUtility.Image2D(imageFormat, newSize); image2.Blit(Vector2I.Zero, imageRender, rectangle.LeftTop); //make bitmap from clamped image Bitmap bitmap2; unsafe { fixed(byte *pImage2 = image2.Data) { bitmap2 = new Bitmap(newSize.X, newSize.Y, newSize.X * PixelFormatUtility.GetNumElemBytes(imageFormat), System.Drawing.Imaging.PixelFormat.Format32bppArgb, (IntPtr)pImage2); } } //var bitmapRender = new Bitmap( imageSizeRender, imageSizeRender, imageSizeRender * PixelFormatUtility.GetNumElemBytes( imageFormat ), System.Drawing.Imaging.PixelFormat.Format32bppArgb, imageData ); //make result bitmap var bitmapResult = ResizeBitmap(bitmap2, imageSizeResult, imageSizeResult); //save result bitmap var destRealFileName = GetCacheFileNameByResourceVirtualFileName(currentTask.virtualFileName); try { if (!Directory.Exists(Path.GetDirectoryName(destRealFileName))) { Directory.CreateDirectory(Path.GetDirectoryName(destRealFileName)); } bitmapResult.Save(destRealFileName, ImageFormat.Png); } catch (Exception e) { Log.Warning($"PreviewImagesManager: Processor: Update: Unable to save bitmap to \'{destRealFileName}\'. " + e.Message); } finally { bitmap2.Dispose(); bitmapResult.Dispose(); } //task end ClearTask(); //update content browsers foreach (var browser in ContentBrowser.AllInstances) { browser.needUpdateImages = true; } } } catch (Exception e) { Log.Warning(e.Message); } } }
static RectangleI FindActualData(ImageUtility.Image2D image) { int minX = 0; for ( ; minX < image.Size.X; minX++) { for (int y = 0; y < image.Size.Y; y++) { if (image.GetPixel(new Vector2I(minX, y)).W >= 0.004) { goto endMinX; } } } endMinX :; int minY = 0; for ( ; minY < image.Size.Y; minY++) { for (int x = 0; x < image.Size.X; x++) { if (image.GetPixel(new Vector2I(x, minY)).W >= 0.004) { goto endMinY; } } } endMinY :; int maxX = image.Size.X - 1; for ( ; maxX >= 0; maxX--) { for (int y = 0; y < image.Size.Y; y++) { if (image.GetPixel(new Vector2I(maxX, y)).W >= 0.004) { goto endMaxX; } } } endMaxX :; int maxY = image.Size.Y - 1; for ( ; maxY >= 0; maxY--) { for (int x = 0; x < image.Size.X; x++) { if (image.GetPixel(new Vector2I(x, maxY)).W >= 0.004) { goto endMaxY; } } } endMaxY :; //fix empty image if (maxX < minX) { maxX = minX; } if (maxY < minY) { maxY = minY; } var rect = new RectangleI(minX, minY, maxX + 2, maxY + 2); //make square if (rect.Size.X < rect.Size.Y) { rect.Expand(new Vector2I((rect.Size.Y - rect.Size.X) / 2, 0)); } else if (rect.Size.Y < rect.Size.X) { rect.Expand(new Vector2I(0, (rect.Size.X - rect.Size.Y) / 2)); } return(rect); }
void RenderMaterial(Component_Camera camera, string destRealFileName) { var renderToFile = RenderToFile; var scene = renderToFile.ParentRoot as Component_Scene; var textureFileNames = new string[5]; //ImageUtility.Image2D opacityImage = null; Vector2I[,] opacityImageNearestCellTable = null; //write textures for (int nChannel = 0; nChannel < 5; nChannel++) { var channel = (MaterialChannel)nChannel; Component_Image texture = null; Component_Image textureRead = null; try { //!!!!все каналы //!!!!какие еще параметры? var prefix = Path.GetFileNameWithoutExtension(destRealFileName) + "_"; string fileName = ""; switch (nChannel) { case 0: fileName = prefix + "Opacity.png"; break; case 1: fileName = prefix + "BaseColor.png"; break; case 2: fileName = prefix + "Metallic.png"; break; case 3: fileName = prefix + "Roughness.png"; break; case 4: fileName = prefix + "Normal.png"; break; } var fullPath = Path.Combine(Path.GetDirectoryName(destRealFileName), fileName); //create var resolution = renderToFile.Resolution.Value; PixelFormat format = PixelFormat.A8R8G8B8; texture = ComponentUtility.CreateComponent <Component_Image>(null, true, false); texture.CreateType = Component_Image.TypeEnum._2D; texture.CreateSize = resolution; texture.CreateMipmaps = false; texture.CreateFormat = format; texture.CreateUsage = Component_Image.Usages.RenderTarget; texture.CreateFSAA = 0; texture.Enabled = true; var renderTexture = texture.Result.GetRenderTarget(0, 0); var viewport = renderTexture.AddViewport(true, true); viewport.AttachedScene = scene; textureRead = ComponentUtility.CreateComponent <Component_Image>(null, true, false); textureRead.CreateType = Component_Image.TypeEnum._2D; textureRead.CreateSize = resolution; textureRead.CreateMipmaps = false; textureRead.CreateFormat = format; textureRead.CreateUsage = Component_Image.Usages.ReadBack | Component_Image.Usages.BlitDestination; textureRead.CreateFSAA = 0; textureRead.Enabled = true; //!!!! textureRead.Result.PrepareNativeObject(); var restorePipeline = scene.RenderingPipeline; var pipeline = ComponentUtility.CreateComponent <Component_RenderingPipeline_Default>(null, true, true); switch (nChannel) { case 0: pipeline.DebugMode = Component_RenderingPipeline_Basic.DebugModeEnum.Normal; break; case 1: pipeline.DebugMode = Component_RenderingPipeline_Basic.DebugModeEnum.BaseColor; break; case 2: pipeline.DebugMode = Component_RenderingPipeline_Basic.DebugModeEnum.Metallic; break; case 3: pipeline.DebugMode = Component_RenderingPipeline_Basic.DebugModeEnum.Roughness; break; case 4: pipeline.DebugMode = Component_RenderingPipeline_Basic.DebugModeEnum.Normal; break; } try { scene.RenderingPipeline = pipeline; scene.GetDisplayDevelopmentDataInThisApplicationOverride += Scene_GetDisplayDevelopmentDataInThisApplicationOverride; var cameraSettings = new Viewport.CameraSettingsClass(viewport, camera); viewport.Update(true, cameraSettings); //clear temp data viewport.RenderingContext.MultiRenderTarget_DestroyAll(); viewport.RenderingContext.DynamicTexture_DestroyAll(); } finally { scene.RenderingPipeline = restorePipeline; scene.GetDisplayDevelopmentDataInThisApplicationOverride -= Scene_GetDisplayDevelopmentDataInThisApplicationOverride; } texture.Result.GetRealObject(true).BlitTo(viewport.RenderingContext.CurrentViewNumber, textureRead.Result.GetRealObject(true), 0, 0); //!!!!pitch //get data var totalBytes = PixelFormatUtility.GetNumElemBytes(format) * resolution.X * resolution.Y; var data = new byte[totalBytes]; unsafe { fixed(byte *pBytes = data) { var demandedFrame = textureRead.Result.GetRealObject(true).Read((IntPtr)pBytes, 0); while (RenderingSystem.CallBgfxFrame() < demandedFrame) { } } } var image = new ImageUtility.Image2D(format, resolution, data); if (channel == MaterialChannel.Opacity) { //convert pixels for (int y = 0; y < image.Size.Y; y++) { for (int x = 0; x < image.Size.X; x++) { var pixel = image.GetPixel(new Vector2I(x, y)); if (pixel.ToVector3F() != Vector3F.Zero) { pixel = Vector4F.One; } else { pixel = Vector4F.Zero; } image.SetPixel(new Vector2I(x, y), pixel); } } //opacityImageNearestCellTable if (renderToFile.FillTransparentPixelsByNearPixels) { var boolOpacityImage = new bool[image.Size.X, image.Size.Y]; for (int y = 0; y < image.Size.Y; y++) { for (int x = 0; x < image.Size.X; x++) { var c = image.GetPixelByte(new Vector2I(x, y)); boolOpacityImage[x, y] = c.Red == 0; } } var distanceMap = GetDistanceMap(image); opacityImageNearestCellTable = new Vector2I[image.Size.X, image.Size.Y]; for (int y = 0; y < image.Size.Y; y++) { for (int x = 0; x < image.Size.X; x++) { opacityImageNearestCellTable[x, y] = new Vector2I(x, y); } } var table = opacityImageNearestCellTable; //!!!!slowly Parallel.For(0, image.Size.Y, delegate(int y) //for( int y = 0; y < image.Size.Y; y++ ) { for (int x = 0; x < image.Size.X; x++) { var transparent = boolOpacityImage[x, y]; if (transparent) { for (int n = 0; n < distanceMap.Length; n++) //foreach( var indexItem in distanceMap ) { ref var indexItem = ref distanceMap[n]; var takeFrom = new Vector2I(x, y) + indexItem.offset; if (takeFrom.X >= 0 && takeFrom.X < image.Size.X && takeFrom.Y >= 0 && takeFrom.Y < image.Size.Y) { var transparent2 = boolOpacityImage[takeFrom.X, takeFrom.Y]; if (!transparent2) { table[x, y] = takeFrom; break; } } } } } }); } }
void RenderScreenshot(Component_Camera camera, string destRealFileName) { var renderToFile = RenderToFile; var scene = renderToFile.ParentRoot as Component_Scene; Component_Image texture = null; Component_Image textureRead = null; try { //create var resolution = renderToFile.Resolution.Value; //!!!!impl var hdr = false; //HDR.Value; PixelFormat format = hdr ? PixelFormat.Float16RGBA : PixelFormat.A8R8G8B8; //PixelFormat format = hdr ? PixelFormat.Float32RGBA : PixelFormat.A8R8G8B8; texture = ComponentUtility.CreateComponent <Component_Image>(null, true, false); texture.CreateType = Component_Image.TypeEnum._2D; texture.CreateSize = resolution; texture.CreateMipmaps = false; texture.CreateFormat = format; texture.CreateUsage = Component_Image.Usages.RenderTarget; texture.CreateFSAA = 0; texture.Enabled = true; var renderTexture = texture.Result.GetRenderTarget(0, 0); var viewport = renderTexture.AddViewport(true, true); viewport.AttachedScene = scene; textureRead = ComponentUtility.CreateComponent <Component_Image>(null, true, false); textureRead.CreateType = Component_Image.TypeEnum._2D; textureRead.CreateSize = resolution; textureRead.CreateMipmaps = false; textureRead.CreateFormat = format; textureRead.CreateUsage = Component_Image.Usages.ReadBack | Component_Image.Usages.BlitDestination; textureRead.CreateFSAA = 0; textureRead.Enabled = true; //!!!! textureRead.Result.PrepareNativeObject(); //render //var image2D = new ImageUtility.Image2D( PixelFormat.Float32RGB, new Vector2I( size * 4, size * 3 ) ); //var position = Transform.Value.Position; //for( int face = 0; face < 6; face++ ) //{ //Vector3 dir = Vector3.Zero; //Vector3 up = Vector3.Zero; ////flipped //switch( face ) //{ //case 0: dir = -Vector3.YAxis; up = Vector3.ZAxis; break; //case 1: dir = Vector3.YAxis; up = Vector3.ZAxis; break; //case 2: dir = Vector3.ZAxis; up = -Vector3.XAxis; break; //case 3: dir = -Vector3.ZAxis; up = Vector3.XAxis; break; //case 4: dir = Vector3.XAxis; up = Vector3.ZAxis; break; //case 5: dir = -Vector3.XAxis; up = Vector3.ZAxis; break; //} try { scene.GetDisplayDevelopmentDataInThisApplicationOverride += Scene_GetDisplayDevelopmentDataInThisApplicationOverride; var cameraSettings = new Viewport.CameraSettingsClass(viewport, camera); //var cameraSettings = new Viewport.CameraSettingsClass( viewport, 1, 90, NearClipPlane.Value, FarClipPlane.Value, position, dir, up, ProjectionType.Perspective, 1, 1, 1 ); viewport.Update(true, cameraSettings); //clear temp data viewport.RenderingContext.MultiRenderTarget_DestroyAll(); viewport.RenderingContext.DynamicTexture_DestroyAll(); } finally { scene.GetDisplayDevelopmentDataInThisApplicationOverride -= Scene_GetDisplayDevelopmentDataInThisApplicationOverride; } texture.Result.GetRealObject(true).BlitTo(viewport.RenderingContext.CurrentViewNumber, textureRead.Result.GetRealObject(true), 0, 0); //!!!!pitch //get data var totalBytes = PixelFormatUtility.GetNumElemBytes(format) * resolution.X * resolution.Y; var data = new byte[totalBytes]; unsafe { fixed(byte *pBytes = data) { var demandedFrame = textureRead.Result.GetRealObject(true).Read((IntPtr)pBytes, 0); while (RenderingSystem.CallBgfxFrame() < demandedFrame) { } } } var image = new ImageUtility.Image2D(format, resolution, data); //reset alpha channel for (int y = 0; y < image.Size.Y; y++) { for (int x = 0; x < image.Size.X; x++) { var pixel = image.GetPixel(new Vector2I(x, y)); pixel.W = 1.0f; image.SetPixel(new Vector2I(x, y), pixel); } } //image.Data //image2D.Blit( index * size, faceImage ); //Vector2I index = Vector2I.Zero; //switch( face ) //{ //case 0: index = new Vector2I( 2, 1 ); break; //case 1: index = new Vector2I( 0, 1 ); break; //case 2: index = new Vector2I( 1, 0 ); break; //case 3: index = new Vector2I( 1, 2 ); break; //case 4: index = new Vector2I( 1, 1 ); break; //case 5: index = new Vector2I( 3, 1 ); break; //} //var faceImage = new ImageUtility.Image2D( format, new Vector2I( size, size ), data ); //image2D.Blit( index * size, faceImage ); //} if (!Directory.Exists(Path.GetDirectoryName(destRealFileName))) { Directory.CreateDirectory(Path.GetDirectoryName(destRealFileName)); } if (!ImageUtility.Save(destRealFileName, image.Data, image.Size, 1, image.Format, 1, 0, out var error)) { throw new Exception(error); } } catch (Exception e) { EditorMessageBox.ShowWarning(e.Message); return; } finally { texture?.Dispose(); textureRead?.Dispose(); } ScreenNotifications.Show("Rendering completed successfully."); }