Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

New "smooth" and "orthogonal" drawing tools #198

Merged
merged 11 commits into from
Mar 22, 2024

Conversation

pascal-niklaus
Copy link
Contributor

@pascal-niklaus pascal-niklaus commented Mar 13, 2024

I have implemented two new drawing tools.

  • smoothing: this tool allows smoothing of the often shaky hand-drawn lines.

  • orthogonal lines: this tool "snaps" to approximately horizontal and vertical direction and "fixes" the sections in between so that a continuous path results. Straight sections are optionally joined with a curved line with a certain radius. This tool is great to draw e.g. arrows connecting things.

Here is a demo:

new_tools-2500-500

Both tools in principle have adjustable parameter:

In both tools, the first step is a path simplification (fewer points) using the Douglas-Peucker algorithm. Here, the degree of "simplification" is set by a parameter indicating how many pixels a point that is removed can be away from the simplified curve.

Smoothing happens with centripetal Catmull-Rom-splines. Here, a parameter indicates how many intermediate points are inserted into the path to create the "curves". I currently have set this to 5, and I don't see much need to vary it.

For the orthogonal pen, an angular tolerance can be set within which the lines "snap" to perfectly horizontal or vertical direction, once a length threshold is exceeded. This guarantees that curves are preserved, even if a short section is H or V. The second option is the radius at which sections are joined. Segments have to be longer than twice this radius, otherwise no arc is inserted to join them to their neighbors.

I think some of the parameter should be settable by the user, because they (1) depend on the screen resolution (e.g. radii), and (2) the desired balance between "simplification" (allowing very fast drawing yet keeping a nice result) and preservation of details.

Before adding the tidied-up code to this PR, I would like to discuss the syntax for the config file. Many options seem possible, for example:

SMART_PEN (mode="orthogonal" simplify=5 radius=20 angular_tolerance=15 min_length=50)
SMART_PEN (mode="smooth" simplify=5)

Or something like:

SMOOTH_PEN(simplify=5 radius=20)
ORTHOGONAL_PEN(simplify=5, angular_tolerance=15 radius=20 min_length=50)

Or even some compact (but unintuitive) form like:

ORTHOGONAL_PEN(simplify=5, params="5,15,20,50")

Of course, all other PEN options remain available (arrows at the ends, for example).

Both tools together are about 800 lines of code (including functions for debugging that I will remove), most of which is in an additional file smooth.c. There are very few changes in config.c and callbacks.c.

I really would not like to separate the tools into 2 PRs because this would be a lot of work and a lot of the code is shared.

@bk138
Copy link
Owner

bk138 commented Mar 13, 2024

Nice! Will review ASAP, but might take a few days. As usual.

Edit: @pascal-niklaus the commit seems empty.

@bk138 bk138 self-assigned this Mar 13, 2024
@bk138 bk138 added the tool label Mar 13, 2024
@pascal-niklaus
Copy link
Contributor Author

Nice! Will review ASAP, but might take a few days. As usual.

Edit: @pascal-niklaus the commit seems empty.

Yes, that's right.

It for the moment is a placeholder because I would like to first agree on the syntax of the tool options.

All the code exists and is functional, but the params are currently hardcoded in callback.c. I simply would like to avoid changing parser options several times if that can easily be avoided.

@bk138
Copy link
Owner

bk138 commented Mar 14, 2024

OK, I'll answer ASAP, currently on the move.

@pascal-niklaus
Copy link
Contributor Author

pascal-niklaus commented Mar 14, 2024

OK, I'll answer ASAP, currently on the move.

I pushed the current code for you to get an idea about it and to try it out if you want.

To play with the two tools, just change if(0) to if(1) in callbacks.c. As said, the params are hard-coded for the moment:

  if (type == GROMIT_SMOOTH)
    {
      if (0) {
          // smooth pen
          GList *coords = devdata->coordlist;
          gboolean joined = snap_ends(coords, 20);
          douglas_peucker(coords, 10);  // was 15
          add_points(coords, 200);
          devdata->coordlist = catmull_rom(coords, 5, joined);
      } else {
          // smart rect pen
          douglas_peucker(devdata->coordlist, 15);  // was 15
          orthogonalize(devdata->coordlist, 20, 40);
          round_corners(devdata->coordlist, 30, 6);
      }

The code in smooth.c also is in a more compact form than the rest of the code, which I find easier to read, but I can change this of course for consistency.

@pascal-niklaus pascal-niklaus force-pushed the devel_smooth_orthogonal branch from b607e9f to 4d58439 Compare March 15, 2024 07:17
@pascal-niklaus pascal-niklaus force-pushed the devel_smooth_orthogonal branch from 4d58439 to 459868b Compare March 15, 2024 07:59
@pascal-niklaus
Copy link
Contributor Author

pascal-niklaus commented Mar 15, 2024

Additional feature added: Smoothed and orthogonal curves can now be circular, i.e. the start and end points can "snap" together when they are close enough. The smoothing is then done in a "circular" fashion, and the same is true for the rounded corners.

@pascal-niklaus
Copy link
Contributor Author

I went ahead and made the tools configurable. For example, like this:

"smooth Pen" = SMOOTH (size=7 color="red" simplify=10 snap=30);
"ortho Pen" = ORTHOGONAL (size=5 color="green" simplify=15 radius=20 minlen=50 maxangle=25 snap=40);
"default" = "smooth Pen";
"default"[SHIFT] = "ortho Pen";

Here, simplify is the argument specifying within which range points are removed in the Douglas-Peucker step.

For the orthogonal pen, minlen is the minimum length of a path segment, radius the curvature radius at the segment linkages, maxangle the maximum angular deviation from perfect H and V so that the line still snaps to H and V, and snap the distance between start and end within which a closed path is produced (the ends "snap" together).

Let me know whether that syntax makes sense...

Copy link
Owner

@bk138 bk138 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

First review round done!

src/config.c Outdated Show resolved Hide resolved
src/main.c Show resolved Hide resolved
src/smooth.h Outdated Show resolved Hide resolved
src/smooth_priv.h Outdated Show resolved Hide resolved
src/smooth.h Outdated Show resolved Hide resolved
@bk138 bk138 assigned pascal-niklaus and unassigned bk138 Mar 21, 2024
@bk138
Copy link
Owner

bk138 commented Mar 21, 2024

@pascal-niklaus looks good to me code-wise now, thanks for the good work! One more thing: can you please document the new tools in the README?

@bk138 bk138 merged commit 19a5569 into bk138:master Mar 22, 2024
1 check passed
@bk138
Copy link
Owner

bk138 commented Mar 22, 2024

@pascal-niklaus Thanks for the good work again! One more idea: why not add one demo gif to each tool in the README if its not too much work? The ones you created look pretty cool.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants