sudara / melatonin_inspector

A JUCE module that gives you the ability to inspect and visually edit (non-destructively) components in your UI.

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

Melatonin Component Inspector

A JUCE module that gives you the ability to inspect and visually edit (non-destructively) components in your UI.

It's inspired by Figma (where I prefer to design UI), web browser web inspectors and Jim Credland's Component Debugger juce-toys.

A big hearty thanks to Dmytro Kiro and Roland Rabien (aka FigBug) for contributing some great features!


✨✨
✨✨✨
...the features...
✨✨✨
✨✨

Browse and select components visually

Point n' click to inspect a component, see its size and distance to parent.

Explore the component hierarchy

Immediately gain clarity over parent/child relationships and see what components are currently visible.

Filter components by name. Names are derived from stock components, label/button text, or demangled class names.

Preview Component

See what exactly is drawing on a per-component basis, even when the component is hidden. A fixed transparency grid helps you understand which components and images have transparency.

Edit component position and spacing

There's like...4 different ways to do this, visually and numerically...

We also display component padding if you follow the convention of storing them as the component properties paddingLeft, paddingTop, paddingRight, paddingBottom. See my PaddedComponent base class as an example.

Inspect and modify component flags and properties

See the most important component properties at a glance, including look and feels, fonts for labels, etc. Where applicable, flags are editable!

Any custom properties you've added the component will also show up here and be editable.

AudioPluginHost - 2023-08-14 01

Nudge components around

Verify new values, get things pixel perfect.

View spacing relative to siblings/neighbors

Hold "alt" while component is selected. A Figma inspired feature.

Display and modify JUCE widget colors

No, it's not a christmas miracle, but we do magically display JUCE's friendly enum ColourIds names from the stock widgets.

See what that Slider's trackColourId is set to, and hey, go ahead and try out a new theme in real time.

(Just to be clear: The color changes are temporary, no code is edited!)

AudioPluginHost - 2023-08-14 38

Color picker

Accurately pinpoint colors. Click to pick and store one. Toggle between RGBA and HEX values.

FPS meter

Overlay an FPS meter on your Editor to get an intuitive understanding of your painting performance. Please see the FAQ for details on usage.

Display component performance in real time

A life saving feature.

See time spent exclusively in a component's paint method as well as conveniently provide you with a sum with all children.

Keep track of the max. Double click to repaint and get fresh timings. See setup paint timing.

AudioPluginHost - 2023-08-16 57

Installing with CMake

CMake option #1: FetchContent

Place this chunk o love somewhere before your call to juce_add_plugin or juce_add_gui_app:

Include (FetchContent)
FetchContent_Declare (melatonin_inspector
  GIT_REPOSITORY https://github.com/sudara/melatonin_inspector.git
  GIT_TAG origin/main
  SOURCE_DIR ${CMAKE_CURRENT_BINARY_DIR}/melatonin_inspector)
FetchContent_MakeAvailable (melatonin_inspector)

CMake option #2 git submodules

If you are a git submodule aficionado, life is great! Add this submodule to your project:

git submodule add -b main https://github.com/sudara/melatonin_inspector.git modules/melatonin_inspector

and then simply call add_subdirectory in your CMakeLists.txt. Remember, modules go before your main call to juce_add_plugin or juce_add_gui_app (this makes life easier in IDEs):

add_subdirectory (modules/melatonin_inspector)

To update melatonin_inspector down the road (gasp! maintained dependencies!?):

git submodule update --remote --merge modules/melatonin_inspector

CMake Step 2: Tell JUCE about the module

Wait wait, not so fast! You couldn't get away that easily.

After your juce_add_plugin call you will need to link your plugin to the module's target, for example:

target_link_libraries("YourProject" PRIVATE melatonin_inspector)

Note: you don't have to call juce_add_module. That's handled by our CMake.

If you use Projucer, add the module manually.

Installing with Projucer

If you're rolling old school, or just prefer Projucer life, you'll be happy to note that though JUCE doesn't make it easy we've bent over backwards to make sure our icons, etc are included in the module.

Download the module

You can still use git to add it as a submodule if you'd like stay up to date with any changes:

git submodule add -b main https://github.com/sudara/melatonin_inspector.git modules/melatonin_inspector

Or just download it and stick it somewhere.

Add it to the projucer

Just "Add a module from a specified folder" and you're done!

2. Add an include to your Plugin Editor

Include the module header:

#include "melatonin_inspector/melatonin_inspector.h"

3. Add the inspector as a private member to your Editor

