Jump to content

GSoC/2021/StatusReports/TanmayChavan

From KDE Community Wiki

Krita - Smarter boolean operations on vector shapes

In Krita, performing boolean operations on vector shapes leads to a large number of unnecessary nodes. This happens because Qt lacks a proper algorithm to find the intersection point of two Bezier curves. I plan on implementing a numerically stable as well as efficient algorithm to find the intersections of two Bezier curves. Then I will implement a clipping algorithm required for carrying out the boolean operations.

Mentors

  • Dmitry Kazakov
  • Iván Santa María

Goals

  • Create a new algorithm to compute intersections of Bezier curves using implicitization
    • done!
  • Manage the dependencies and implement parts of Qt private modules in Krita codebase
    • done!
  • Integrate the algorithm with current intersection finding routine
    • done!
  • Implement the new routine for boolean operations
    • not fully done
  • Write proper documentation and unit tests for all the above goals while doing them
    • ongoing

Status Report

Algorithm for Bezier curve intersection

  • This was the exact feature which Qt lacked for boolean operations. As a result, it flattened the curves which in turn made too many nodes. I have implemented an algorithm to compute the intersection points using the implicit representation of a cubic Bezier curve. I have finished implementing curve-curve, line-curve intersections by now. The algorithm works well even with double precision. But, the curve-curve intersections can get really costly. On an average, a single curve-curve intersection-finding routine requires ~280 microseconds. However, it is reported this is the fastest method for curves with degree 5 or less. The line-line intersection point is obtained in the same manner as Qt did.
curve-curve intersection
curve-curve intersection
Line-curve intersection
Line-curve intersection

Qt private dependencies

  • As the set operations were handled by the class QPathClipper which lies behind the Qt API, I had to manage the dependencies so that the new implementation would not rely on Qt private modules. This was relatively easy as QPathClipper was fairly independent and it's dependencies didn't have too many dependencies. I also managed to get a downstream copy of the QPainterPath class, albeit had to skip some parts (the PathStroker related part) as it was too closely linked to other Qt private modules. However, it was not required for the purpose of the current tasks. So, I have managed to get the project to run without any private or additional dependencies.

Intersection finding routine

  • For this, I am using a downstream copy of QPainterPath. The elements of QPainterPaths are extracted and stored in the local version of PainterPath. After this, two vectors for subject shape and clip shape are created, where elements are added with the classes CubicBezier or Line as per the type of elements. Then, it proceeds to find the intersection points between the elements of the two vectors. These are stored in a separate class member with some additional information which could be helpful to perform set operations.

Shape-clipping algorithm

The clipped shapes, without any boolean operations and all edges kept. All intersection points are found and the shapes are clipped correctly.
The clipped shapes, without any boolean operations and all edges kept. All intersection points are found and the shapes are clipped correctly.

Clipping and the following selection of edges proved to be a challenging task. Most of the problems arose as a suitable implementation was not available online, at least in FOSS. Most of them relied on flattening of the curves, much like Qt's algorithm. The wiki page mentioned Greiner-Hormann algorithm as one possibility for handling clipping for shapes with curves. However, there was no implementation available online. There were also several degenerate cases in Greiner-Hormann. It would fail if the two shapes had common vertices. It will also fail for common/overlapping edges, and for points of intersection lying on the vertices. The solution suggested was to change the co-ordinates of such problematic points. However this might lead to unexpected and undesired results. There is an improved version of the algorithm explained in this paper, but the preprocessing routines were complicated and still had a few drawbacks. If there were a point of intersection coinciding with a point of self-intersection, the shape should be divided into several smaller non-intersecting shapes. Another major issue was that we needed to know at least one vertex of one shape lying outside the other shape. This is implemented in Qt, but works only for lines. (Currently, this can be implemented by using the tight bounding box function, and using an even-odd approach instead of Qt's winding-number approach.)

Apart from this, there was always an uncertainty as it had not been done before. There could be several problems there which remain unknown till it is implemented. For this reason, I chose not to continue with that approach. However, I am writing down everything that has been done, what remains to be done, and how to do it should one choose to follow this approach.

