// ImGuiTexInspect, a texture inspector widget for dear imgui //------------------------------------------------------------------------- // [SECTION] INCLUDES //------------------------------------------------------------------------- #define IMGUI_DEFINE_MATH_OPERATORS #include "imgui_tex_inspect.h" #include "imgui_tex_inspect_internal.h" #include "imgui.h" #include "imgui_internal.h" #if defined(_MSC_VER) #pragma warning(disable : 4996) // 'sprintf' considered unsafe #endif namespace ImGuiTexInspect { //------------------------------------------------------------------------- // [SECTION] FORWARD DECLARATIONS //------------------------------------------------------------------------- void UpdateShaderOptions(Inspector *inspector); void InspectorDrawCallback(const ImDrawList *parent_list, const ImDrawCmd *cmd); bool GetVisibleTexelRegionAndGetData(Inspector *inspector, ImVec2 &texelTL, ImVec2 &texelBR); //------------------------------------------------------------------------- // [SECTION] GLOBAL STATE //------------------------------------------------------------------------- // Input mapping structure, default values listed in the comments. struct InputMap { ImGuiMouseButton PanButton; // LMB enables panning when held InputMap(); }; InputMap::InputMap() { PanButton = ImGuiMouseButton_Left; } // Settings configured via SetNextPanelOptions etc. struct NextPanelSettings { InspectorFlags ToSet = 0; InspectorFlags ToClear = 0; }; // Main context / configuration structure for imgui_tex_inspect struct Context { InputMap Input; // Input mapping config ImGuiStorage Inspectors; // All the inspectors we've seen Inspector * CurrentInspector; // Inspector currently being processed NextPanelSettings NextPanelOptions; // Options configured for next inspector panel float ZoomRate = 1.3f; // How fast mouse wheel affects zoom float DefaultPanelHeight = 600; // Height of panel in pixels float DefaultInitialPanelWidth = 600; // Only applies when window first appears int MaxAnnotations = 1000; // Limit number of texel annotations for performance }; Context *GContext = nullptr; //------------------------------------------------------------------------- // [SECTION] USER FUNCTIONS //------------------------------------------------------------------------- void Init() { // Nothing to do here. But there might be in a later version. So client code should still call it! } void Shutdown() { // Nothing to do here. But there might be in a later version. So client code should still call it! } Context *CreateContext() { GContext = IM_NEW(Context); SetCurrentContext(GContext); return GContext; } void DestroyContext(Context *ctx) { if (ctx == NULL) { ctx = GContext; } if (ctx == GContext) { GContext = NULL; } for (ImGuiStorage::ImGuiStoragePair &pair : ctx->Inspectors.Data) { Inspector *inspector = (Inspector *)pair.val_p; if (inspector) { IM_DELETE(inspector); } } IM_DELETE(ctx); } void SetCurrentContext(Context *context) { ImGuiTexInspect::GContext = context; } void SetNextPanelFlags(InspectorFlags setFlags, InspectorFlags clearFlags) { SetFlag(GContext->NextPanelOptions.ToSet, setFlags); SetFlag(GContext->NextPanelOptions.ToClear, clearFlags); } bool BeginInspectorPanel(const char *title, ImTextureID texture, ImVec2 textureSize, InspectorFlags flags, SizeIncludingBorder sizeIncludingBorder) { const int borderWidth = 1; // Unpack size param. It's in the SizeIncludingBorder structure just to make sure users know what they're requesting ImVec2 size = sizeIncludingBorder.Size; ImGuiWindow *window = ImGui::GetCurrentWindow(); Context *ctx = GContext; const ImGuiID ID = window->GetID(title); const ImGuiIO &IO = ImGui::GetIO(); // Create or find inspector bool justCreated = GetByKey(ctx, ID) == NULL; ctx->CurrentInspector = GetOrAddByKey(ctx, ID); Inspector *inspector = ctx->CurrentInspector; justCreated |= !inspector->Initialized; // Cache the basics inspector->ID = ID; inspector->Texture = texture; inspector->TextureSize = textureSize; inspector->Initialized = true; // Handle incoming flags. We keep special track of the // newly set flags because somethings only take effect // the first time the flag is set. InspectorFlags newlySetFlags = ctx->NextPanelOptions.ToSet; if (justCreated) { SetFlag(newlySetFlags, flags); inspector->MaxAnnotatedTexels = ctx->MaxAnnotations; } SetFlag(inspector->Flags, newlySetFlags); ClearFlag(inspector->Flags, ctx->NextPanelOptions.ToClear); ClearFlag(newlySetFlags, ctx->NextPanelOptions.ToClear); ctx->NextPanelOptions = NextPanelSettings(); // Calculate panel size ImVec2 contentRegionAvail = ImGui::GetContentRegionAvail(); ImVec2 panelSize; // A size value of zero indicates we should use defaults if (justCreated) { panelSize = {size.x == 0 ? ImMax(ctx->DefaultInitialPanelWidth, contentRegionAvail.x) : size.x, size.y == 0 ? ctx->DefaultPanelHeight : size.y}; } else { panelSize = {size.x == 0 ? contentRegionAvail.x : size.x, size.y == 0 ? ctx->DefaultPanelHeight : size.y}; } inspector->PanelSize = panelSize; ImVec2 availablePanelSize = panelSize - ImVec2(borderWidth, borderWidth) * 2; { // Possibly update scale float newScale = -1; if (HasFlag(newlySetFlags, InspectorFlags_FillVertical)) { newScale = availablePanelSize.y / textureSize.y; } else if (HasFlag(newlySetFlags, InspectorFlags_FillHorizontal)) { newScale = availablePanelSize.x / textureSize.x; } else if (justCreated) { newScale = 1; } if (newScale != -1) { inspector->Scale = ImVec2(newScale, newScale); SetPanPos(inspector, ImVec2(0.5f, 0.5f)); } } RoundPanPos(inspector); ImVec2 textureSizePixels = inspector->Scale * textureSize; // Size whole texture would appear on screen ImVec2 viewSizeUV = availablePanelSize / textureSizePixels; // Cropped size in terms of UV ImVec2 uv0 = inspector->PanPos - viewSizeUV * 0.5; ImVec2 uv1 = inspector->PanPos + viewSizeUV * 0.5; ImVec2 drawImageOffset{borderWidth, borderWidth}; ImVec2 viewSize = availablePanelSize; if ((inspector->Flags & InspectorFlags_ShowWrap) == 0) { /* Don't crop the texture to UV [0,1] range. What you see outside this * range will depend on API and texture properties */ if (textureSizePixels.x < availablePanelSize.x) { // Not big enough to horizontally fill view viewSize.x = ImFloor(textureSizePixels.x); drawImageOffset.x += ImFloor((availablePanelSize.x - textureSizePixels.x) / 2); uv0.x = 0; uv1.x = 1; viewSizeUV.x = 1; inspector->PanPos.x = 0.5f; } if (textureSizePixels.y < availablePanelSize.y) { // Not big enough to vertically fill view viewSize.y = ImFloor(textureSizePixels.y); drawImageOffset.y += ImFloor((availablePanelSize.y - textureSizePixels.y) / 2); uv0.y = 0; uv1.y = 1; viewSizeUV.y = 1; inspector->PanPos.y = 0.5; } } if (HasFlag(flags,InspectorFlags_FlipX)) { ImSwap(uv0.x, uv1.x); viewSizeUV.x *= -1; } if (HasFlag(flags,InspectorFlags_FlipY)) { ImSwap(uv0.y, uv1.y); viewSizeUV.y *= -1; } inspector->ViewSize = viewSize; inspector->ViewSizeUV = viewSizeUV; /* We use mouse scroll to zoom so we don't want scroll to propagate to * parent window. For this to happen we must NOT set * ImGuiWindowFlags_NoScrollWithMouse. This seems strange but it's the way * ImGui works. Also we must ensure the ScrollMax.y is not zero for the * child window. */ if (ImGui::BeginChild(title, panelSize, false, ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoMove)) { // See comment above ImGui::GetCurrentWindow()->ScrollMax.y = 1.0f; // Callback for using our own image shader ImGui::GetWindowDrawList()->AddCallback(InspectorDrawCallback, inspector); // Keep track of size of area that we draw for borders later inspector->PanelTopLeftPixel = ImGui::GetCursorScreenPos(); ImGui::SetCursorPos(ImGui::GetCursorPos() + drawImageOffset); inspector->ViewTopLeftPixel = ImGui::GetCursorScreenPos(); UpdateShaderOptions(inspector); inspector->CachedShaderOptions = inspector->ActiveShaderOptions; ImGui::Image(texture, viewSize, uv0, uv1); ImGui::GetWindowDrawList()->AddCallback(ImDrawCallback_ResetRenderState, nullptr); /* Matrices for going back and forth between texel coordinates in the * texture and screen coordinates based on where texture is drawn. * Useful for annotations and mouse hover etc. */ inspector->TexelsToPixels = GetTexelsToPixels(inspector->ViewTopLeftPixel, viewSize, uv0, viewSizeUV, inspector->TextureSize); inspector->PixelsToTexels = inspector->TexelsToPixels.Inverse(); ImVec2 mousePos = ImGui::GetMousePos(); ImVec2 mousePosTexel = inspector->PixelsToTexels * mousePos; ImVec2 mouseUV = mousePosTexel / textureSize; mousePosTexel.x = Modulus(mousePosTexel.x, textureSize.x); mousePosTexel.y = Modulus(mousePosTexel.y, textureSize.y); if (ImGui::IsItemHovered() && (inspector->Flags & ImGuiTexInspect::InspectorFlags_NoTooltip) == 0) { // Show a tooltip for currently hovered texel ImVec2 texelTL; ImVec2 texelBR; if (GetVisibleTexelRegionAndGetData(inspector, texelTL, texelBR)) { ImVec4 color = GetTexel(&inspector->Buffer, (int)mousePosTexel.x, (int)mousePosTexel.y); char buffer[128]; sprintf(buffer, "UV: (%.5f, %.5f)\nTexel: (%d, %d)", mouseUV.x, mouseUV.y, (int)mousePosTexel.x, (int)mousePosTexel.y); ImGui::ColorTooltip(buffer, &color.x, 0); } } bool hovered = ImGui::IsWindowHovered(); { //DRAGGING // start drag if (!inspector->IsDragging && hovered && IO.MouseClicked[ctx->Input.PanButton]) { inspector->IsDragging = true; } // carry on dragging else if (inspector->IsDragging) { ImVec2 uvDelta = IO.MouseDelta * viewSizeUV / viewSize; inspector->PanPos -= uvDelta; RoundPanPos(inspector); } // end drag if (inspector->IsDragging && (IO.MouseReleased[ctx->Input.PanButton] || !IO.MouseDown[ctx->Input.PanButton])) { inspector->IsDragging = false; } } // ZOOM if (hovered && IO.MouseWheel != 0) { float zoomRate = ctx->ZoomRate; float scale = inspector->Scale.y; float prevScale = scale; bool keepTexelSizeRegular = scale > inspector->MinimumGridSize && !HasFlag(inspector->Flags, InspectorFlags_NoGrid); if (IO.MouseWheel > 0) { scale *= zoomRate; if (keepTexelSizeRegular) { // It looks nicer when all the grid cells are the same size // so keep scale integer when zoomed in scale = ImCeil(scale); } } else { scale /= zoomRate; if (keepTexelSizeRegular) { // See comment above. We're doing a floor this time to make // sure the scale always changes when scrolling scale = ImFloorSigned(scale); } } /* To make it easy to get back to 1:1 size we ensure that we stop * here without going straight past it*/ if ((prevScale < 1 && scale > 1) || (prevScale > 1 && scale < 1)) { scale = 1; } SetScale(inspector, ImVec2(inspector->PixelAspectRatio * scale, scale)); SetPanPos(inspector, inspector->PanPos + (mouseUV - inspector->PanPos) * (1 - prevScale / scale)); } return true; } else { return false; } } bool BeginInspectorPanel(const char *name, ImTextureID texture, ImVec2 textureSize, InspectorFlags flags) { return BeginInspectorPanel(name, texture, textureSize, flags, SizeIncludingBorder{{0, 0}}); } bool BeginInspectorPanel(const char *name, ImTextureID texture, ImVec2 textureSize, InspectorFlags flags, SizeExcludingBorder size) { // Correct the size to include the border, but preserve 0 which has a special meaning return BeginInspectorPanel(name, texture, textureSize, flags, SizeIncludingBorder{ImVec2{size.size.x == 0 ? 0 : size.size.x + 2, size.size.y == 0 ? 0 : size.size.y + 2}}); } void EndInspectorPanel() { const ImU32 innerBorderColour = 0xFFFFFFFF; const ImU32 outerBorderColour = 0xFF888888; Inspector *inspector = GContext->CurrentInspector; // Draw out border around whole inspector panel ImGui::GetWindowDrawList()->AddRect(inspector->PanelTopLeftPixel, inspector->PanelTopLeftPixel + inspector->PanelSize, outerBorderColour); // Draw innder border around texture. If zoomed in this will completely cover the outer border ImGui::GetWindowDrawList()->AddRect(inspector->ViewTopLeftPixel - ImVec2(1, 1), inspector->ViewTopLeftPixel + inspector->ViewSize + ImVec2(1, 1), innerBorderColour); ImGui::EndChild(); // We set this back to false every frame in case the texture is dynamic if (!HasFlag(inspector->Flags, InspectorFlags_NoAutoReadTexture)) { inspector->HaveCurrentTexelData = false; } } void ReleaseInspectorData(ImGuiID ID) { Inspector *inspector = GetByKey(GContext, ID); if (inspector == NULL) return; if (inspector->DataBuffer) { IM_FREE(inspector->DataBuffer); inspector->DataBuffer = NULL; inspector->DataBufferSize = 0; } /* In a later version we will remove inspector from the inspector table * altogether. For now we reset the whole inspector structure to prevent * clients relying on persisted data. */ *inspector = Inspector(); } ImGuiID CurrentInspector_GetID() { return GContext->CurrentInspector->ID; } void CurrentInspector_SetColorMatrix(const float (&matrix)[16], const float (&colorOffset)[4]) { Inspector *inspector = GContext->CurrentInspector; ShaderOptions *shaderOptions = &inspector->ActiveShaderOptions; memcpy(shaderOptions->ColorTransform, matrix, sizeof(matrix)); memcpy(shaderOptions->ColorOffset, colorOffset, sizeof(colorOffset)); } void CurrentInspector_ResetColorMatrix() { Inspector *inspector = GContext->CurrentInspector; ShaderOptions *shaderOptions = &inspector->ActiveShaderOptions; shaderOptions->ResetColorTransform(); } void CurrentInspector_SetAlphaMode(InspectorAlphaMode mode) { Inspector *inspector = GContext->CurrentInspector; ShaderOptions *shaderOptions = &inspector->ActiveShaderOptions; inspector->AlphaMode = mode; switch (mode) { case InspectorAlphaMode_Black: shaderOptions->BackgroundColor = ImVec4(0, 0, 0, 1); shaderOptions->DisableFinalAlpha = 1; shaderOptions->PremultiplyAlpha = 1; break; case InspectorAlphaMode_White: shaderOptions->BackgroundColor = ImVec4(1, 1, 1, 1); shaderOptions->DisableFinalAlpha = 1; shaderOptions->PremultiplyAlpha = 1; break; case InspectorAlphaMode_ImGui: shaderOptions->BackgroundColor = ImVec4(0, 0, 0, 0); shaderOptions->DisableFinalAlpha = 0; shaderOptions->PremultiplyAlpha = 0; break; case InspectorAlphaMode_CustomColor: shaderOptions->BackgroundColor = inspector->CustomBackgroundColor; shaderOptions->DisableFinalAlpha = 1; shaderOptions->PremultiplyAlpha = 1; break; } } void CurrentInspector_SetFlags(InspectorFlags toSet, InspectorFlags toClear) { Inspector *inspector = GContext->CurrentInspector; SetFlag(inspector->Flags, toSet); ClearFlag(inspector->Flags, toClear); } void CurrentInspector_SetGridColor(ImU32 color) { Inspector *inspector = GContext->CurrentInspector; float alpha = inspector->ActiveShaderOptions.GridColor.w; inspector->ActiveShaderOptions.GridColor = ImColor(color); inspector->ActiveShaderOptions.GridColor.w = alpha; } void CurrentInspector_SetMaxAnnotations(int maxAnnotations) { Inspector *inspector = GContext->CurrentInspector; inspector->MaxAnnotatedTexels = maxAnnotations; } void CurrentInspector_InvalidateTextureCache() { Inspector *inspector = GContext->CurrentInspector; inspector->HaveCurrentTexelData = false; } void CurrentInspector_SetCustomBackgroundColor(ImVec4 color) { Inspector *inspector = GContext->CurrentInspector; inspector->CustomBackgroundColor = color; if (inspector->AlphaMode == InspectorAlphaMode_CustomColor) { inspector->ActiveShaderOptions.BackgroundColor = color; } } void CurrentInspector_SetCustomBackgroundColor(ImU32 color) { CurrentInspector_SetCustomBackgroundColor(ImGui::ColorConvertU32ToFloat4(color)); } void DrawColorMatrixEditor() { const char *colorVectorNames[] = {"R", "G", "B", "A", "1"}; const char *finalColorVectorNames[] = {"R'", "G'", "B'", "A'"}; const float dragSpeed = 0.02f; Inspector *inspector = GContext->CurrentInspector; ShaderOptions *shaderOptions = &inspector->ActiveShaderOptions; // Left hand side of equation. The final color vector which is the actual drawn color TextVector("FinalColorVector", finalColorVectorNames, IM_ARRAYSIZE(finalColorVectorNames)); ImGui::SameLine(); ImGui::TextUnformatted("="); ImGui::SameLine(); // Right hand side of the equation: the Matrix. This is the editable part ImGui::BeginGroup(); for (int i = 0; i < 4; ++i) { ImGui::PushID(i); for (int j = 0; j < 4; ++j) { ImGui::PushID(j); ImGui::SetNextItemWidth(50); ImGui::DragFloat("##f", &shaderOptions->ColorTransform[j * 4 + i], dragSpeed); ImGui::PopID(); ImGui::SameLine(); } ImGui::SetNextItemWidth(50); ImGui::DragFloat("##offset", &shaderOptions->ColorOffset[i], dragSpeed); ImGui::PopID(); } ImGui::EndGroup(); ImGui::SameLine(); ImGui::TextUnformatted("*"); ImGui::SameLine(); // Right hand side of equation. The input vector, the source color of the texel. TextVector("ColorVector", colorVectorNames, IM_ARRAYSIZE(colorVectorNames)); } void DrawGridEditor() { Inspector *inspector = GContext->CurrentInspector; ImGui::BeginGroup(); bool gridEnabled = !HasFlag(inspector->Flags, InspectorFlags_NoGrid); if (ImGui::Checkbox("Grid", &gridEnabled)) { if (gridEnabled) { CurrentInspector_ClearFlags(InspectorFlags_NoGrid); } else { CurrentInspector_SetFlags(InspectorFlags_NoGrid); } } if (gridEnabled) { ImGui::SameLine(); ImGui::ColorEdit3("Grid Color", (float *)&inspector->ActiveShaderOptions.GridColor, ImGuiColorEditFlags_NoInputs); } ImGui::EndGroup(); } void DrawColorChannelSelector() { Inspector *inspector = GContext->CurrentInspector; ShaderOptions *shaderOptions = &inspector->ActiveShaderOptions; ImGuiStorage *storage = ImGui::GetStateStorage(); const ImGuiID greyScaleID = ImGui::GetID("greyScale"); bool greyScale = storage->GetBool(greyScaleID, false); bool red = shaderOptions->ColorTransform[0] > 0; bool green = shaderOptions->ColorTransform[5] > 0; bool blue = shaderOptions->ColorTransform[10] > 0; bool changed = false; // In greyScale made we draw the red, green, blue checkboxes as disabled if (greyScale) { PushDisabled(); } ImGui::BeginGroup(); changed |= ImGui::Checkbox("Red", &red); changed |= ImGui::Checkbox("Green", &green); changed |= ImGui::Checkbox("Blue", &blue); ImGui::EndGroup(); ImGui::SameLine(); if (greyScale) { PopDisabled(); } if (changed) { // Overwrite the color transform matrix with one based on the settings shaderOptions->ResetColorTransform(); shaderOptions->ColorTransform[0] = red ? 1.0f : 0.0f; shaderOptions->ColorTransform[5] = green ? 1.0f : 0.0f; shaderOptions->ColorTransform[10] = blue ? 1.0f : 0.0f; } ImGui::BeginGroup(); if (ImGui::Checkbox("Grey", &greyScale)) { shaderOptions->ResetColorTransform(); storage->SetBool(greyScaleID, greyScale); if (greyScale) { for (int i = 0; i < 3; i++) { for (int j = 0; j < 3; j++) { shaderOptions->ColorTransform[i * 4 + j] = 0.333f; } } } } ImGui::EndGroup(); } void DrawAlphaModeSelector() { Inspector *inspector = GContext->CurrentInspector; const char *alphaModes[] = {"ImGui Background", "Black", "White", "Custom Color"}; ImGui::SetNextItemWidth(200); InspectorAlphaMode currentAlphaMode = inspector->AlphaMode; ImGui::Combo("Alpha Modes", (int *)¤tAlphaMode, alphaModes, IM_ARRAYSIZE(alphaModes)); CurrentInspector_SetAlphaMode(currentAlphaMode); if (inspector->AlphaMode == InspectorAlphaMode_CustomColor) { ImVec4 backgroundColor = inspector->CustomBackgroundColor; if (ImGui::ColorEdit3("Background Color", (float *)&backgroundColor, 0)) { CurrentInspector_SetCustomBackgroundColor(backgroundColor); } } } void SetZoomRate(float rate) { GContext->ZoomRate = rate; } //------------------------------------------------------------------------- // [SECTION] Life Cycle //------------------------------------------------------------------------- Inspector::~Inspector() { if (DataBuffer) { IM_FREE(DataBuffer); } } //------------------------------------------------------------------------- // [SECTION] Scaling and Panning //------------------------------------------------------------------------- void RoundPanPos(Inspector *inspector) { if ((inspector->Flags & InspectorFlags_ShowWrap) > 0) { /* PanPos is the point in the center of the current view. Allow the * user to pan anywhere as long as the view center is inside the * texture.*/ inspector->PanPos = ImClamp(inspector->PanPos, ImVec2(0, 0), ImVec2(1, 1)); } else { /* When ShowWrap mode is disabled the limits are a bit more strict. We * try to keep it so that the user cannot pan past the edge of the * texture at all.*/ ImVec2 absViewSizeUV = Abs(inspector->ViewSizeUV); inspector->PanPos = ImMax(inspector->PanPos - absViewSizeUV / 2, ImVec2(0, 0)) + absViewSizeUV / 2; inspector->PanPos = ImMin(inspector->PanPos + absViewSizeUV / 2, ImVec2(1, 1)) - absViewSizeUV / 2; } /* If inspector->scale is 1 then we should ensure that pixels are aligned * with texel centers to get pixel-perfect texture rendering*/ ImVec2 topLeftSubTexel = inspector->PanPos * inspector->Scale * inspector->TextureSize - inspector->ViewSize * 0.5f; if (inspector->Scale.x >= 1) { topLeftSubTexel.x = Round(topLeftSubTexel.x); } if (inspector->Scale.y >= 1) { topLeftSubTexel.y = Round(topLeftSubTexel.y); } inspector->PanPos = (topLeftSubTexel + inspector->ViewSize * 0.5f) / (inspector->Scale * inspector->TextureSize); } void SetPanPos(Inspector *inspector, ImVec2 pos) { inspector->PanPos = pos; RoundPanPos(inspector); } void SetScale(Inspector *inspector, ImVec2 scale) { scale = ImClamp(scale, inspector->ScaleMin, inspector->ScaleMax); inspector->ViewSizeUV *= inspector->Scale / scale; inspector->Scale = scale; // Only force nearest sampling if zoomed in inspector->ActiveShaderOptions.ForceNearestSampling = (inspector->Scale.x > 1.0f || inspector->Scale.y > 1.0f) && !HasFlag(inspector->Flags, InspectorFlags_NoForceFilterNearest); inspector->ActiveShaderOptions.GridWidth = ImVec2(1.0f / inspector->Scale.x, 1.0f / inspector->Scale.y); } void SetScale(Inspector *inspector, float scaleY) { SetScale(inspector, ImVec2(scaleY * inspector->PixelAspectRatio, scaleY)); } //------------------------------------------------------------------------- // [SECTION] INSPECTOR MAP //------------------------------------------------------------------------- Inspector *GetByKey(const Context *ctx, ImGuiID key) { return (Inspector *)ctx->Inspectors.GetVoidPtr(key); } Inspector *GetOrAddByKey(Context *ctx, ImGuiID key) { Inspector *inspector = GetByKey(ctx, key); if (inspector) { return inspector; } else { inspector = IM_NEW(Inspector); ctx->Inspectors.SetVoidPtr(key, inspector); return inspector; } } //------------------------------------------------------------------------- // [SECTION] TextureConversion class //------------------------------------------------------------------------- void ShaderOptions::ResetColorTransform() { memset(ColorTransform, 0, sizeof(ColorTransform)); for (int i = 0; i < 4; ++i) { ColorTransform[i * 4 + i] = 1; } } ShaderOptions::ShaderOptions() { ResetColorTransform(); memset(ColorOffset, 0, sizeof(ColorOffset)); } //------------------------------------------------------------------------- // [SECTION] UI and CONFIG //------------------------------------------------------------------------- void UpdateShaderOptions(Inspector *inspector) { if (HasFlag(inspector->Flags, InspectorFlags_NoGrid) == false && inspector->Scale.y > inspector->MinimumGridSize) { // Enable grid in shader inspector->ActiveShaderOptions.GridColor.w = 1; SetScale(inspector, Round(inspector->Scale.y)); } else { // Disable grid in shader inspector->ActiveShaderOptions.GridColor.w = 0; } inspector->ActiveShaderOptions.ForceNearestSampling = (inspector->Scale.x > 1.0f || inspector->Scale.y > 1.0f) && !HasFlag(inspector->Flags, InspectorFlags_NoForceFilterNearest); } // Draws a single column ImGui table with one row for each provided string void TextVector(const char *title, const char *const *strings, int stringCount) { ImGui::BeginGroup(); ImGui::SetNextItemWidth(50); if (ImGui::BeginTable(title, 1, ImGuiTableFlags_BordersOuter | ImGuiTableFlags_SizingFixedFit | ImGuiTableFlags_NoHostExtendX)) { for (int i = 0; i < stringCount; ++i) { ImGui::TableNextRow(); ImGui::TableSetColumnIndex(0); ImGui::SetNextItemWidth(50); ImGui::Text("%s", strings[i]); } ImGui::EndTable(); } ImGui::EndGroup(); } const ImGuiCol disabledUIColorIds[] = {ImGuiCol_FrameBg, ImGuiCol_FrameBgActive, ImGuiCol_FrameBgHovered, ImGuiCol_Text, ImGuiCol_CheckMark}; // Push disabled style for ImGui elements void PushDisabled() { for (ImGuiCol colorId : disabledUIColorIds) { ImVec4 color = ImGui::GetStyleColorVec4(colorId); color = color * ImVec4(0.5f, 0.5f, 0.5f, 0.5f); ImGui::PushStyleColor(colorId, color); } ImGui::PushItemFlag(ImGuiItemFlags_Disabled, true); } // Pop disabled style for ImGui elements void PopDisabled() { for (ImGuiCol colorId : disabledUIColorIds) { (void)colorId; ImGui::PopStyleColor(); } ImGui::PopItemFlag(); } //------------------------------------------------------------------------- // [SECTION] Rendering & Buffer Management //------------------------------------------------------------------------- void InspectorDrawCallback(const ImDrawList *parent_list, const ImDrawCmd *cmd) { // Forward call to API-specific backend Inspector *inspector = (Inspector *)cmd->UserCallbackData; BackEnd_SetShader(parent_list, cmd, inspector); } // Calculate a transform to convert from texel coordinates to screen pixel coordinates Transform2D GetTexelsToPixels(ImVec2 screenTopLeft, ImVec2 screenViewSize, ImVec2 uvTopLeft, ImVec2 uvViewSize, ImVec2 textureSize) { ImVec2 uvToPixel = screenViewSize / uvViewSize; Transform2D transform; transform.Scale = uvToPixel / textureSize; transform.Translate.x = screenTopLeft.x - uvTopLeft.x * uvToPixel.x; transform.Translate.y = screenTopLeft.y - uvTopLeft.y * uvToPixel.y; return transform; } /* Fills in the AnnotationsDesc structure which provides all necessary * information for code which draw annoations. Returns false if no annoations * should be drawn. The maxAnnotatedTexels argument provides a way to override * the default maxAnnotatedTexels. */ bool GetAnnotationDesc(AnnotationsDesc *ad, ImU64 maxAnnotatedTexels) { Inspector *inspector = GContext->CurrentInspector; if (maxAnnotatedTexels == 0) { maxAnnotatedTexels = inspector->MaxAnnotatedTexels; } if (maxAnnotatedTexels != 0) { /* Check if we would draw too many annotations. This is to avoid poor * frame rate when too zoomed out. Increase MaxAnnotatedTexels if you * want to draw more annotations. Note that we don't use texelTL & * texelBR to get total visible texels as this would cause flickering * while panning as the exact number of visible texels changes. */ ImVec2 screenViewSizeTexels = Abs(inspector->PixelsToTexels.Scale) * inspector->ViewSize; ImU64 approxVisibleTexelCount = (ImU64)screenViewSizeTexels.x * (ImU64)screenViewSizeTexels.y; if (approxVisibleTexelCount > maxAnnotatedTexels) { return false; } } // texelTL & texelBL will describe the currently visible texel region ImVec2 texelTL; ImVec2 texelBR; if (GetVisibleTexelRegionAndGetData(inspector, texelTL, texelBR)) { ad->Buffer= inspector->Buffer; ad->DrawList = ImGui::GetWindowDrawList(); ad->TexelsToPixels = inspector->TexelsToPixels; ad->TexelTopLeft = texelTL; ad->TexelViewSize = texelBR - texelTL; return true; } return false; } /* Calculates currently visible region of texture (which is returned in texelTL * and texelBR) then also actually ensure that that data is in memory. Returns * false if fetching data failed. */ bool GetVisibleTexelRegionAndGetData(Inspector *inspector, ImVec2 &texelTL, ImVec2 &texelBR) { /* Figure out which texels correspond to the top left and bottom right * corners of the texture view. The plus + ImVec2(1,1) is because we * want to draw partially visible texels on the bottom and right edges. */ texelTL = ImFloor(inspector->PixelsToTexels * inspector->ViewTopLeftPixel); texelBR = ImFloor(inspector->PixelsToTexels * (inspector->ViewTopLeftPixel + inspector->ViewSize)); if (texelTL.x > texelBR.x) { ImSwap(texelTL.x, texelBR.x); } if (texelTL.y > texelBR.y) { ImSwap(texelTL.y, texelBR.y); } /* Add ImVec2(1,1) because we want to draw partially visible texels on the * bottom and right edges.*/ texelBR += ImVec2(1,1); texelTL = ImClamp(texelTL, ImVec2(0, 0), inspector->TextureSize); texelBR = ImClamp(texelBR, ImVec2(0, 0), inspector->TextureSize); if (inspector->HaveCurrentTexelData) { return true; } // Now request pixel data for this region from backend ImVec2 texelViewSize = texelBR - texelTL; if (ImMin(texelViewSize.x, texelViewSize.y) > 0) { if (BackEnd_GetData(inspector, inspector->Texture, (int)texelTL.x, (int)texelTL.y, (int)texelViewSize.x, (int)texelViewSize.y, &inspector->Buffer)) { inspector->HaveCurrentTexelData = true; return true; } } return false; } /* This is a function the backends can use to allocate a buffer for storing * texture texel data. The buffer is owned by the inpsector so the backend * code doesn't need to worry about freeing it. */ ImU8 *GetBuffer(Inspector *inspector, size_t bytes) { if (inspector->DataBufferSize < bytes || inspector->DataBuffer == nullptr) { // We need to allocate a buffer if (inspector->DataBuffer) { IM_FREE(inspector->DataBuffer); } // Allocate slightly more than we need to avoid reallocating // very frequently in the case that size is increasing. size_t size = bytes * 5 / 4; inspector->DataBuffer = (ImU8 *)IM_ALLOC(size); inspector->DataBufferSize = size; } return inspector->DataBuffer; } ImVec4 GetTexel(const BufferDesc *bd, int x, int y) { if (x < bd->StartX || x >= bd->StartX + bd->Width || y < bd->StartY || y >= bd->StartY + bd->Height) { // Outside the range of data in the buffer. return ImVec4(); } // Calculate position in array size_t offset = ((size_t)bd->LineStride * (y - bd->StartY) + bd->Stride * (x - bd->StartX)); if (bd->Data_float) { const float *texel = bd->Data_float + offset; // It's possible our buffer doesn't have all 4 channels so fill gaps in with zeros return ImVec4( texel[bd->Red], bd->ChannelCount >= 2 ? texel[bd->Green] : 0, bd->ChannelCount >= 3 ? texel[bd->Blue] : 0, bd->ChannelCount >= 4 ? texel[bd->Alpha] : 0); } else if (bd->Data_uint8_t) { const ImU8 *texel = bd->Data_uint8_t + offset; // It's possible our buffer doesn't have all 4 channels so fill gaps in with zeros. // Also map from [0,255] to [0,1] return ImVec4( (float)texel[bd->Red] / 255.0f, bd->ChannelCount >= 2 ? (float)texel[bd->Green] / 255.0f : 0, bd->ChannelCount >= 3 ? (float)texel[bd->Blue] / 255.0f : 0, bd->ChannelCount >= 4 ? (float)texel[bd->Alpha] / 255.0f : 0); } else { return ImVec4(); } } //------------------------------------------------------------------------- // [SECTION] Annotations //------------------------------------------------------------------------- ValueText::ValueText(Format format) { /* The ValueText annotation draws a string inside each texel displaying the * values of each channel. We now select a format string based on the enum * parameter*/ switch (format) { case Format::HexString: TextFormatString = "#%02X%02X%02X%02X"; TextColumnCount = 9; TextRowCount = 1; FormatAsFloats = false; break; case Format::BytesHex: TextFormatString = "R:#%02X\nG:#%02X\nB:#%02X\nA:#%02X"; TextColumnCount = 5; TextRowCount = 4; FormatAsFloats = false; break; case Format::BytesDec: TextFormatString = "R:%3d\nG:%3d\nB:%3d\nA:%3d"; TextColumnCount = 5; TextRowCount = 4; FormatAsFloats = false; break; case Format::Floats: TextFormatString = "%5.3f\n%5.3f\n%5.3f\n%5.3f"; TextColumnCount = 5; TextRowCount = 4; FormatAsFloats = true; break; } } void ValueText::DrawAnnotation(ImDrawList *drawList, ImVec2 texel, Transform2D texelsToPixels, ImVec4 value) { char buffer[64]; float fontHeight = ImGui::GetFontSize(); float fontWidth = fontHeight / 2; /* WARNING this is a hack that gets a constant * character width from half the height. This work for the default font but * won't work on other fonts which may even not be monospace.*/ // Calculate size of text and check if it fits ImVec2 textSize = ImVec2((float)TextColumnCount * fontWidth, (float)TextRowCount * fontHeight); if (textSize.x > ImAbs(texelsToPixels.Scale.x) || textSize.y > ImAbs(texelsToPixels.Scale.y)) { // Not enough room in texel to fit the text. Don't draw it. return; } /* Choose black or white text based on how bright the texel. I.e. don't * draw black text on a dark background or vice versa. */ float brightness = (value.x + value.y + value.z) * value.w / 3; ImU32 lineColor = brightness > 0.5 ? 0xFF000000 : 0xFFFFFFFF; if (FormatAsFloats) { sprintf(buffer, TextFormatString, value.x, value.y, value.z, value.w); } else { /* Map [0,1] to [0,255]. Also clamp it since input data wasn't * necessarily in [0,1] range. */ ImU8 r = (ImU8)Round((ImClamp(value.x, 0.0f, 1.0f)) * 255); ImU8 g = (ImU8)Round((ImClamp(value.y, 0.0f, 1.0f)) * 255); ImU8 b = (ImU8)Round((ImClamp(value.z, 0.0f, 1.0f)) * 255); ImU8 a = (ImU8)Round((ImClamp(value.w, 0.0f, 1.0f)) * 255); sprintf(buffer, TextFormatString, r, g, b, a); } // Add text to drawlist! ImVec2 pixelCenter = texelsToPixels * texel; drawList->AddText(pixelCenter - textSize * 0.5f, lineColor, buffer); } Arrow::Arrow(int xVectorIndex, int yVectorIndex, ImVec2 lineScale) : VectorIndex_x(xVectorIndex), VectorIndex_y(yVectorIndex), LineScale(lineScale) { } Arrow &Arrow::UsePreset(Preset preset) { switch (preset) { default: case Preset::NormalMap: VectorIndex_x = 0; VectorIndex_y = 1; LineScale = ImVec2(1, -1); ZeroPoint = ImVec2(128.0f / 255, 128.0f / 255); break; case Preset::NormalizedFloat: VectorIndex_x = 0; VectorIndex_y = 1; LineScale = ImVec2(0.5f, -0.5f); ZeroPoint = ImVec2(0, 0); break; } return *this; } void Arrow::DrawAnnotation(ImDrawList *drawList, ImVec2 texel, Transform2D texelsToPixels, ImVec4 value) { const float arrowHeadScale = 0.35f; const ImU32 lineColor = 0xFFFFFFFF; float *vecPtr = &value.x; // Draw an arrow! ImVec2 lineDir = (ImVec2(vecPtr[VectorIndex_x], vecPtr[VectorIndex_y]) - ZeroPoint) * LineScale; ImVec2 lineStart = texel; ImVec2 lineEnd = lineStart + lineDir; ImVec2 arrowHead1 = ImVec2(lineDir.x - lineDir.y, lineDir.x + lineDir.y) * -arrowHeadScale; ImVec2 arrowHead2 = ImVec2(lineDir.x + lineDir.y, -lineDir.x + lineDir.y) * -arrowHeadScale; DrawAnnotationLine(drawList, lineStart, lineEnd, texelsToPixels, lineColor); DrawAnnotationLine(drawList, lineEnd, lineEnd + arrowHead1, texelsToPixels, lineColor); DrawAnnotationLine(drawList, lineEnd, lineEnd + arrowHead2, texelsToPixels, lineColor); } void DrawAnnotationLine(ImDrawList *drawList, ImVec2 fromTexel, ImVec2 toTexel, Transform2D texelsToPixels, ImU32 color) { ImVec2 lineFrom = texelsToPixels * fromTexel; ImVec2 lineTo = texelsToPixels * toTexel; drawList->AddLine(lineFrom, lineTo, color, 1.0f); } } // namespace ImGuiTexInspect