The easiest way to get started is to pass a reference to the root component of your UI (typically the Editor itself like in this example, but you could also inspect anything that derives from juce::Component).

melatonin::Inspector inspector { *this };

If you prefer the inspector open in the disabled state by default, you can pass false as the second argument.

melatonin::Inspector inspector { *this, false };

4. Set it visible

When the inspector as an editor member, you can use cmd/ctrl i to toggle whether the inspector is enabled.

setVisible on the member will also pop the window open.

What I do is have a GUI toggle that pops open the window and enables inspection:

// open the inspector window
inspector.setVisible(true); 

// enable the inspector
inspector.toggle(true);

5. Optional: Make it smarter

Setting up as above means that the inspector will always be constructed with your editor. Clicking close on the inspector's DocumentWindow will just hide it while disabling inspection.

If you wrap the inspector with #if DEBUG this might be fine for you.

However, if you'd plan to ship a product that includes the inspector, or otherwise want to lazily construct it to be more efficient, use a unique_ptr instead and set the onClose callback to reset the pointer.

// PluginEditor.h
std::unique_ptr<melatonin::Inspector> inspector;

// in some button on-click logic
// replace `this` with a reference to your editor if necessary
if (!inspector)
{
    inspector = std::make_unique<melatonin::Inspector> (*this);
    inspector->onClose = [this]() { inspector.reset(); };
}

inspector->setVisible (true);

Thanks to @FigBug for this feature.

6. Optional: Setup component timing

Just #include modules/melatonin_inspector/melatonin/helpers/timing.h and then call the RAII helper at the top of a component's paint method:

void paint (juce::Graphics& g) override
{
    melatonin::ComponentTimer timer { this };

    // do all your expensive painting...

This simply times the method and stores it in the component's own properties. It will store up to 3 values named timing1, timing2, timing3.

Want automatic timings for every JUCE component, including stock widgets? Upvote this FR.

Want timings for your custom components right now? Do what I do and derive all your components from a juce::Component subclass which wraps the paint call and adds the helper before paint is called.

Check out the forum post for detail. Or, if you run a JUCE fork, you might prefer Roland's solution.

FAQ

Can I use this in a GUI app/standalone?

Yup! See the tests folder for an example.

Can I save my component resizes or edits?

Nope!

For that, one would need a component system relying on data for placement and size vs. code. See Daniel's Foley GUI Magic.

How is the component hierarchy created?

It traverses components from the root, building a TreeView.

In the special case of TabbedComponent, each tab is added as a child.

My FPS seems low, is it accurate?

It's a smoothed running average.

If you see low FPS rates, check the following:

  • Are you running just your plugin? Make sure other plugin UIs are closed.
  • I optimize the inspector for Debug usage, but it can be tough for complex UIs to hit 60fps in Debug, especially on macOS (see note below). See what happens in Release.
  • You might have legitimately expensive paint calls (esp. in Debug). You can verify this out via Perfetto.

On recent macOS, a repaint() on even small sections of a window (ie, what the FPS meter does) will cause the OS to paint the entire plugin window. You can use Flash Screen Updates in Quartz Debug to verify this. Because of this macOS behavior, the FPS meter will actually trigger full repaints of your UI, so anything expensive (especially in Debug) will slow down what the FPS meter reports.

If you are using the JUCE flag JUCE_COREGRAPHICS_RENDER_WITH_MULTIPLE_PAINT_CALLS, JUCE will internally manage the rectangles that need to be repainted, with the aim of being more precise/hygenic with what actually gets painted. This might be a good choice if your plugin already frequently repainting parts of the UI. But please don't switch over to that flag just to appease the FPS meter! It needs to be a choice you make depending on your internal testing (without the FPS meter in play).

Feel free to ask for other ideas in the forum thread.

I have a problem / feature request

Please do submit an Issue or PR on the repository! Or visit the official thread on the JUCE forum.

Contributing

Contributions are always welcome!

The inspector wouldn't be half as awesome without the help of the community.

If you'd like to contribute, look out for the issues tagged with "Good First Issue"

Note that CI tests for compilation and treats errors on both macOS and Windows as errors.

Assets

All assets are PNG exported at 2x.

Please see the CMakelists.txt file for details on how to add icons in a Projucer friendly way. There's a script for it!

About

A JUCE module that gives you the ability to inspect and visually edit (non-destructively) components in your UI.

License:MIT License


Languages

Language:C++ 98.2%Language:Ruby 1.0%Language:CMake 0.8%