For the latter approaches, the algorithm to split the shapes is crucial. Basically, until now, all of the intersection points between the two shapes can be found. The next task would be to process these shapes and select the edges which should be there in the final shape. Here, the splitting comes in play. We split all the elements of the shape about all of the intersection points. Thus, we get a shape which contains all of the segments in such a way that they need not be split again. Essentially, the final shape will consist of curves and lines exactly as in the splitted shapes, as they can't be split any further. This is half of the clipping job done; we have successfully marked the intersections and have generated a structure to readily traverse. The only job remaining is to choose which segments (lines or curves) should make it to the final shape.

This process makes it feasible to use Qt's algorithm for clipping it. Qt flattens the curves to process it, but preserves the end points. Since all of the intersection points are converted to end-points, they will remain as they are, without any loss in accuracy. This also makes it possible for us to resubstitute the curves after we flatten them, to obtain the original curves. Also, this covers two out of three steps required for Greiner-Hormann: finding all points of intersection and then processing the vertices while marking them correctly. After finding out at least one point lying outside the other shape, the only thing remaining is to select the segments to be added to the final shape.

My second approach was to carry out the process by modifying Qt's original algorithm. The basic idea is to mark the segments while flattening the curves, and then to restore the curves again when the WingedEdge structure is converted back to QPainterPath. The code is implemented. However, I was not able to fully finish it. The original code was complex, and was largely undocumented. I have implemented the marking and substitution process. However, there seems to be a bug introduced which causes the shapes to be improperly clipped. the fault mostly lies in the functions QWingedEdge::doClip and QWingedEdge::handleCrossingEdges. Debugging messages reveal the splitted edges are received correctly by doClip, but it decides not to select them. It could also lie in the `traverse` function, as it decides upto which edge should be considered. However, I was not able to debug it properly, due to limited time at hand. Given some more time, I'm pretty sure I'll be able to get it to work. However, I have documented Qt's code to the best of my capacity, and I hope it will quicken the process of understanding it's algorithm. The algorithm needs some time to be understood, and once understood, it can be corrected easily. Due to the time constraint, I stopped debugging it, and went to the third - and current - approach; to explicitly substitute the curves.

The final and current approach is to carry out the substitution routine outside Qt's clipping algorithm. After finding all points of intersection, we process the shapes to convert all points of intersection to regular vertices (in the function KisIntersectionFinder::processShapes.) Then we record all the clipped curves with the help of the class CubicBezierData. Then, we clip the paths by Qt's own algorithm, without modifying it. Then, we traverse the final QPainterPath, and find the segments which are supposed to represent the flattened curve. As discussed above, the end points remain the same before and after flattening. Thus, we can find out a chain of continuous line segments (or lineTo elements) and if the end points of this chain match the end-points of one of the curves we have recorded, we replace the chain with the curve. This algorithm is the one being currently employed. It works for the union, intersection, subtraction operations for most of the cases.

Links

Completed Tasks

  • I have managed to write the code to find points of intersection of curves with curves, lines with curves, and integrate it with line-line intersection finding routine.
  • I have managed all the private Qt dependencies needed for the project by creating copies of private QT programs in the Krita codebase, and integrated them successfully.
  • I have written the classes needed to handle intersections of two QPainterPaths and then process them by splitting them about their intersection points.
  • I have written documentation for the programs I have written and added some for the copied Qt files. I have also written unit tests for my code.

Task partially done

I have not implemented the algorithm of selecting edges for boolean operations properly yet. The current algorithm (in KisIntersectionFinder::resubstituteCurves()) is able to produce valid results for some shapes, for union, intersection, and difference operation. Apart from the basic shapes, it also works for all of the tools (freehand curves, Bezier tool).

message box similar to the one in bug report, before
message box similar to the one in bug report, before
message box similar to the one in bug report, after
message box similar to the one in bug report, after


Operations with multiple shapes also work.

multiple shape operation, before
multiple shape operation, before
multiple shape operation, after
multiple shape operation, after


The freehand path tool also works.

freehand tool, before
freehand tool, before
freehand tool, after, union
freehand tool, after, union
freehand tool, after, intersection
freehand tool, after, intersection


The subtraction tool conditionally works.

polyline tool operation, before
polyline tool operation, before
polyline tool operation, after, union
polyline tool operation, after, union
polyline tool operation, after, subtraction/difference
polyline tool operation, after, subtraction/difference

However, it has bugs. It is not perfectly commutative. One order of selection will produce incorrect results, but a different order will produce the correct results. This could lead to missing edges or some extra lines. There are some issues while clipping curves with polygons drawn using polyline tool. However, as these issues do not occur with rectangle tool, these are most certainly bugs.

incorrect clipping, before
incorrect clipping, before
wrong clipping
wrong clipping

But, if we re-select the shapes in a different order, the clipping works as expected.

correct clipping due to different order of selection
correct clipping due to different order of selection


However, as most of the operations do work one way or the other, I guess the algorithm is sound. With some extra time and effort it will produce the correct results. However, given more time, one of the two aforementioned approaches will be a better fit for our purposes as they tend to be cleaner without so many conditions and need for treating the special cases differently.

Future Plan

Although I tried my best to finish it, and managed to get a working demo, the project still is not fully complete. Everything uptill clipping has been done. The only part remaining is to either debug the current resubstitution algorithm and iron out the remaining kinks, or to implement a new cleaner way to select the edges for the final shape. This can be achieved by any of the two above approaches.

Since I spent a lot of time trying to understand each approach, I have got a pretty good idea of how to implement it using any of the two methods. I will keep on working on this project, but am writing this down for easy reference.

  • Greiner-Hormann is the cleaner way, as it can be implemented with significantly less conditional codes. The wikipedia page is a good place to start. After that, the the original paper explains the basic procedure. However, there are several degenerate cases with this original algorithm. The improved algorithm covers most of these issues by using additional flags and some extra preprocessing. It will still fail if the point of intersection of two shapes overlaps with a point of self-intersection. This can be tackled by breaking down the original shape in non intersecting shapes (however we need to make sure the shape is closed.) We could also slightly move the point in this case, as it is a really rare case.
    The arrangement of points and splitting is already implemented. the classes GreinerClippingVertex and GreinerHormannClipping contain the structure needed to implement Greiner-Hormann. The only thing remaining as per the algorithm is to parse the shapes to select the final edges as per the boolean operation. For that, we need to know at least one point in each shape which lies outside the other. This can be done by using the even-odd rule along with the line-curve intersection present in KisIntersectionFinder::intersectionPoint(). Qt's method uses the winding-number approach, and is more robust, but requires flattening. It is present in QPainterPath::contains()


  • We can also use Qt's default clipping algorithm. The working is mostly as explained in the above sections, but I will update some notes for more clarification if needed. The code has already been ported in KisPathClipper.cpp, and it's private dependencies are also managed and kept in defaulttool/3rdparty. The flattening of curves happens in KisPathSegments::addPath(). Here we can mark the segments which were originally curves. Then KisWingedEdge::intersectAndAdd() handles intersections, but since we have already handled that, it only converts the segments to edges. Then, KisPathClipper::doClip() handles the clipping, where KisPathClipper::handleCrossingEdges() does the main job. We resubstitute the curves in the function "add" called after KisWingedEdge::toPath(). There is a bug over here preventing it's correct operation. finding and solving it will lead to a fully working clipping algorithm without any regressions.

Current Status


I'm currently trying to debug on the clipping algorithm, and write better documentation and improve the coding style. The project runs, and can perform the required operations for many cases. However it is still buggy, and needs some more work. This project has been an extremely awesome experience, and I learned a lot. Due to the lack of prior implemented softwares doing the task, I had to implement stuff to know if it would work or not. It took more time than I had initially anticipated. I have tried my best to document my design choices and my decisions to make it easy for anyone to understand it and the nature of the task. I will try my best to finish the project by implementing a more stable way to handle the boolean operation selection, and would absolutely love to keep on working afterwards with Krita as well.