diff --git a/.clang-tidy b/.clang-tidy index e516426d03..6ccf3b98db 100644 --- a/.clang-tidy +++ b/.clang-tidy @@ -12,7 +12,8 @@ Checks: > -fuchsia-overloaded-operator, -cppcoreguidelines-avoid-capturing-lambda-coroutines, -llvm-header-guard, - -bugprone-easily-swappable-parameters + -bugprone-easily-swappable-parameters, + -*-default-arguments-declarations WarningsAsErrors: '-*' HeaderFilterRegex: '' FormatStyle: none @@ -52,4 +53,4 @@ CheckOptions: - key: readability-identifier-length.IgnoredVariableNames value: '_p|p0|p1|i|j|k|x|X|y|Y|z|Z|a|A|b|B|c|C|d|D|ab|AB|ba|BA|bc|BC|cb|CB|cd|CD|dc|DC|ad|AD|da|DA|ip|os' - key: readability-identifier-length.IgnoredLoopCounterNames - value: '_p|p0|p1|i|j|k|x|X|y|Y|z|Z|a|A|b|B|c|C|d|D|ab|AB|ba|BA|bc|BC|cb|CB|cd|CD|dc|DC|ad|AD|da|DA|ip|os' \ No newline at end of file + value: '_p|p0|p1|i|j|k|x|X|y|Y|z|Z|a|A|b|B|c|C|d|D|ab|AB|ba|BA|bc|BC|cb|CB|cd|CD|dc|DC|ad|AD|da|DA|ip|os' diff --git a/.github/workflows/conan-package.yml b/.github/workflows/conan-package.yml index 0acc07d220..eb5829a6a7 100644 --- a/.github/workflows/conan-package.yml +++ b/.github/workflows/conan-package.yml @@ -20,6 +20,23 @@ on: tags: - '[0-9]+.[0-9]+.[0-9]*' - '[0-9]+.[0-9]+.[0-9]' + pull_request: + types: [opened, reopened, synchronize] + paths: + - 'include/**' + - 'src/**' + - 'test_package/**' + - 'conanfile.py' + - 'conandata.yml' + - 'CMakeLists.txt' + - '.github/workflows/conan-package.yml' + branches: + - main + - 'CURA-*' + - 'PP-*' + - 'NP-*' + - '[0-9].[0-9]*' + - '[0-9].[0-9][0-9]*' jobs: conan-recipe-version: @@ -37,7 +54,7 @@ jobs: conan-package-create-macos: needs: [ conan-recipe-version, conan-package-export ] - if: ${{ (github.event_name == 'push' && (github.ref_name == 'main' || github.ref_name == 'master' || needs.conan-recipe-version.outputs.is_release_branch == 'true')) }} + if: ${{ ((github.event_name == 'push' && (github.ref_name == 'main' || needs.conan-recipe-version.outputs.is_release_branch == 'true')) || github.event_name == 'pull_request') }} uses: ultimaker/cura-workflows/.github/workflows/conan-package-create-macos.yml@main with: recipe_id_full: ${{ needs.conan-recipe-version.outputs.recipe_id_full }} @@ -45,7 +62,7 @@ jobs: conan-package-create-windows: needs: [ conan-recipe-version, conan-package-export ] - if: ${{ (github.event_name == 'push' && (github.ref_name == 'main' || github.ref_name == 'master' || needs.conan-recipe-version.outputs.is_release_branch == 'true')) }} + if: ${{ ((github.event_name == 'push' && (github.ref_name == 'main' || needs.conan-recipe-version.outputs.is_release_branch == 'true')) || github.event_name == 'pull_request') }} uses: ultimaker/cura-workflows/.github/workflows/conan-package-create-windows.yml@main with: recipe_id_full: ${{ needs.conan-recipe-version.outputs.recipe_id_full }} @@ -53,8 +70,16 @@ jobs: conan-package-create-linux: needs: [ conan-recipe-version, conan-package-export ] - if: ${{ (github.event_name == 'push' && (github.ref_name == 'main' || github.ref_name == 'master' || needs.conan-recipe-version.outputs.is_release_branch == 'true')) }} + if: ${{ ((github.event_name == 'push' && (github.ref_name == 'main' || needs.conan-recipe-version.outputs.is_release_branch == 'true')) || github.event_name == 'pull_request') }} uses: ultimaker/cura-workflows/.github/workflows/conan-package-create-linux.yml@main with: recipe_id_full: ${{ needs.conan-recipe-version.outputs.recipe_id_full }} secrets: inherit + + conan-package-create-wasm: + needs: [ conan-recipe-version, conan-package-export ] + if: ${{ (github.event_name == 'push' && (github.ref_name == 'main' || needs.conan-recipe-version.outputs.is_release_branch == 'true')) }} + uses: ultimaker/cura-workflows/.github/workflows/conan-package-create-wasm.yml@main + with: + recipe_id_full: ${{ needs.conan-recipe-version.outputs.recipe_id_full }} + secrets: inherit diff --git a/.github/workflows/gcodeanalyzer.yml b/.github/workflows/gcodeanalyzer.yml index 29c83817de..0b85fb75aa 100644 --- a/.github/workflows/gcodeanalyzer.yml +++ b/.github/workflows/gcodeanalyzer.yml @@ -81,28 +81,6 @@ jobs: fetch-depth: 1 token: ${{ secrets.GITHUB_TOKEN }} - - name: Determine the corresponding Cura branch - id: curabranch - run: | - status_code=$(curl -s -o /dev/null -w "%{http_code}" "https://api.github.com/repos/ultimaker/cura/branches/${{ github.head_ref }}") - if [ "$status_code" -eq 200 ]; then - echo "The branch exists in Cura" - echo "branch=${{ github.head_ref }}" >> $GITHUB_OUTPUT - else - echo "branch=main" >> $GITHUB_OUTPUT - fi - - - name: Checkout Cura - uses: actions/checkout@v3 - with: - repository: 'Ultimaker/Cura' - ref: ${{ steps.curabranch.outputs.branch}} - path: 'Cura' - fetch-depth: 1 - sparse-checkout: | - resources/definitions - resources/extruders - - name: Sync pip requirements run: curl -O https://raw.githubusercontent.com/Ultimaker/cura-workflows/main/.github/workflows/requirements-runner.txt working-directory: CuraEngine/.github/workflows @@ -159,7 +137,7 @@ jobs: ${{ runner.os }}-conan-downloads- - name: Install dependencies - run: conan install . ${{ needs.conan-recipe-version.outputs.recipe_id_full }} -s build_type=Release --build=missing --update -g GitHubActionsRunEnv -g GitHubActionsBuildEnv -c tools.build:skip_test=True + run: conan install . ${{ needs.conan-recipe-version.outputs.recipe_id_full }} -s build_type=Release --build=missing --update -g GitHubActionsRunEnv -g GitHubActionsBuildEnv -c tools.build:skip_test=True -o with_cura_resources=True working-directory: CuraEngine - name: Set Environment variables from Conan install (bash) @@ -167,7 +145,6 @@ jobs: run: | . ./activate_github_actions_runenv.sh . ./activate_github_actions_buildenv.sh - echo "CURA_ENGINE_SEARCH_PATH=$GITHUB_WORKSPACE/Cura/resources/definitions:$GITHUB_WORKSPACE/Cura/resources/extruders" >> $GITHUB_ENV working-directory: CuraEngine/build/Release/generators - name: Build CuraEngine and tests @@ -180,7 +157,7 @@ jobs: run: | for file in `ls ../NightlyTestModels/*.stl`; do - ( time ./build/Release/CuraEngine slice --force-read-parent --force-read-nondefault -v -p -j ../Cura/resources/definitions/ultimaker_s3.def.json -l $file -o ../`basename $file .stl`.gcode ) 2> ../`basename $file .stl`.time + ( time ./build/Release/CuraEngine slice --force-read-parent --force-read-nondefault -v -p -j $CURA_RESOURCES/definitions/ultimaker_s3.def.json -l $file -o ../`basename $file .stl`.gcode ) 2> ../`basename $file .stl`.time done working-directory: CuraEngine diff --git a/.github/workflows/lint-poster.yml b/.github/workflows/lint-poster.yml index a016599fec..3ea00c6572 100644 --- a/.github/workflows/lint-poster.yml +++ b/.github/workflows/lint-poster.yml @@ -10,72 +10,88 @@ jobs: # Trigger the job only if the previous (insecure) workflow completed successfully if: ${{ github.event.workflow_run.conclusion == 'success' }} runs-on: ubuntu-latest + permissions: + pull-requests: write steps: - name: Download analysis results - uses: actions/github-script@v3.1.0 + uses: actions/github-script@v7 with: script: | - let artifacts = await github.actions.listWorkflowRunArtifacts({ + const artifacts = await github.rest.actions.listWorkflowRunArtifacts({ owner: context.repo.owner, repo: context.repo.repo, run_id: ${{github.event.workflow_run.id }}, }); - let matchArtifact = artifacts.data.artifacts.filter((artifact) => { + const matchArtifact = artifacts.data.artifacts.filter((artifact) => { return artifact.name == "linter-result" })[0]; - let download = await github.actions.downloadArtifact({ + const download = await github.rest.actions.downloadArtifact({ owner: context.repo.owner, repo: context.repo.repo, artifact_id: matchArtifact.id, archive_format: "zip", }); - let fs = require("fs"); + const fs = require("fs"); fs.writeFileSync("${{github.workspace}}/linter-result.zip", Buffer.from(download.data)); - - name: Set environment variables + - name: Extract analysis results run: | mkdir linter-result - unzip linter-result.zip -d linter-result - echo "pr_id=$(cat linter-result/pr-id.txt)" >> $GITHUB_ENV - echo "pr_head_repo=$(cat linter-result/pr-head-repo.txt)" >> $GITHUB_ENV - echo "pr_head_ref=$(cat linter-result/pr-head-ref.txt)" >> $GITHUB_ENV + unzip -j linter-result.zip -d linter-result + + - name: Set PR details environment variables + uses: actions/github-script@v7 + with: + script: | + const assert = require("node:assert").strict; + const fs = require("fs"); + function exportVar(varName, fileName, regEx) { + const val = fs.readFileSync("${{ github.workspace }}/linter-result/" + fileName, { + encoding: "ascii" + }).trimEnd(); + assert.ok(regEx.test(val), "Invalid value format for " + varName); + core.exportVariable(varName, val); + } + exportVar("PR_ID", "pr-id.txt", /^[0-9]+$/); + exportVar("PR_HEAD_REPO", "pr-head-repo.txt", /^[-./0-9A-Z_a-z]+$/); + exportVar("PR_HEAD_SHA", "pr-head-sha.txt", /^[0-9A-Fa-f]+$/); - - uses: actions/checkout@v3 + - uses: actions/checkout@v4 with: - repository: ${{ env.pr_head_repo }} - ref: ${{ env.pr_head_ref }} + repository: ${{ env.PR_HEAD_REPO }} + ref: ${{ env.PR_HEAD_SHA }} persist-credentials: false - name: Redownload analysis results - uses: actions/github-script@v3.1.0 + uses: actions/github-script@v7 with: script: | - let artifacts = await github.actions.listWorkflowRunArtifacts({ + const artifacts = await github.rest.actions.listWorkflowRunArtifacts({ owner: context.repo.owner, repo: context.repo.repo, run_id: ${{github.event.workflow_run.id }}, }); - let matchArtifact = artifacts.data.artifacts.filter((artifact) => { + const matchArtifact = artifacts.data.artifacts.filter((artifact) => { return artifact.name == "linter-result" })[0]; - let download = await github.actions.downloadArtifact({ + const download = await github.rest.actions.downloadArtifact({ owner: context.repo.owner, repo: context.repo.repo, artifact_id: matchArtifact.id, archive_format: "zip", }); - let fs = require("fs"); + const fs = require("fs"); fs.writeFileSync("${{github.workspace}}/linter-result.zip", Buffer.from(download.data)); - name: Extract analysis results run: | mkdir linter-result - unzip linter-result.zip -d linter-result + unzip -j linter-result.zip -d linter-result - name: Run clang-tidy-pr-comments action - uses: platisd/clang-tidy-pr-comments + uses: platisd/clang-tidy-pr-comments@v1 with: github_token: ${{ secrets.GITHUB_TOKEN }} clang_tidy_fixes: linter-result/fixes.yml - pull_request_id: ${{ env.pr_id }} + pull_request_id: ${{ env.PR_ID }} request_changes: true diff --git a/.github/workflows/lint-tidier.yml b/.github/workflows/lint-tidier.yml index d178f52f3a..a4bdf1b9d6 100644 --- a/.github/workflows/lint-tidier.yml +++ b/.github/workflows/lint-tidier.yml @@ -91,7 +91,7 @@ jobs: run: | echo ${{ github.event.number }} > linter-result/pr-id.txt echo ${{ github.event.pull_request.head.repo.full_name }} > linter-result/pr-head-repo.txt - echo ${{ github.event.pull_request.head.ref }} > linter-result/pr-head-ref.txt + echo ${{ github.event.pull_request.head.sha }} > linter-result/pr-head-sha.txt - uses: actions/upload-artifact@v2 with: diff --git a/CMakeLists.txt b/CMakeLists.txt index 210d8b5c57..3d3cb270ed 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -62,7 +62,7 @@ set(engine_SRCS # Except main.cpp. src/MeshGroup.cpp src/Mold.cpp src/multiVolumes.cpp - src/PathOrderPath.cpp + src/path_ordering.cpp src/Preheat.cpp src/PrimeTower.cpp src/raft.cpp @@ -135,7 +135,7 @@ set(engine_SRCS # Except main.cpp. src/utils/ExtrusionLine.cpp src/utils/ExtrusionSegment.cpp src/utils/gettime.cpp - src/utils/LinearAlg2D.cpp + src/utils/linearAlg2D.cpp src/utils/ListPolyIt.cpp src/utils/Matrix4x3D.cpp src/utils/MinimumSpanningTree.cpp @@ -144,7 +144,6 @@ set(engine_SRCS # Except main.cpp. src/utils/PolygonsPointIndex.cpp src/utils/PolygonsSegmentIndex.cpp src/utils/polygonUtils.cpp - src/utils/polygon.cpp src/utils/PolylineStitcher.cpp src/utils/Simplify.cpp src/utils/SVG.cpp @@ -153,6 +152,17 @@ set(engine_SRCS # Except main.cpp. src/utils/ToolpathVisualizer.cpp src/utils/VoronoiUtils.cpp src/utils/VoxelUtils.cpp + src/utils/MixedPolylineStitcher.cpp + + src/geometry/Polygon.cpp + src/geometry/Shape.cpp + src/geometry/PointsSet.cpp + src/geometry/SingleShape.cpp + src/geometry/PartsView.cpp + src/geometry/LinesSet.cpp + src/geometry/Polyline.cpp + src/geometry/ClosedPolyline.cpp + src/geometry/MixedLinesSet.cpp ) add_library(_CuraEngine STATIC ${engine_SRCS} ${engine_PB_SRCS}) diff --git a/benchmark/infill_benchmark.h b/benchmark/infill_benchmark.h index 459174eecc..29423a1b70 100644 --- a/benchmark/infill_benchmark.h +++ b/benchmark/infill_benchmark.h @@ -1,4 +1,4 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #ifndef CURAENGINE_INFILL_BENCHMARK_H @@ -6,6 +6,9 @@ #include +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/LinesSet.h" #include "infill.h" namespace cura @@ -14,14 +17,14 @@ class InfillTest : public benchmark::Fixture { public: Settings settings{}; - Polygons square_shape; - Polygons ff_holes; + Shape square_shape; + Shape ff_holes; std::vector all_paths; - Polygons outline_polygons; + Shape outline_polygons; EFillMethod pattern{ EFillMethod::LINES }; bool zig_zagify{ true }; bool connect_polygons{ true }; @@ -58,8 +61,8 @@ class InfillTest : public benchmark::Fixture ff_holes.back().emplace_back(MM2INT(60), MM2INT(40)); ff_holes.back().emplace_back(MM2INT(90), MM2INT(25)); - outline_polygons.add(square_shape); - outline_polygons.add(ff_holes); + outline_polygons.push_back(square_shape); + outline_polygons.push_back(ff_holes); settings.add("fill_outline_gaps", "false"); settings.add("meshfix_maximum_deviation", "0.1"); @@ -93,29 +96,30 @@ class InfillTest : public benchmark::Fixture BENCHMARK_DEFINE_F(InfillTest, Infill_generate_connect)(benchmark::State& st) { - Infill infill(pattern, - zig_zagify, - connect_polygons, - outline_polygons, - INFILL_LINE_WIDTH, - line_distance, - INFILL_OVERLAP, - INFILL_MULTIPLIER, - FILL_ANGLE, - Z, - SHIFT, - MAX_RESOLUTION, - MAX_DEVIATION); // There are some optional parameters, but these will do for now (future improvement?). + Infill infill( + pattern, + zig_zagify, + connect_polygons, + outline_polygons, + INFILL_LINE_WIDTH, + line_distance, + INFILL_OVERLAP, + INFILL_MULTIPLIER, + FILL_ANGLE, + Z, + SHIFT, + MAX_RESOLUTION, + MAX_DEVIATION); // There are some optional parameters, but these will do for now (future improvement?). for (auto _ : st) { std::vector result_paths; - Polygons result_polygons; - Polygons result_lines; + Shape result_polygons; + OpenLinesSet result_lines; infill.generate(result_paths, result_polygons, result_lines, settings, 0, SectionType::INFILL, nullptr, nullptr); } } -BENCHMARK_REGISTER_F(InfillTest, Infill_generate_connect)->ArgsProduct({{true, false}, {400, 800, 1200}})->Unit(benchmark::kMillisecond); +BENCHMARK_REGISTER_F(InfillTest, Infill_generate_connect)->ArgsProduct({ { true, false }, { 400, 800, 1200 } })->Unit(benchmark::kMillisecond); } // namespace cura #endif // CURAENGINE_INFILL_BENCHMARK_H diff --git a/benchmark/simplify_benchmark.h b/benchmark/simplify_benchmark.h index 88360af005..9e65c62f0b 100644 --- a/benchmark/simplify_benchmark.h +++ b/benchmark/simplify_benchmark.h @@ -34,7 +34,7 @@ class SimplifyTestFixture : public benchmark::Fixture std::filesystem::path(__FILE__).parent_path().append("tests/resources/slice_polygon_3.txt").string(), std::filesystem::path(__FILE__).parent_path().append("tests/resources/slice_polygon_4.txt").string() }; - std::vector shapes; + std::vector shapes; void SetUp(const ::benchmark::State& state) { @@ -51,7 +51,7 @@ BENCHMARK_DEFINE_F(SimplifyTestFixture, simplify_local)(benchmark::State& st) Simplify simplify(MM2INT(0.25), MM2INT(0.025), 50000); for (auto _ : st) { - Polygons simplified; + Shape simplified; for (const auto& polys : shapes) { benchmark::DoNotOptimize(simplified = simplify.polygon(polys)); @@ -66,7 +66,7 @@ BENCHMARK_DEFINE_F(SimplifyTestFixture, simplify_slot_noplugin)(benchmark::State { for (auto _ : st) { - Polygons simplified; + Shape simplified; for (const auto& polys : shapes) { benchmark::DoNotOptimize(simplified = slots::instance().modify(polys, MM2INT(0.25), MM2INT(0.025), 50000)); diff --git a/benchmark/wall_benchmark.h b/benchmark/wall_benchmark.h index 13f2021ceb..089524cf9c 100644 --- a/benchmark/wall_benchmark.h +++ b/benchmark/wall_benchmark.h @@ -4,21 +4,20 @@ #ifndef CURAENGINE_WALL_BENCHMARK_H #define CURAENGINE_WALL_BENCHMARK_H +#include +#include +#include +#include #include -#include #include +#include #include "InsetOrderOptimizer.h" #include "WallsComputation.h" +#include "geometry/Polygon.h" #include "settings/Settings.h" #include "sliceDataStorage.h" -#include "utils/polygon.h" - -#include -#include -#include -#include namespace cura { @@ -27,8 +26,8 @@ class WallTestFixture : public benchmark::Fixture public: Settings settings{}; WallsComputation walls_computation{ settings, LayerIndex(100) }; - Polygons square_shape; - Polygons ff_holes; + Shape square_shape; + Shape ff_holes; bool outer_to_inner; SliceLayer layer; @@ -85,7 +84,7 @@ class WallTestFixture : public benchmark::Fixture layer.parts.emplace_back(); SliceLayerPart& part = layer.parts.back(); - part.outline.add(ff_holes); + part.outline.push_back(ff_holes); } void TearDown(const ::benchmark::State& state) @@ -98,8 +97,8 @@ class HolesWallTestFixture : public benchmark::Fixture public: Settings settings{}; WallsComputation walls_computation{ settings, LayerIndex(100) }; - Polygons shape; - Polygons ff_holes; + Shape shape; + Shape ff_holes; bool outer_to_inner; SliceLayer layer; @@ -113,7 +112,7 @@ class HolesWallTestFixture : public benchmark::Fixture buffer << file.rdbuf(); const auto wkt = buffer.str(); - const auto shape = Polygons::fromWkt(buffer.str()); + const auto shape = Shape::fromWkt(buffer.str()); // Settings for a simple 2 walls, about as basic as possible. settings.add("alternate_extra_perimeter", "false"); @@ -145,7 +144,7 @@ class HolesWallTestFixture : public benchmark::Fixture layer.parts.emplace_back(); SliceLayerPart& part = layer.parts.back(); - part.outline.add(shape.paths.front()); + part.outline.push_back(shape.front()); part.print_outline = shape; } @@ -168,7 +167,7 @@ BENCHMARK_DEFINE_F(WallTestFixture, InsetOrderOptimizer_getRegionOrder)(benchmar { walls_computation.generateWalls(&layer, SectionType::WALL); std::vector all_paths; - for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join ) + for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join) { all_paths.emplace_back(line); } @@ -184,7 +183,7 @@ BENCHMARK_DEFINE_F(WallTestFixture, InsetOrderOptimizer_getInsetOrder)(benchmark { walls_computation.generateWalls(&layer, SectionType::WALL); std::vector all_paths; - for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join ) + for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join) { all_paths.emplace_back(line); } @@ -210,7 +209,7 @@ BENCHMARK_DEFINE_F(HolesWallTestFixture, InsetOrderOptimizer_getRegionOrder)(ben { walls_computation.generateWalls(&layer, SectionType::WALL); std::vector all_paths; - for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join ) + for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join) { all_paths.emplace_back(line); } @@ -226,7 +225,7 @@ BENCHMARK_DEFINE_F(HolesWallTestFixture, InsetOrderOptimizer_getInsetOrder)(benc { walls_computation.generateWalls(&layer, SectionType::WALL); std::vector all_paths; - for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join ) + for (auto& line : layer.parts.back().wall_toolpaths | ranges::views::join) { all_paths.emplace_back(line); } diff --git a/conandata.yml b/conandata.yml index 858634c9a0..38e20233f1 100644 --- a/conandata.yml +++ b/conandata.yml @@ -5,3 +5,5 @@ requirements_arcus: - "arcus/5.3.1" requirements_plugins: - "curaengine_grpc_definitions/(latest)@ultimaker/testing" +requirements_cura_resources: + - "cura_resources/(latest)@ultimaker/testing" diff --git a/conanfile.py b/conanfile.py index a696e03a1c..8e325411cd 100644 --- a/conanfile.py +++ b/conanfile.py @@ -32,6 +32,7 @@ class CuraEngineConan(ConanFile): "enable_plugins": [True, False], "enable_sentry": [True, False], "enable_remote_plugins": [True, False], + "with_cura_resources": [True, False], } default_options = { "enable_arcus": True, @@ -40,6 +41,7 @@ class CuraEngineConan(ConanFile): "enable_plugins": True, "enable_sentry": False, "enable_remote_plugins": False, + "with_cura_resources": False, } def set_version(self): @@ -116,6 +118,9 @@ def requirements(self): self.requires("grpc/1.50.1") for req in self.conan_data["requirements_plugins"]: self.requires(req) + if self.options.with_cura_resources: + for req in self.conan_data["requirements_cura_resources"]: + self.requires(req) if self.options.enable_arcus or self.options.enable_plugins: self.requires("protobuf/3.21.12") self.requires("clipper/6.4.2@ultimaker/stable") @@ -212,6 +217,9 @@ def build(self): self.run(f"sentry-cli --auth-token {os.environ['SENTRY_TOKEN']} releases set-commits -o {sentry_org} -p {sentry_project} --commit \"Ultimaker/CuraEngine@{self.conan_data['commit']}\" {self.version}") self.run(f"sentry-cli --auth-token {os.environ['SENTRY_TOKEN']} releases finalize -o {sentry_org} -p {sentry_project} {self.version}") + def deploy(self): + copy(self, "CuraEngine*", src=os.path.join(self.package_folder, "bin"), dst=self.install_folder) + def package(self): match self.settings.os: case "Windows": diff --git a/include/BeadingStrategy/BeadingStrategy.h b/include/BeadingStrategy/BeadingStrategy.h index d72645592f..0a4b1d32f2 100644 --- a/include/BeadingStrategy/BeadingStrategy.h +++ b/include/BeadingStrategy/BeadingStrategy.h @@ -1,14 +1,15 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #ifndef BEADING_STRATEGY_H #define BEADING_STRATEGY_H #include +#include -#include "../settings/types/Angle.h" -#include "../settings/types/Ratio.h" //For the wall transition threshold. -#include "../utils/Point2LL.h" +#include "geometry/Point2LL.h" +#include "settings/types/Angle.h" +#include "settings/types/Ratio.h" //For the wall transition threshold. namespace cura { @@ -45,9 +46,7 @@ class BeadingStrategy BeadingStrategy(const BeadingStrategy& other); - virtual ~BeadingStrategy() - { - } + virtual ~BeadingStrategy() = default; /*! * Retrieve the bead widths with which to cover a given thickness. diff --git a/include/BeadingStrategy/BeadingStrategyFactory.h b/include/BeadingStrategy/BeadingStrategyFactory.h index 2789fa4657..a2b5c1bd51 100644 --- a/include/BeadingStrategy/BeadingStrategyFactory.h +++ b/include/BeadingStrategy/BeadingStrategyFactory.h @@ -1,11 +1,13 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef BEADING_STRATEGY_FACTORY_H #define BEADING_STRATEGY_FACTORY_H -#include "../settings/types/Ratio.h" +#include + #include "BeadingStrategy.h" +#include "settings/types/Ratio.h" namespace cura { diff --git a/include/BeadingStrategy/LimitedBeadingStrategy.h b/include/BeadingStrategy/LimitedBeadingStrategy.h index 42d34f86a4..3e15eefec2 100644 --- a/include/BeadingStrategy/LimitedBeadingStrategy.h +++ b/include/BeadingStrategy/LimitedBeadingStrategy.h @@ -35,7 +35,7 @@ class LimitedBeadingStrategy : public BeadingStrategy coord_t getOptimalThickness(coord_t bead_count) const override; coord_t getTransitionThickness(coord_t lower_bead_count) const override; coord_t getOptimalBeadCount(coord_t thickness) const override; - virtual std::string toString() const override; + std::string toString() const override; coord_t getTransitioningLength(coord_t lower_bead_count) const override; diff --git a/include/BeadingStrategy/OuterWallInsetBeadingStrategy.h b/include/BeadingStrategy/OuterWallInsetBeadingStrategy.h index c8c6eae31d..840c04f888 100644 --- a/include/BeadingStrategy/OuterWallInsetBeadingStrategy.h +++ b/include/BeadingStrategy/OuterWallInsetBeadingStrategy.h @@ -25,7 +25,7 @@ class OuterWallInsetBeadingStrategy : public BeadingStrategy coord_t getOptimalBeadCount(coord_t thickness) const override; coord_t getTransitioningLength(coord_t lower_bead_count) const override; - virtual std::string toString() const; + std::string toString() const override; private: BeadingStrategyPtr parent_; diff --git a/include/BeadingStrategy/RedistributeBeadingStrategy.h b/include/BeadingStrategy/RedistributeBeadingStrategy.h index 2f8304bf39..52d41e1a93 100644 --- a/include/BeadingStrategy/RedistributeBeadingStrategy.h +++ b/include/BeadingStrategy/RedistributeBeadingStrategy.h @@ -45,7 +45,7 @@ class RedistributeBeadingStrategy : public BeadingStrategy coord_t getTransitioningLength(coord_t lower_bead_count) const override; double getTransitionAnchorPos(coord_t lower_bead_count) const override; - virtual std::string toString() const; + std::string toString() const override; protected: /*! diff --git a/include/BeadingStrategy/WideningBeadingStrategy.h b/include/BeadingStrategy/WideningBeadingStrategy.h index bda69f345d..cd0f00c406 100644 --- a/include/BeadingStrategy/WideningBeadingStrategy.h +++ b/include/BeadingStrategy/WideningBeadingStrategy.h @@ -27,14 +27,14 @@ class WideningBeadingStrategy : public BeadingStrategy virtual ~WideningBeadingStrategy() override = default; - virtual Beading compute(coord_t thickness, coord_t bead_count) const override; - virtual coord_t getOptimalThickness(coord_t bead_count) const override; - virtual coord_t getTransitionThickness(coord_t lower_bead_count) const override; - virtual coord_t getOptimalBeadCount(coord_t thickness) const override; - virtual coord_t getTransitioningLength(coord_t lower_bead_count) const override; - virtual double getTransitionAnchorPos(coord_t lower_bead_count) const override; - virtual std::vector getNonlinearThicknesses(coord_t lower_bead_count) const override; - virtual std::string toString() const override; + Beading compute(coord_t thickness, coord_t bead_count) const override; + coord_t getOptimalThickness(coord_t bead_count) const override; + coord_t getTransitionThickness(coord_t lower_bead_count) const override; + coord_t getOptimalBeadCount(coord_t thickness) const override; + coord_t getTransitioningLength(coord_t lower_bead_count) const override; + double getTransitionAnchorPos(coord_t lower_bead_count) const override; + std::vector getNonlinearThicknesses(coord_t lower_bead_count) const override; + std::string toString() const override; protected: BeadingStrategyPtr parent_; diff --git a/include/BoostInterface.hpp b/include/BoostInterface.hpp index a294450a11..b7a305dd54 100644 --- a/include/BoostInterface.hpp +++ b/include/BoostInterface.hpp @@ -7,9 +7,9 @@ #include #include -#include "utils/Point2LL.h" +#include "geometry/Point2LL.h" +#include "geometry/Polygon.h" #include "utils/PolygonsSegmentIndex.h" -#include "utils/polygon.h" using CSegment = cura::PolygonsSegmentIndex; diff --git a/include/ExtruderPlan.h b/include/ExtruderPlan.h index daab6c8e4d..338dfba115 100644 --- a/include/ExtruderPlan.h +++ b/include/ExtruderPlan.h @@ -7,12 +7,12 @@ #include "FanSpeedLayerTime.h" #include "RetractionConfig.h" #include "gcodeExport.h" +#include "geometry/Point2LL.h" #include "pathPlanning/GCodePath.h" #include "pathPlanning/NozzleTempInsert.h" #include "pathPlanning/TimeMaterialEstimates.h" #include "settings/types/LayerIndex.h" #include "settings/types/Ratio.h" -#include "utils/Point2LL.h" #ifdef BUILD_TESTS #include //Friend tests, so that they can inspect the privates. diff --git a/include/FffGcodeWriter.h b/include/FffGcodeWriter.h index 4c4061c37b..9c2482a937 100644 --- a/include/FffGcodeWriter.h +++ b/include/FffGcodeWriter.h @@ -9,11 +9,9 @@ #include "ExtruderUse.h" #include "FanSpeedLayerTime.h" +#include "GCodePathConfig.h" #include "LayerPlanBuffer.h" #include "gcodeExport.h" -#include "settings/MeshPathConfigs.h" -#include "settings/PathConfigStorage.h" //For the MeshPathConfigs subclass. -#include "utils/ExtrusionLine.h" //Processing variable-width paths. #include "utils/NoCopy.h" #include "utils/gettime.h" @@ -21,12 +19,13 @@ namespace cura { class AngleDegrees; -class Polygons; +class Shape; class SkinPart; class SliceDataStorage; class SliceMeshStorage; class SliceLayer; class SliceLayerPart; +struct MeshPathConfigs; /*! * Secondary stage in Fused Filament Fabrication processing: The generated polygons are used in the gcode generation. @@ -295,9 +294,11 @@ class FffGcodeWriter : public NoCopy * * \param[in] storage where the slice data is stored. * \param current_extruder The current extruder with which we last printed + * \param global_extruders_used The extruders that are at some point used for the print job * \return The order of extruders for a layer beginning with \p current_extruder */ - std::vector getUsedExtrudersOnLayer(const SliceDataStorage& storage, const size_t start_extruder, const LayerIndex& layer_nr) const; + std::vector + getUsedExtrudersOnLayer(const SliceDataStorage& storage, const size_t start_extruder, const LayerIndex& layer_nr, const std::vector& global_extruders_used) const; /*! * Calculate in which order to plan the meshes of a specific extruder @@ -561,7 +562,7 @@ class FffGcodeWriter : public NoCopy LayerPlan& gcode_layer, const SliceMeshStorage& mesh, const size_t extruder_nr, - const Polygons& area, + const Shape& area, const GCodePathConfig& config, EFillMethod pattern, const AngleDegrees skin_angle, @@ -594,7 +595,7 @@ class FffGcodeWriter : public NoCopy * \param last_position The position the print head is in before going to fill the part * \return The location near where to start filling the part */ - std::optional getSeamAvoidingLocation(const Polygons& filling_part, int filling_angle, Point2LL last_position) const; + std::optional getSeamAvoidingLocation(const Shape& filling_part, int filling_angle, Point2LL last_position) const; /*! * Add the g-code for ironing the top surface. @@ -639,7 +640,7 @@ class FffGcodeWriter : public NoCopy * \param gcodeLayer The initial planning of the g-code of the layer. * \return Whether any support skin was added to the layer plan. */ - bool addSupportRoofsToGCode(const SliceDataStorage& storage, const Polygons& support_roof_outlines, const GCodePathConfig& current_roof_config, LayerPlan& gcode_layer) const; + bool addSupportRoofsToGCode(const SliceDataStorage& storage, const Shape& support_roof_outlines, const GCodePathConfig& current_roof_config, LayerPlan& gcode_layer) const; /*! * Add the support bottoms to the layer plan \p gcodeLayer of the current @@ -709,8 +710,8 @@ class FffGcodeWriter : public NoCopy * \return true if there needs to be a skin edge support wall in this layer, otherwise false */ static bool partitionInfillBySkinAbove( - Polygons& infill_below_skin, - Polygons& infill_not_below_skin, + Shape& infill_below_skin, + Shape& infill_not_below_skin, const LayerPlan& gcode_layer, const SliceMeshStorage& mesh, const SliceLayerPart& part, diff --git a/include/InsetOrderOptimizer.h b/include/InsetOrderOptimizer.h index 7ff5300090..f03a7fc461 100644 --- a/include/InsetOrderOptimizer.h +++ b/include/InsetOrderOptimizer.h @@ -55,7 +55,8 @@ class InsetOrderOptimizer const size_t wall_0_extruder_nr, const size_t wall_x_extruder_nr, const ZSeamConfig& z_seam_config, - const std::vector& paths); + const std::vector& paths, + const Shape& disallowed_areas_for_seams = {}); /*! * Adds the insets to the given layer plan. @@ -106,10 +107,11 @@ class InsetOrderOptimizer const ZSeamConfig& z_seam_config_; const std::vector& paths_; const LayerIndex layer_nr_; + Shape disallowed_areas_for_seams_; - std::vector> inset_polys_; // vector of vectors holding the inset polygons - Polygons retraction_region_; // After printing an outer wall, move into this region so that retractions do not leave visible blobs. Calculated lazily if needed (see - // retraction_region_calculated). + std::vector> inset_polys_; // vector of vectors holding the inset polygons + Shape retraction_region_; // After printing an outer wall, move into this region so that retractions do not leave visible blobs. Calculated lazily if needed (see + // retraction_region_calculated). /*! * Determine if the paths should be reversed diff --git a/include/InterlockingGenerator.h b/include/InterlockingGenerator.h index da37b66e53..3ab4e27335 100644 --- a/include/InterlockingGenerator.h +++ b/include/InterlockingGenerator.h @@ -8,8 +8,9 @@ #include #include +#include "geometry/PointMatrix.h" +#include "geometry/Polygon.h" #include "utils/VoxelUtils.h" -#include "utils/polygon.h" namespace cura { @@ -106,7 +107,7 @@ class InterlockingGenerator * \param detec The expand distance. (Not equal to offset, but a series of small offsets and differences). * \return A pair of polygons that repressent the 'borders' of a and b, but expanded 'perpendicularly'. */ - std::pair growBorderAreasPerpendicular(const Polygons& a, const Polygons& b, const coord_t& detect) const; + std::pair growBorderAreasPerpendicular(const Shape& a, const Shape& b, const coord_t& detect) const; /*! Special handling for thin strips of material. * @@ -132,7 +133,7 @@ class InterlockingGenerator * \param kernel The dilation kernel to give the returned voxel shell more thickness * \param[out] cells The output cells which elong to the shell */ - void addBoundaryCells(const std::vector& layers, const DilationKernel& kernel, std::unordered_set& cells) const; + void addBoundaryCells(const std::vector& layers, const DilationKernel& kernel, std::unordered_set& cells) const; /*! * Compute the regions occupied by both models. @@ -140,13 +141,13 @@ class InterlockingGenerator * A morphological close is performed so that we don't register small gaps between the two models as being separate. * \return layer_regions The computed layer regions */ - std::vector computeUnionedVolumeRegions() const; + std::vector computeUnionedVolumeRegions() const; /*! * Generate the polygons for the beams of a single cell * \return cell_area_per_mesh_per_layer The output polygons for each beam */ - std::vector> generateMicrostructure() const; + std::vector> generateMicrostructure() const; /*! * Change the outlines of the meshes with the computed interlocking structure. @@ -154,7 +155,7 @@ class InterlockingGenerator * \param cells The cells where we want to apply the interlocking structure. * \param layer_regions The total volume of the two meshes combined (and small gaps closed) */ - void applyMicrostructureToOutlines(const std::unordered_set& cells, const std::vector& layer_regions) const; + void applyMicrostructureToOutlines(const std::unordered_set& cells, const std::vector& layer_regions) const; static const coord_t ignored_gap_ = 100u; //!< Distance between models to be considered next to each other so that an interlocking structure will be generated there diff --git a/include/LayerPlan.h b/include/LayerPlan.h index b27f77c04e..9b6aa7cd06 100644 --- a/include/LayerPlan.h +++ b/include/LayerPlan.h @@ -10,6 +10,9 @@ #include "PathOrderOptimizer.h" #include "SpaceFillType.h" #include "gcodeExport.h" +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/Polygon.h" #include "pathPlanning/GCodePath.h" #include "pathPlanning/NozzleTempInsert.h" #include "pathPlanning/TimeMaterialEstimates.h" @@ -17,7 +20,6 @@ #include "settings/PathConfigStorage.h" #include "settings/types/LayerIndex.h" #include "utils/ExtrusionJunction.h" -#include "utils/polygon.h" #ifdef BUILD_TESTS #include //Friend tests, so that they can inspect the privates. @@ -91,13 +93,13 @@ class LayerPlan : public NoCopy std::optional> next_layer_acc_jerk_; //!< If there is a next layer, the first acceleration and jerk it starts with. bool was_inside_; //!< Whether the last planned (extrusion) move was inside a layer part bool is_inside_; //!< Whether the destination of the next planned travel move is inside a layer part - Polygons comb_boundary_minimum_; //!< The minimum boundary within which to comb, or to move into when performing a retraction. - Polygons comb_boundary_preferred_; //!< The boundary preferably within which to comb, or to move into when performing a retraction. + Shape comb_boundary_minimum_; //!< The minimum boundary within which to comb, or to move into when performing a retraction. + Shape comb_boundary_preferred_; //!< The boundary preferably within which to comb, or to move into when performing a retraction. Comb* comb_; coord_t comb_move_inside_distance_; //!< Whenever using the minimum boundary for combing it tries to move the coordinates inside by this distance after calculating the combing. - Polygons bridge_wall_mask_; //!< The regions of a layer part that are not supported, used for bridging - Polygons overhang_mask_; //!< The regions of a layer part where the walls overhang - Polygons roofing_mask_; //!< The regions of a layer part where the walls are exposed to the air + Shape bridge_wall_mask_; //!< The regions of a layer part that are not supported, used for bridging + Shape overhang_mask_; //!< The regions of a layer part where the walls overhang + Shape roofing_mask_; //!< The regions of a layer part where the walls are exposed to the air bool min_layer_time_used = false; //!< Wether or not the minimum layer time (cool_min_layer_time) was actually used in this layerplan. @@ -179,7 +181,7 @@ class LayerPlan : public NoCopy */ ExtruderTrain* getLastPlannedExtruderTrain(); - const Polygons* getCombBoundaryInside() const; + const Shape* getCombBoundaryInside() const; LayerIndex getLayerNr() const; @@ -272,21 +274,21 @@ class LayerPlan : public NoCopy * * \param polys The unsupported areas of the part currently being processed that will require bridges. */ - void setBridgeWallMask(const Polygons& polys); + void setBridgeWallMask(const Shape& polys); /*! * Set overhang_mask. * * \param polys The overhung areas of the part currently being processed that will require modified print settings */ - void setOverhangMask(const Polygons& polys); + void setOverhangMask(const Shape& polys); /*! * Set roofing_mask. * * \param polys The areas of the part currently being processed that will require roofing. */ - void setRoofingMask(const Polygons& polys); + void setRoofingMask(const Shape& polys); /*! * Travel to a certain point, with all of the procedures necessary to do so. @@ -372,7 +374,7 @@ class LayerPlan : public NoCopy * \param always_retract Whether to force a retraction when moving to the start of the polygon (used for outer walls) */ void addPolygon( - ConstPolygonRef polygon, + const Polygon& polygon, int startIdx, const bool reverse, const GCodePathConfig& config, @@ -410,7 +412,7 @@ class LayerPlan : public NoCopy * If unset, this causes it to start near the last planned location. */ void addPolygonsByOptimizer( - const Polygons& polygons, + const Shape& polygons, const GCodePathConfig& config, const ZSeamConfig& z_seam_config = ZSeamConfig(), coord_t wall_0_wipe_dist = 0, @@ -471,7 +473,7 @@ class LayerPlan : public NoCopy * start of the wall (used for outer walls). */ void addWall( - ConstPolygonRef wall, + const Polygon& wall, int start_idx, const Settings& settings, const GCodePathConfig& default_config, @@ -540,7 +542,7 @@ class LayerPlan : public NoCopy * \param alternate_inset_direction_modifier Whether to alternate the direction of the walls for each inset. */ void addWalls( - const Polygons& walls, + const Shape& walls, const Settings& settings, const GCodePathConfig& default_config, const GCodePathConfig& roofing_config, @@ -552,7 +554,7 @@ class LayerPlan : public NoCopy /*! * Add lines to the gcode with optimized order. - * \param polygons The lines + * \param lines The lines * \param config The config of the lines * \param space_fill_type The type of space filling used to generate the line segments (should be either Lines or PolyLines!) * \param enable_travel_optimization Whether to enable some potentially time consuming optimization of order the lines are printed to reduce the travel time required. @@ -561,10 +563,11 @@ class LayerPlan : public NoCopy * \param near_start_location Optional: Location near where to add the first line. If not provided the last position is used. * \param fan_speed optional fan speed override for this path * \param reverse_print_direction Whether to reverse the optimized order and their printing direction. - * \param order_requirements Pairs where first needs to be printed before second. Pointers are pointing to elements of \p polygons + * \param order_requirements Pairs where first needs to be printed before second. Pointers are pointing to elements of \p lines */ + template void addLinesByOptimizer( - const Polygons& polygons, + const LinesSet& lines, const GCodePathConfig& config, const SpaceFillType space_fill_type, const bool enable_travel_optimization = false, @@ -573,11 +576,36 @@ class LayerPlan : public NoCopy const std::optional near_start_location = std::optional(), const double fan_speed = GCodePathConfig::FAN_SPEED_DEFAULT, const bool reverse_print_direction = false, - const std::unordered_multimap& order_requirements = PathOrderOptimizer::no_order_requirements_); + const std::unordered_multimap& order_requirements = PathOrderOptimizer::no_order_requirements_); /*! - * Add polygons to the g-code with monotonic order. - * \param polygons The lines to add. + * Add lines to the gcode with optimized order. + * \param lines The lines + * \param config The config of the lines + * \param space_fill_type The type of space filling used to generate the line segments (should be either Lines or PolyLines!) + * \param enable_travel_optimization Whether to enable some potentially time consuming optimization of order the lines are printed to reduce the travel time required. + * \param wipe_dist (optional) the distance wiped without extruding after laying down a line. + * \param flow_ratio The ratio with which to multiply the extrusion amount + * \param near_start_location Optional: Location near where to add the first line. If not provided the last position is used. + * \param fan_speed optional fan speed override for this path + * \param reverse_print_direction Whether to reverse the optimized order and their printing direction. + * \param order_requirements Pairs where first needs to be printed before second. Pointers are pointing to elements of \p lines + */ + void addLinesByOptimizer( + const MixedLinesSet& lines, + const GCodePathConfig& config, + const SpaceFillType space_fill_type, + const bool enable_travel_optimization = false, + const coord_t wipe_dist = 0, + const Ratio flow_ratio = 1.0, + const std::optional near_start_location = std::optional(), + const double fan_speed = GCodePathConfig::FAN_SPEED_DEFAULT, + const bool reverse_print_direction = false, + const std::unordered_multimap& order_requirements = PathOrderOptimizer::no_order_requirements_); + + /*! + * Add lines to the g-code with monotonic order. + * \param lines The lines to add. * \param config The settings to print those lines with. * \param space_fill_type The type of space filling used to generate the * line segments (should be either Lines or PolyLines!) @@ -597,8 +625,8 @@ class LayerPlan : public NoCopy * \param fan_speed Fan speed override for this path. */ void addLinesMonotonic( - const Polygons& area, - const Polygons& polygons, + const Shape& area, + const OpenLinesSet& lines, const GCodePathConfig& config, const SpaceFillType space_fill_type, const AngleRadians monotonic_direction, @@ -608,25 +636,6 @@ class LayerPlan : public NoCopy const Ratio flow_ratio = 1.0_r, const double fan_speed = GCodePathConfig::FAN_SPEED_DEFAULT); -protected: - /*! - * Add order optimized lines to the gcode. - * \param paths The paths in order - * \param config The config of the lines - * \param space_fill_type The type of space filling used to generate the line segments (should be either Lines or PolyLines!) - * \param wipe_dist (optional) the distance wiped without extruding after laying down a line. - * \param flow_ratio The ratio with which to multiply the extrusion amount - * \param fan_speed optional fan speed override for this path - */ - void addLinesInGivenOrder( - const std::vector>& paths, - const GCodePathConfig& config, - const SpaceFillType space_fill_type, - const coord_t wipe_dist, - const Ratio flow_ratio, - const double fan_speed); - -public: /*! * Add a spiralized slice of wall that is interpolated in X/Y between \p last_wall and \p wall. * @@ -642,8 +651,8 @@ class LayerPlan : public NoCopy */ void spiralizeWallSlice( const GCodePathConfig& config, - ConstPolygonRef wall, - ConstPolygonRef last_wall, + const Polygon& wall, + const Polygon& last_wall, int seam_vertex_idx, int last_seam_vertex_idx, const bool is_top_layer, @@ -778,9 +787,26 @@ class LayerPlan : public NoCopy * - If CombingMode::INFILL: Add the infill (infill only). * * \param boundary_type The boundary type to compute. - * \return the combing boundary or an empty Polygons if no combing is required + * \return the combing boundary or an empty Shape if no combing is required + */ + Shape computeCombBoundary(const CombBoundary boundary_type); + + /*! + * Add order optimized lines to the gcode. + * \param lines The lines in order + * \param config The config of the lines + * \param space_fill_type The type of space filling used to generate the line segments (should be either Lines or PolyLines!) + * \param wipe_dist (optional) the distance wiped without extruding after laying down a line. + * \param flow_ratio The ratio with which to multiply the extrusion amount + * \param fan_speed optional fan speed override for this path */ - Polygons computeCombBoundary(const CombBoundary boundary_type); + void addLinesInGivenOrder( + const std::vector>& lines, + const GCodePathConfig& config, + const SpaceFillType space_fill_type, + const coord_t wipe_dist, + const Ratio flow_ratio, + const double fan_speed); }; } // namespace cura diff --git a/include/LayerPlanBuffer.h b/include/LayerPlanBuffer.h index 1fb0ef56a7..3916321729 100644 --- a/include/LayerPlanBuffer.h +++ b/include/LayerPlanBuffer.h @@ -7,17 +7,15 @@ #include #include -#include "ExtruderPlan.h" -#include "LayerPlan.h" #include "Preheat.h" -#include "gcodeExport.h" #include "settings/Settings.h" #include "settings/types/Duration.h" namespace cura { - +class LayerPlan; +class ExtruderPlan; class GCodeExport; /*! diff --git a/include/MeshGroup.h b/include/MeshGroup.h index 232bf08662..6f20523f4a 100644 --- a/include/MeshGroup.h +++ b/include/MeshGroup.h @@ -18,9 +18,21 @@ class Matrix4x3D; * One MeshGroup is a whole which is printed at once. * Generally there is one single MeshGroup, though when using one-at-a-time printing, multiple MeshGroups are processed consecutively. */ -class MeshGroup : public NoCopy +class MeshGroup { public: + MeshGroup() = default; + + ~MeshGroup() = default; + + MeshGroup(MeshGroup&& other) noexcept = default; + + MeshGroup& operator=(MeshGroup&& other) noexcept = default; + + /* Copying a MeshGroup is not allowed */ + MeshGroup(const MeshGroup& other) = delete; + MeshGroup& operator=(const MeshGroup& other) = delete; + std::vector meshes; Settings settings; diff --git a/include/PathOrder.h b/include/PathOrder.h index 7fd338fbfc..d1c826990f 100644 --- a/include/PathOrder.h +++ b/include/PathOrder.h @@ -4,7 +4,7 @@ #ifndef PATHORDER_H #define PATHORDER_H -#include "PathOrdering.h" +#include "path_ordering.h" #include "settings/ZSeamConfig.h" //To get the seam settings. #include "utils/polygonUtils.h" diff --git a/include/PathOrderMonotonic.h b/include/PathOrderMonotonic.h index 2b8aa5dfc1..b2334e7a9e 100644 --- a/include/PathOrderMonotonic.h +++ b/include/PathOrderMonotonic.h @@ -9,7 +9,7 @@ #include //To track starting points of monotonic sequences. #include "PathOrder.h" -#include "PathOrdering.h" +#include "path_ordering.h" namespace cura { @@ -62,7 +62,7 @@ class PathOrderMonotonic : public PathOrder // Get the vertex data and store it in the paths. for (Path& path : this->paths_) { - path.converted_ = path.getVertexData(); + path.converted_ = &path.getVertexData(); } std::vector reordered; // To store the result in. At the end, we'll std::swap with the real paths. diff --git a/include/PathOrderOptimizer.h b/include/PathOrderOptimizer.h index df1cc07c7e..cf54338a06 100644 --- a/include/PathOrderOptimizer.h +++ b/include/PathOrderOptimizer.h @@ -1,9 +1,10 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #ifndef PATHORDEROPTIMIZER_H #define PATHORDEROPTIMIZER_H +#include #include #include @@ -15,10 +16,9 @@ #include #include -#include "InsetOrderOptimizer.h" // for makeOrderIncludeTransitive -#include "PathOrdering.h" #include "pathPlanning/CombPath.h" //To calculate the combing distance if we want to use combing. #include "pathPlanning/LinePolygonsCrossings.h" //To prevent calculating combing distances if we don't cross the combing borders. +#include "path_ordering.h" #include "settings/EnumSettings.h" //To get the seam settings. #include "settings/ZSeamConfig.h" //To read the seam configuration. #include "utils/linearAlg2D.h" //To find the angle of corners to hide seams. @@ -61,6 +61,8 @@ class PathOrderOptimizer { public: using OrderablePath = PathOrdering; + /* Areas defined here are not allowed to have the start the prints */ + Shape disallowed_area_for_seams; /*! * After optimizing, this contains the paths that need to be printed in the * correct order. @@ -107,10 +109,11 @@ class PathOrderOptimizer const Point2LL& start_point, const ZSeamConfig seam_config = ZSeamConfig(), const bool detect_loops = false, - const Polygons* combing_boundary = nullptr, + const Shape* combing_boundary = nullptr, const bool reverse_direction = false, const std::unordered_multimap& order_requirements = no_order_requirements_, - const bool group_outer_walls = false) + const bool group_outer_walls = false, + const Shape& disallowed_areas_for_seams = {}) : start_point_(start_point) , seam_config_(seam_config) , combing_boundary_((combing_boundary != nullptr && ! combing_boundary->empty()) ? combing_boundary : nullptr) @@ -118,6 +121,8 @@ class PathOrderOptimizer , reverse_direction_(reverse_direction) , _group_outer_walls(group_outer_walls) , order_requirements_(&order_requirements) + , disallowed_area_for_seams{ disallowed_areas_for_seams } + { } @@ -157,7 +162,7 @@ class PathOrderOptimizer // Get the vertex data and store it in the paths. for (auto& path : paths_) { - path.converted_ = path.getVertexData(); + path.converted_ = &path.getVertexData(); vertices_to_paths_.emplace(path.vertices_, &path); } @@ -257,7 +262,7 @@ class PathOrderOptimizer /*! * Boundary to avoid when making travel moves. */ - const Polygons* combing_boundary_; + const Shape* combing_boundary_; /*! * Whether to check polylines to see if they are closed, before optimizing. @@ -402,7 +407,7 @@ class PathOrderOptimizer } auto local_current_position = current_position; - while (candidates.size() != 0) + while (! candidates.empty()) { Path best_candidate = findClosestPathVertices(local_current_position, candidates); @@ -604,6 +609,52 @@ class PathOrderOptimizer return best_candidate; } + /** + * @brief Analyze the positions in a path and determine the next optimal position based on a proximity criterion. + * + * This function iteratively examines positions along the given path, checking if the position is close to 3D model. + * Each position is specified by an index, starting with `best_pos`. If the position is close to the model according to + * `isVertexCloseToPolygonPath` function, the function recursively calls itself with the next position. This process is + * repeated until all positions have been checked or `number_of_paths_analysed` becomes equal to `path_size`. + * If `number_of_paths_analysed` becomes equal to `path_size`, it logs a warning and returns the current best position. + * + * @param best_pos The index of the initial position for analysis in the path. + * @param path An OrderablePath instance containing the path to be examined. + * @param number_of_paths_analysed Optionally, the initial index of paths analysed. Defaults to 0. + * @return The index of the next optimal position in the path sequence. May be the same as the input `best_pos`, + * or may be incremented to a different location based on the proximity criterion. + * + * @note This function uses recursion to evaluate each position in the path. + * @note The process stops prematurely if no start path is found for the support z seam distance. + * This typically happens when the distance of the support seam from the model is bigger than all the support wall points. + */ + + size_t pathIfZseamIsInDisallowedArea(size_t best_pos, const OrderablePath& path, size_t number_of_paths_analysed) + { + size_t path_size = path.converted_->size(); + if (path_size > number_of_paths_analysed) + { + if (! disallowed_area_for_seams.empty()) + { + Point2LL current_candidate = (path.converted_)->at(best_pos); + if (disallowed_area_for_seams.inside(current_candidate, true)) + { + size_t next_best_position = (path_size > best_pos + 1) ? best_pos + 1 : 0; + number_of_paths_analysed += 1; + best_pos = pathIfZseamIsInDisallowedArea(next_best_position, path, number_of_paths_analysed); + } + } + } + else + { + spdlog::warn("No start path found for support z seam distance"); + // We can also calculate the best point to start at this point. + // This usually happens when the distance of support seam from model is bigger than the whole support wall points. + } + return best_pos; + } + + /*! * Find the vertex which will be the starting point of printing a polygon or * polyline. @@ -633,10 +684,7 @@ class PathOrderOptimizer { return path.converted_->size() - 1; // Back end is closer. } - else - { - return 0; // Front end is closer. - } + return 0; // Front end is closer. } // Rest of the function only deals with (closed) polygons. We need to be able to find the seam location of those polygons. @@ -661,7 +709,7 @@ class PathOrderOptimizer size_t best_i; double best_score = std::numeric_limits::infinity(); - for (const auto& [i, here] : **path.converted_ | ranges::views::drop_last(1) | ranges::views::enumerate) + for (const auto& [i, here] : *path.converted_ | ranges::views::drop_last(1) | ranges::views::enumerate) { // For most seam types, the shortest distance matters. Not for SHARPEST_CORNER though. // For SHARPEST_CORNER, use a fixed starting score of 0. @@ -674,6 +722,7 @@ class PathOrderOptimizer // angles > 0 are convex (right turning) double corner_shift; + if (seam_config_.type_ == EZSeamType::SHORTEST) { // the more a corner satisfies our criteria, the closer it appears to be @@ -740,6 +789,10 @@ class PathOrderOptimizer } } + if (! disallowed_area_for_seams.empty()) + { + best_i = pathIfZseamIsInDisallowedArea(best_i, path, 0); + } return best_i; } @@ -884,7 +937,7 @@ class PathOrderOptimizer * \param polygon A polygon to get a random vertex of. * \return A random index in that polygon. */ - size_t getRandomPointInPolygon(ConstPolygonRef const& polygon) const + size_t getRandomPointInPolygon(const PointsSet& polygon) const { return rand() % polygon.size(); } diff --git a/include/PrimeTower.h b/include/PrimeTower.h index 61a0beb901..c1f4927880 100644 --- a/include/PrimeTower.h +++ b/include/PrimeTower.h @@ -8,9 +8,9 @@ #include #include "ExtruderUse.h" +#include "geometry/Polygon.h" #include "settings/EnumSettings.h" #include "settings/types/LayerIndex.h" -#include "utils/polygon.h" // Polygons #include "utils/polygonUtils.h" namespace cura @@ -28,8 +28,8 @@ class LayerPlan; class PrimeTower { private: - using MovesByExtruder = std::vector; - using MovesByLayer = std::vector; + using MovesByExtruder = std::map; + using MovesByLayer = std::map>; size_t extruder_count_; //!< Number of extruders @@ -38,7 +38,7 @@ class PrimeTower Point2LL post_wipe_point_; //!< Location to post-wipe the unused nozzle off on - std::vector prime_tower_start_locations_; //!< The differernt locations where to pre-wipe the active nozzle + std::vector prime_tower_start_locations_; //!< The differernt locations where to pre-wipe the active nozzle const unsigned int number_of_prime_tower_start_locations_ = 21; //!< The required size of \ref PrimeTower::wipe_locations MovesByExtruder prime_moves_; //!< For each extruder, the moves to be processed for actual priming. @@ -49,13 +49,13 @@ class PrimeTower * The polygons represent the sparse pattern to be printed when all the given extruders are unused for this layer * and the given extruder is currently in use */ - std::map> sparse_pattern_per_extruders_; + std::map> sparse_pattern_per_extruders_; MovesByLayer base_extra_moves_; //!< For each layer and each extruder, the extra moves to be processed for better adhesion/strength MovesByExtruder inset_extra_moves_; //!< For each extruder, the extra inset moves to be processed for better adhesion on initial layer - Polygons outer_poly_; //!< The outline of the outermost prime tower. - std::vector outer_poly_base_; //!< The outline of the layers having extra width for the base + Shape outer_poly_; //!< The outline of the outermost prime tower. + std::vector outer_poly_base_; //!< The outline of the layers having extra width for the base public: bool enabled_; //!< Whether the prime tower is enabled. @@ -79,10 +79,12 @@ class PrimeTower */ PrimeTower(); + void initializeExtruders(const std::vector& used_extruders); + /*! * Check whether we actually use the prime tower. */ - void checkUsed(const SliceDataStorage& storage); + void checkUsed(); /*! * Generate the prime tower area to be used on each layer @@ -126,12 +128,12 @@ class PrimeTower * \param[in] layer_nr The index of the layer * \return The outer polygon for the prime tower at the given layer */ - const Polygons& getOuterPoly(const LayerIndex& layer_nr) const; + const Shape& getOuterPoly(const LayerIndex& layer_nr) const; /*! * Get the outer polygon for the very first layer, which may be the priming polygon only, or a larger polygon if there is a base */ - const Polygons& getGroundPoly() const; + const Shape& getGroundPoly() const; private: /*! @@ -160,7 +162,7 @@ class PrimeTower * \param line_width The actual line width of the extruder * \param actual_extruder_nr The number of the actual extruder to be used */ - Polygons generatePath_sparseInfill( + Shape generatePath_sparseInfill( const size_t first_extruder_idx, const size_t last_extruder_idx, const std::vector& rings_radii, diff --git a/include/SkeletalTrapezoidation.h b/include/SkeletalTrapezoidation.h index 86f13f2782..fafb69a002 100644 --- a/include/SkeletalTrapezoidation.h +++ b/include/SkeletalTrapezoidation.h @@ -14,12 +14,12 @@ #include "SkeletalTrapezoidationEdge.h" #include "SkeletalTrapezoidationGraph.h" #include "SkeletalTrapezoidationJoint.h" +#include "geometry/Polygon.h" #include "settings/types/Ratio.h" #include "utils/ExtrusionJunction.h" #include "utils/ExtrusionLine.h" #include "utils/HalfEdgeGraph.h" #include "utils/PolygonsSegmentIndex.h" -#include "utils/polygon.h" #include "utils/section_type.h" namespace cura @@ -110,7 +110,7 @@ class SkeletalTrapezoidation * distance. */ SkeletalTrapezoidation( - const Polygons& polys, + const Shape& polys, const BeadingStrategy& beading_strategy, AngleRadians transitioning_angle, coord_t discretization_step_size, @@ -165,7 +165,7 @@ class SkeletalTrapezoidation * Another complication arises because the VD uses floating logic, which can result in zero-length segments after rounding to integers. * We therefore collapse edges and their whole cells afterwards. */ - void constructFromPolygons(const Polygons& polys); + void constructFromPolygons(const Shape& polys); node_t& makeNode(vd_t::vertex_type& vd_node, Point2LL p); //!< Get the node which the VD node maps to, or create a new mapping if there wasn't any yet. diff --git a/include/SkeletalTrapezoidationJoint.h b/include/SkeletalTrapezoidationJoint.h index daa9087cdf..059f6d50d1 100644 --- a/include/SkeletalTrapezoidationJoint.h +++ b/include/SkeletalTrapezoidationJoint.h @@ -7,7 +7,7 @@ #include // smart pointers #include "BeadingStrategy/BeadingStrategy.h" -#include "utils/Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/SkirtBrim.h b/include/SkirtBrim.h index 3cad8b4f29..696ecf94aa 100644 --- a/include/SkirtBrim.h +++ b/include/SkirtBrim.h @@ -14,7 +14,7 @@ namespace cura { -class Polygons; +class Shape; class SliceDataStorage; constexpr coord_t min_brim_line_length = 3000u; //!< open polyline brim lines smaller than this will be removed @@ -28,7 +28,7 @@ class SkirtBrim struct Offset { Offset( - const std::variant& reference_outline_or_index, + const std::variant& reference_outline_or_index, const bool outside, const bool inside, const coord_t offset_value, @@ -47,7 +47,7 @@ class SkirtBrim { } - std::variant reference_outline_or_index_; + std::variant reference_outline_or_index_; bool outside_; //!< Wether to offset outward from the reference polygons bool inside_; //!< Wether to offset inward from the reference polygons coord_t offset_value_; //!< Distance by which to offset from the reference @@ -124,7 +124,7 @@ class SkirtBrim * \param[out] starting_outlines The first layer outlines from which to compute the offsets. Returned as output parameter because pointers need to stay valid. * \return An ordered list of offsets to perform in the order in which they are to be performed. */ - std::vector generateBrimOffsetPlan(std::vector& starting_outlines); + std::vector generateBrimOffsetPlan(std::vector& starting_outlines); /*! * Generate the primary skirt/brim of the one skirt_brim_extruder or of all extruders simultaneously. @@ -134,7 +134,7 @@ class SkirtBrim * \param[in,out] allowed_areas_per_extruder The difference between the machine bed area (offsetted by the nozzle offset) and the covered_area. * \return The total length of the brim lines added by this method per extruder. */ - std::vector generatePrimaryBrim(std::vector& all_brim_offsets, Polygons& covered_area, std::vector& allowed_areas_per_extruder); + std::vector generatePrimaryBrim(std::vector& all_brim_offsets, Shape& covered_area, std::vector& allowed_areas_per_extruder); /*! * Generate the brim inside the ooze shield and draft shield @@ -145,7 +145,7 @@ class SkirtBrim * \param[in,out] brim_covered_area The area that was covered with brim before (in) and after (out) adding the shield brims * \param[in,out] allowed_areas_per_extruder The difference between the machine areas and the \p covered_area */ - void generateShieldBrim(Polygons& brim_covered_area, std::vector& allowed_areas_per_extruder); + void generateShieldBrim(Shape& brim_covered_area, std::vector& allowed_areas_per_extruder); /*! * \brief Get the reference outline of the first layer around which to @@ -158,7 +158,7 @@ class SkirtBrim * \param extruder_nr The extruder for which to get the outlines. -1 to include outliens for all extruders * \return The resulting reference polygons */ - Polygons getFirstLayerOutline(const int extruder_nr = -1); + Shape getFirstLayerOutline(const int extruder_nr = -1); /*! * The disallowed area around the internal holes of parts with other parts inside which would get an external brim. @@ -170,7 +170,7 @@ class SkirtBrim * \param extruder_nr The extruder for which to compute disallowed areas * \return The disallowed areas */ - Polygons getInternalHoleExclusionArea(const Polygons& outline, const int extruder_nr) const; + Shape getInternalHoleExclusionArea(const Shape& outline, const int extruder_nr) const; /*! * Generate a brim line with offset parameters given by \p offset from the \p starting_outlines and store it in the \ref storage. @@ -183,7 +183,7 @@ class SkirtBrim * \param[out] result Where to store the resulting brim line * \return The length of the added lines */ - coord_t generateOffset(const Offset& offset, Polygons& covered_area, std::vector& allowed_areas_per_extruder, SkirtBrimLine& result); + coord_t generateOffset(const Offset& offset, Shape& covered_area, std::vector& allowed_areas_per_extruder, MixedLinesSet& result); /*! * Generate a skirt of extruders which don't yet comply with the minimum length requirement. @@ -196,7 +196,7 @@ class SkirtBrim * \param[in,out] allowed_areas_per_extruder The difference between the machine areas and the \p covered_area * \param[in,out] total_length The total length of the brim lines for each extruder. */ - void generateSecondarySkirtBrim(Polygons& covered_area, std::vector& allowed_areas_per_extruder, std::vector& total_length); + void generateSecondarySkirtBrim(Shape& covered_area, std::vector& allowed_areas_per_extruder, std::vector& total_length); /*! * Generate the allowed areas for each extruder. Allowed areas represent where the brim/skirt is allowed to grow @@ -206,7 +206,7 @@ class SkirtBrim * \param[in] starting_outlines The previously generated starting outlines for each extruder * \return The list of allowed areas for each extruder */ - std::vector generateAllowedAreas(const std::vector& starting_outlines) const; + std::vector generateAllowedAreas(const std::vector& starting_outlines) const; public: /*! diff --git a/include/SupportInfillPart.h b/include/SupportInfillPart.h index 23a983276a..c864a67be6 100644 --- a/include/SupportInfillPart.h +++ b/include/SupportInfillPart.h @@ -6,9 +6,10 @@ #include +#include "geometry/Polygon.h" +#include "geometry/SingleShape.h" #include "utils/AABB.h" #include "utils/ExtrusionLine.h" -#include "utils/polygon.h" namespace cura @@ -25,23 +26,23 @@ namespace cura class SupportInfillPart { public: - PolygonsPart outline_; //!< The outline of the support infill area + SingleShape outline_; //!< The outline of the support infill area AABB outline_boundary_box_; //!< The boundary box for the infill area coord_t support_line_width_; //!< The support line width int inset_count_to_generate_; //!< The number of insets need to be generated from the outline. This is not the actual insets that will be generated. - std::vector> infill_area_per_combine_per_density_; //!< a list of separated sub-areas which requires different infill densities and combined thicknesses - // for infill_areas[x][n], x means the density level and n means the thickness + std::vector> infill_area_per_combine_per_density_; //!< a list of separated sub-areas which requires different infill densities and combined thicknesses + // for infill_areas[x][n], x means the density level and n means the thickness std::vector wall_toolpaths_; //!< Any walls go here, not in the areas, where they could be combined vertically (don't combine walls). Binned by inset_idx. coord_t custom_line_distance_; //!< The distance between support infill lines. 0 means use the default line distance instead. bool use_fractional_config_; //!< Request to use the configuration used to fill a partial layer height here, instead of the normal full layer height configuration. - SupportInfillPart(const PolygonsPart& outline, coord_t support_line_width, bool use_fractional_config, int inset_count_to_generate = 0, coord_t custom_line_distance = 0); + SupportInfillPart(const SingleShape& outline, coord_t support_line_width, bool use_fractional_config, int inset_count_to_generate = 0, coord_t custom_line_distance = 0); - const Polygons& getInfillArea() const; + const Shape& getInfillArea() const; }; -inline const Polygons& SupportInfillPart::getInfillArea() const +inline const Shape& SupportInfillPart::getInfillArea() const { // if there is no wall, we use the original outline as the infill area return outline_; diff --git a/include/TopSurface.h b/include/TopSurface.h index 77505bfa94..e54438904b 100644 --- a/include/TopSurface.h +++ b/include/TopSurface.h @@ -5,7 +5,7 @@ #define TOPSURFACE_H #include "GCodePathConfig.h" -#include "utils/polygon.h" //For the polygon areas. +#include "geometry/Shape.h" namespace cura { @@ -58,7 +58,7 @@ class TopSurface /*! * \brief The areas of top surface, for each layer. */ - Polygons areas; + Shape areas; }; } // namespace cura diff --git a/include/TreeModelVolumes.h b/include/TreeModelVolumes.h index 9c9013888d..0dee5f62c1 100644 --- a/include/TreeModelVolumes.h +++ b/include/TreeModelVolumes.h @@ -10,11 +10,11 @@ #include #include "TreeSupportSettings.h" +#include "geometry/Polygon.h" //For polygon parameters. #include "settings/EnumSettings.h" //To store whether X/Y or Z distance gets priority. #include "settings/types/LayerIndex.h" //Part of the RadiusLayerPair. #include "sliceDataStorage.h" #include "utils/Simplify.h" -#include "utils/polygon.h" //For polygon parameters. namespace cura { @@ -41,7 +41,7 @@ class TreeModelVolumes size_t current_mesh_idx, double progress_multiplier, double progress_offset, - const std::vector& additional_excluded_areas = std::vector()); + const std::vector& additional_excluded_areas = std::vector()); TreeModelVolumes(TreeModelVolumes&&) = default; TreeModelVolumes& operator=(TreeModelVolumes&&) = default; @@ -66,9 +66,9 @@ class TreeModelVolumes * \param radius The radius of the node of interest * \param layer_idx The layer of interest * \param min_xy_dist Is the minimum xy distance used. - * \return Polygons object + * \return Shape object */ - const Polygons& getCollision(coord_t radius, LayerIndex layer_idx, bool min_xy_dist = false); + const Shape& getCollision(coord_t radius, LayerIndex layer_idx, bool min_xy_dist = false); /*! * \brief Provides the areas that have to be avoided by the tree's branches to prevent collision with the model on this layer. Holes are removed. @@ -79,9 +79,9 @@ class TreeModelVolumes * \param radius The radius of the node of interest * \param layer_idx The layer of interest * \param min_xy_dist Is the minimum xy distance used. - * \return Polygons object + * \return Shape object */ - const Polygons& getCollisionHolefree(coord_t radius, LayerIndex layer_idx, bool min_xy_dist = false); + const Shape& getCollisionHolefree(coord_t radius, LayerIndex layer_idx, bool min_xy_dist = false); /*! @@ -89,9 +89,9 @@ class TreeModelVolumes * * The result is a 2D area that represents where if support were to be placed in and just dropped down it would not rest on support blocker. * \param layer_idx The layer of interest - * \return Polygons object + * \return Shape object */ - const Polygons& getAccumulatedPlaceable0(LayerIndex layer_idx); + const Shape& getAccumulatedPlaceable0(LayerIndex layer_idx); /*! * \brief Provides the areas that have to be avoided by the tree's branches @@ -108,24 +108,24 @@ class TreeModelVolumes * \param slow Is the propagation with the maximum move distance slow required. * \param to_model Does the avoidance allow good connections with the model. * \param min_xy_dist is the minimum xy distance used. - * \return Polygons object + * \return Shape object */ - const Polygons& getAvoidance(coord_t radius, LayerIndex layer_idx, AvoidanceType type, bool to_model = false, bool min_xy_dist = false); + const Shape& getAvoidance(coord_t radius, LayerIndex layer_idx, AvoidanceType type, bool to_model = false, bool min_xy_dist = false); /*! * \brief Provides the area represents all areas on the model where the branch does completely fit on the given layer. * \param radius The radius of the node of interest * \param layer_idx The layer of interest - * \return Polygons object + * \return Shape object */ - const Polygons& getPlaceableAreas(coord_t radius, LayerIndex layer_idx); + const Shape& getPlaceableAreas(coord_t radius, LayerIndex layer_idx); /*! * \brief Provides the area that represents the walls, as in the printed area, of the model. This is an abstract representation not equal with the outline. See * calculateWallRestrictions for better description. \param radius The radius of the node of interest. \param layer_idx The layer of interest. \param min_xy_dist is the minimum - * xy distance used. \return Polygons object + * xy distance used. \return Shape object */ - const Polygons& getWallRestriction(coord_t radius, LayerIndex layer_idx, bool min_xy_dist); + const Shape& getWallRestriction(coord_t radius, LayerIndex layer_idx, bool min_xy_dist); /*! * \brief Round \p radius upwards to either a multiple of radius_sample_resolution_ or a exponentially increasing value @@ -165,9 +165,9 @@ class TreeModelVolumes * \brief Extracts the relevant outline from a mesh * \param[in] mesh The mesh which outline will be extracted * \param layer_idx The layer which should be extracted from the mesh - * \return Polygons object representing the outline + * \return Shape object representing the outline */ - Polygons extractOutlineFromMesh(const SliceMeshStorage& mesh, LayerIndex layer_idx) const; + Shape extractOutlineFromMesh(const SliceMeshStorage& mesh, LayerIndex layer_idx) const; /*! @@ -243,7 +243,7 @@ class TreeModelVolumes calculateCollisionHolefree(std::deque{ RadiusLayerPair(key) }); } - Polygons safeOffset(const Polygons& me, coord_t distance, ClipperLib::JoinType jt, coord_t max_safe_step_distance, const Polygons& collision) const; + Shape safeOffset(const Shape& me, coord_t distance, ClipperLib::JoinType jt, coord_t max_safe_step_distance, const Shape& collision) const; /*! * \brief Creates the areas that have to be avoided by the tree's branches to prevent collision with the model. @@ -337,7 +337,7 @@ class TreeModelVolumes * \return A wrapped optional reference of the requested area (if it was found, an empty optional if nothing was found) */ template - const std::optional> getArea(const std::unordered_map& cache, const KEY key) const; + const std::optional> getArea(const std::unordered_map& cache, const KEY key) const; bool checkSettingsEquality(const Settings& me, const Settings& other) const; @@ -348,9 +348,9 @@ class TreeModelVolumes * * \return A wrapped optional reference of the requested area (if it was found, an empty optional if nothing was found) */ - LayerIndex getMaxCalculatedLayer(coord_t radius, const std::unordered_map& map) const; + LayerIndex getMaxCalculatedLayer(coord_t radius, const std::unordered_map& map) const; - static Polygons calculateMachineBorderCollision(const Polygons&& machine_border); + static Shape calculateMachineBorderCollision(const Shape&& machine_border); /*! * \brief The maximum distance that the center point of a tree branch may move in consecutive layers if it has to avoid the model. @@ -427,25 +427,25 @@ class TreeModelVolumes coord_t increase_until_radius_; /*! - * \brief Polygons representing the limits of the printable area of the + * \brief Shape representing the limits of the printable area of the * machine */ - Polygons machine_border_; + Shape machine_border_; /*! - * \brief Polygons representing the printable area of the machine + * \brief Shape representing the printable area of the machine */ - Polygons machine_area_; + Shape machine_area_; /*! * \brief Storage for layer outlines and the corresponding settings of the meshes grouped by meshes with identical setting. */ - std::vector>> layer_outlines_; + std::vector>> layer_outlines_; /*! * \brief Storage for areas that should be avoided, like support blocker or previous generated trees. */ - std::vector anti_overhang_; + std::vector anti_overhang_; /*! * \brief Radii that can be ignored by ceilRadius as they will never be requested. @@ -471,52 +471,52 @@ class TreeModelVolumes * (ie there is no difference in behaviour for the user between * calculating the values each time vs caching the results). */ - mutable std::unordered_map collision_cache_; + mutable std::unordered_map collision_cache_; std::unique_ptr critical_collision_cache_ = std::make_unique(); - mutable std::unordered_map collision_cache_holefree_; + mutable std::unordered_map collision_cache_holefree_; std::unique_ptr critical_collision_cache_holefree_ = std::make_unique(); - mutable std::unordered_map accumulated_placeables_cache_radius_0_; + mutable std::unordered_map accumulated_placeables_cache_radius_0_; std::unique_ptr critical_accumulated_placeables_cache_radius_0_ = std::make_unique(); - mutable std::unordered_map avoidance_cache_collision_; + mutable std::unordered_map avoidance_cache_collision_; std::unique_ptr critical_avoidance_cache_collision_ = std::make_unique(); - mutable std::unordered_map avoidance_cache_; + mutable std::unordered_map avoidance_cache_; std::unique_ptr critical_avoidance_cache_ = std::make_unique(); - mutable std::unordered_map avoidance_cache_slow_; + mutable std::unordered_map avoidance_cache_slow_; std::unique_ptr critical_avoidance_cache_slow_ = std::make_unique(); - mutable std::unordered_map avoidance_cache_to_model_; + mutable std::unordered_map avoidance_cache_to_model_; std::unique_ptr critical_avoidance_cache_to_model_ = std::make_unique(); - mutable std::unordered_map avoidance_cache_to_model_slow_; + mutable std::unordered_map avoidance_cache_to_model_slow_; std::unique_ptr critical_avoidance_cache_to_model_slow_ = std::make_unique(); - mutable std::unordered_map placeable_areas_cache_; + mutable std::unordered_map placeable_areas_cache_; std::unique_ptr critical_placeable_areas_cache_ = std::make_unique(); /*! * \brief Caches to avoid holes smaller than the radius until which the radius is always increased, as they are free of holes. Also called safe avoidances, as they are safe * regarding not running into holes. */ - mutable std::unordered_map avoidance_cache_hole_; + mutable std::unordered_map avoidance_cache_hole_; std::unique_ptr critical_avoidance_cache_holefree_ = std::make_unique(); - mutable std::unordered_map avoidance_cache_hole_to_model_; + mutable std::unordered_map avoidance_cache_hole_to_model_; std::unique_ptr critical_avoidance_cache_holefree_to_model_ = std::make_unique(); /*! * \brief Caches to represent walls not allowed to be passed over. */ - mutable std::unordered_map wall_restrictions_cache_; + mutable std::unordered_map wall_restrictions_cache_; std::unique_ptr critical_wall_restrictions_cache_ = std::make_unique(); // A different cache for min_xy_dist as the maximal safe distance an influence area can be increased(guaranteed overlap of two walls in consecutive layer) is much smaller when // min_xy_dist is used. This causes the area of the wall restriction to be thinner and as such just using the min_xy_dist wall restriction would be slower. - mutable std::unordered_map wall_restrictions_cache_min_; + mutable std::unordered_map wall_restrictions_cache_min_; std::unique_ptr critical_wall_restrictions_cache_min_ = std::make_unique(); std::unique_ptr critical_progress_ = std::make_unique(); diff --git a/include/TreeSupport.h b/include/TreeSupport.h index 61932f0231..a4f1be756a 100644 --- a/include/TreeSupport.h +++ b/include/TreeSupport.h @@ -10,11 +10,11 @@ #include "TreeSupportEnums.h" #include "TreeSupportSettings.h" #include "boost/functional/hash.hpp" // For combining hashes +#include "geometry/Polygon.h" #include "polyclipping/clipper.hpp" #include "settings/EnumSettings.h" #include "sliceDataStorage.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" namespace cura { @@ -42,12 +42,12 @@ constexpr auto SUPPORT_TREE_EXPONENTIAL_FACTOR = 1.5; constexpr size_t SUPPORT_TREE_PRE_EXPONENTIAL_STEPS = 1; constexpr coord_t SUPPORT_TREE_COLLISION_RESOLUTION = 500; // Only has an effect if SUPPORT_TREE_USE_EXPONENTIAL_COLLISION_RESOLUTION is false -using PropertyAreasUnordered = std::unordered_map; -using PropertyAreas = std::map; +using PropertyAreasUnordered = std::unordered_map; +using PropertyAreas = std::map; struct FakeRoofArea { - FakeRoofArea(Polygons area, coord_t line_distance, bool fractional) + FakeRoofArea(Shape area, coord_t line_distance, bool fractional) : area_(area) , line_distance_(line_distance) , fractional_(fractional) @@ -56,7 +56,7 @@ struct FakeRoofArea /*! * \brief Area that should be a fake roof. */ - Polygons area_; + Shape area_; /*! * \brief Distance between support lines @@ -185,10 +185,10 @@ class TreeSupport AreaIncreaseSettings settings, LayerIndex layer_idx, TreeSupportElement* parent, - const Polygons& relevant_offset, - Polygons& to_bp_data, - Polygons& to_model_data, - Polygons& increased, + const Shape& relevant_offset, + Shape& to_bp_data, + Shape& to_model_data, + Shape& increased, const coord_t overspeed, const bool mergelayer); @@ -262,7 +262,7 @@ class TreeSupport */ void generateBranchAreas( std::vector>& linear_data, - std::vector>& layer_tree_polygons, + std::vector>& layer_tree_polygons, const std::map& inverse_tree_order); /*! @@ -270,7 +270,7 @@ class TreeSupport * * \param layer_tree_polygons[in,out] Resulting branch areas with the layerindex they appear on. */ - void smoothBranchAreas(std::vector>& layer_tree_polygons); + void smoothBranchAreas(std::vector>& layer_tree_polygons); /*! * \brief Drop down areas that do rest non-gracefully on the model to ensure the branch actually rests on something. @@ -281,13 +281,13 @@ class TreeSupport * \param inverse_tree_order[in] A mapping that returns the child of every influence area. */ void dropNonGraciousAreas( - std::vector>& layer_tree_polygons, + std::vector>& layer_tree_polygons, const std::vector>& linear_data, - std::vector>>& dropped_down_areas, + std::vector>>& dropped_down_areas, const std::map& inverse_tree_order); - void filterFloatingLines(std::vector& support_layer_storage); + void filterFloatingLines(std::vector& support_layer_storage); /*! * \brief Generates Support Floor, ensures Support Roof can not cut of branches, and saves the branches as support to storage @@ -297,9 +297,9 @@ class TreeSupport * \param storage[in,out] The storage where the support should be stored. */ void finalizeInterfaceAndSupportAreas( - std::vector& support_layer_storage, - std::vector& support_roof_storage, - std::vector& support_layer_storage_fractional, + std::vector& support_layer_storage, + std::vector& support_roof_storage, + std::vector& support_layer_storage_fractional, SliceDataStorage& storage); /*! @@ -318,7 +318,7 @@ class TreeSupport /*! * \brief Areas that should have been support roof, but where the roof settings would not allow any lines to be generated. */ - std::vector additional_required_support_area; + std::vector additional_required_support_area; /*! * \brief Areas that use a higher density pattern of regular support to support the model (fake_roof). diff --git a/include/TreeSupportBaseCircle.h b/include/TreeSupportBaseCircle.h index fe872ddcc7..9010e066b3 100644 --- a/include/TreeSupportBaseCircle.h +++ b/include/TreeSupportBaseCircle.h @@ -7,8 +7,9 @@ #include +#include "geometry/Polygon.h" +#include "settings/types/Angle.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" namespace cura { diff --git a/include/TreeSupportElement.h b/include/TreeSupportElement.h index 83b47ba3fb..b633aa863c 100644 --- a/include/TreeSupportElement.h +++ b/include/TreeSupportElement.h @@ -10,9 +10,9 @@ #include "TreeModelVolumes.h" #include "TreeSupportBaseCircle.h" #include "TreeSupportEnums.h" +#include "geometry/Shape.h" #include "settings/types/LayerIndex.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" namespace cura { @@ -95,7 +95,7 @@ struct TreeSupportElement RecreateInfluenceLimitArea(); } - TreeSupportElement(const TreeSupportElement& elem, Polygons* newArea = nullptr) + TreeSupportElement(const TreeSupportElement& elem, Shape* newArea = nullptr) : // copy constructor with possibility to set a new area target_height_(elem.target_height_) , target_position_(elem.target_position_) @@ -277,7 +277,7 @@ struct TreeSupportElement * \brief The resulting influence area. * Will only be set in the results of createLayerPathing, and will be nullptr inside! */ - Polygons* area_; + Shape* area_; /*! * \brief The resulting center point around which a circle will be drawn later. @@ -351,7 +351,7 @@ struct TreeSupportElement /*! * \brief Area that influence area has to be inside to conform to influence_area_limit_range. */ - Polygons influence_area_limit_area_; + Shape influence_area_limit_area_; /*! * \brief Additional locations that the tip should reach @@ -402,7 +402,7 @@ struct TreeSupportElement Polygon circle; for (Point2LL corner : TreeSupportBaseCircle::getBaseCircle()) { - circle.add(p + corner * influence_area_limit_range_ / double(TreeSupportBaseCircle::base_radius)); + circle.push_back(p + corner * influence_area_limit_range_ / double(TreeSupportBaseCircle::base_radius)); } if (influence_area_limit_area_.empty()) { diff --git a/include/TreeSupportSettings.h b/include/TreeSupportSettings.h index b6b38dab4a..2e3162a71a 100644 --- a/include/TreeSupportSettings.h +++ b/include/TreeSupportSettings.h @@ -1,3 +1,4 @@ +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef TREESUPPORTSETTINGS_H @@ -10,8 +11,11 @@ #include "settings/EnumSettings.h" #include "settings/Settings.h" #include "settings/types/Angle.h" +#include "settings/types/Ratio.h" #include "utils/Coord_t.h" #include "utils/Simplify.h" +#include "utils/math.h" + namespace cura { diff --git a/include/TreeSupportTipGenerator.h b/include/TreeSupportTipGenerator.h index de518f5682..e483500ea0 100644 --- a/include/TreeSupportTipGenerator.h +++ b/include/TreeSupportTipGenerator.h @@ -11,11 +11,13 @@ #include "TreeSupportEnums.h" #include "TreeSupportSettings.h" #include "boost/functional/hash.hpp" // For combining hashes +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/Polygon.h" #include "polyclipping/clipper.hpp" #include "settings/EnumSettings.h" #include "sliceDataStorage.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" namespace cura { @@ -40,7 +42,7 @@ class TreeSupportTipGenerator SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector>& move_bounds, - std::vector& additional_support_areas, + std::vector& additional_support_areas, std::vector>& placed_fake_roof_areas); private: @@ -63,7 +65,7 @@ class TreeSupportTipGenerator * \param layer_idx[in] The current layer. * \return All lines of the \p polylines object, with information for each point regarding in which avoidance it is currently valid in. */ - std::vector convertLinesToInternal(Polygons polylines, LayerIndex layer_idx); + std::vector convertLinesToInternal(const OpenLinesSet& polylines, LayerIndex layer_idx); /*! * \brief Converts lines in internal format into a Polygons object representing these lines. @@ -71,7 +73,7 @@ class TreeSupportTipGenerator * \param lines[in] The lines that will be converted. * \return All lines of the \p lines object as a Polygons object. */ - Polygons convertInternalToLines(std::vector lines); + OpenLinesSet convertInternalToLines(std::vector lines); /*! * \brief Returns a function, evaluating if a point has to be added now. Required for a splitLines call in generateInitialAreas. @@ -102,7 +104,7 @@ class TreeSupportTipGenerator * \param enforce_distance[in] If points should not be added if they are closer than distance to other points. * \return A Polygons object containing the evenly spaced points. Does not represent an area, more a collection of points on lines. */ - Polygons ensureMaximumDistancePolyline(const Polygons& input, coord_t distance, size_t min_points, bool enforce_distance) const; + OpenLinesSet ensureMaximumDistancePolyline(const OpenLinesSet& input, coord_t distance, size_t min_points, bool enforce_distance) const; /*! * \brief Creates a valid CrossInfillProvider @@ -121,7 +123,7 @@ class TreeSupportTipGenerator * \param result[out] The dropped overhang ares * \param roof[in] Whether the result is for roof generation. */ - void dropOverhangAreas(const SliceMeshStorage& mesh, std::vector& result, bool roof); + void dropOverhangAreas(const SliceMeshStorage& mesh, std::vector& result, bool roof); /*! * \brief Calculates which areas should be supported with roof, and saves these in roof support_roof_drawn @@ -174,7 +176,7 @@ class TreeSupportTipGenerator * \param storage[in] Background storage, required for adding roofs. * \param additional_support_areas[in] Areas that should have been roofs, but are now support, as they would not generate any lines as roof. */ - void removeUselessAddedPoints(std::vector>& move_bounds, SliceDataStorage& storage, std::vector& additional_support_areas); + void removeUselessAddedPoints(std::vector>& move_bounds, SliceDataStorage& storage, std::vector& additional_support_areas); /*! * \brief Contains config settings to avoid loading them in every function. This was done to improve readability of the code. @@ -296,17 +298,17 @@ class TreeSupportTipGenerator /*! * \brief Areas that will be saved as support roof */ - std::vector support_roof_drawn_; + std::vector support_roof_drawn_; /*! * \brief Areas that require fractional roof above it. */ - std::vector support_roof_drawn_fractional_; + std::vector support_roof_drawn_fractional_; /*! * \brief Areas that will be saved as support roof, originating from tips being replaced with roof areas. */ - std::vector roof_tips_drawn_; + std::vector roof_tips_drawn_; std::mutex critical_move_bounds_; std::mutex critical_roof_tips_; diff --git a/include/TreeSupportUtils.h b/include/TreeSupportUtils.h index ca301b6ad4..d11364fd2d 100644 --- a/include/TreeSupportUtils.h +++ b/include/TreeSupportUtils.h @@ -12,12 +12,15 @@ #include "TreeSupportEnums.h" #include "TreeSupportSettings.h" #include "boost/functional/hash.hpp" // For combining hashes +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" #include "infill.h" #include "polyclipping/clipper.hpp" #include "settings/EnumSettings.h" #include "sliceDataStorage.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" namespace cura { @@ -29,20 +32,14 @@ class TreeSupportUtils * \brief Adds the implicit line from the last vertex of a Polygon to the first one. * * \param poly[in] The Polygons object, of which its lines should be extended. - * \return A Polygons object with implicit line from the last vertex of a Polygon to the first one added. + * \return A Polygons object with explicit line from the last vertex of a Polygon to the first one added. */ - static Polygons toPolylines(const Polygons& poly) + static OpenLinesSet toPolylines(const Shape& poly) { - Polygons result; + OpenLinesSet result; for (const auto& path : poly) { - Polygon part; - for (const auto& p : path) - { - part.add(p); - } - part.add(path[0]); - result.add(part); + result.push_back(path.toPseudoOpenPolyline()); } return result; } @@ -54,29 +51,30 @@ class TreeSupportUtils * \param toolpaths[in] The toolpaths. * \return A Polygons object. */ - [[nodiscard]] static Polygons toPolylines(const std::vector toolpaths) + [[nodiscard]] static OpenLinesSet toPolylines(const std::vector toolpaths) { - Polygons result; - for (VariableWidthLines lines : toolpaths) + OpenLinesSet result; + for (const VariableWidthLines& lines : toolpaths) { - for (ExtrusionLine line : lines) + for (const ExtrusionLine& line : lines) { - if (line.size() == 0) + if (line.empty()) { continue; } - Polygon result_line; - for (ExtrusionJunction junction : line) + + OpenPolyline result_line; + for (const ExtrusionJunction& junction : line) { - result_line.add(junction.p_); + result_line.push_back(junction.p_); } if (line.is_closed_) { - result_line.add(line[0].p_); + result_line.push_back(line[0].p_); } - result.add(result_line); + result.push_back(result_line); } } return result; @@ -96,8 +94,8 @@ class TreeSupportUtils * todo doku * \return A Polygons object that represents the resulting infill lines. */ - [[nodiscard]] static Polygons generateSupportInfillLines( - const Polygons& area, + [[nodiscard]] static OpenLinesSet generateSupportInfillLines( + const Shape& area, const TreeSupportSettings& config, bool roof, LayerIndex layer_idx, @@ -106,7 +104,7 @@ class TreeSupportUtils bool include_walls, bool generate_support_supporting = false) { - Polygons gaps; + Shape gaps; // As we effectivly use lines to place our supportPoints we may use the Infill class for it, while not made for it, it works perfectly. const EFillMethod pattern = generate_support_supporting ? EFillMethod::GRID : roof ? config.roof_pattern : config.support_pattern; @@ -158,21 +156,21 @@ class TreeSupportUtils zag_skip_count, pocket_size); - Polygons areas; - Polygons lines; + Shape areas; + OpenLinesSet lines; roof_computation.generate(toolpaths, areas, lines, config.settings, layer_idx, SectionType::SUPPORT, cross_fill_provider); - lines.add(toPolylines(areas)); - lines.add(toPolylines(toolpaths)); + lines.push_back(toPolylines(areas)); + lines.push_back(toPolylines(toolpaths)); return lines; } /*! - * \brief Unions two Polygons. Ensures that if the input is non empty that the output also will be non empty. + * \brief Unions two Shape. Ensures that if the input is non empty that the output also will be non empty. * \param first[in] The first Polygon. * \param second[in] The second Polygon. - * \return The union of both Polygons + * \return The union of both Shape */ - [[nodiscard]] static Polygons safeUnion(const Polygons& first, const Polygons& second = Polygons()) + [[nodiscard]] static Shape safeUnion(const Shape& first, const Shape& second = Shape()) { // The unionPolygons function can slowly remove Polygons under certain circumstances, because of rounding issues (Polygons that have a thin area). // This does not cause a problem when actually using it on large areas, but as influence areas (representing centerpoints) can be very thin, this does occur so this ugly @@ -187,33 +185,33 @@ class TreeSupportUtils */ const bool was_empty = first.empty() && second.empty(); - Polygons result = first.unionPolygons(second); + Shape result = first.unionPolygons(second); if (result.empty() && ! was_empty) // Some error occurred. { spdlog::warn("Caught an area destroying union, enlarging areas a bit."); // Just take the few lines we have, and offset them a tiny bit. Needs to be offsetPolylines, as offset may already have problems with the area. - return toPolylines(first).offsetPolyLine(2).unionPolygons(toPolylines(second).offsetPolyLine(2)); + return toPolylines(first).offset(2).unionPolygons(toPolylines(second).offset(2)); } return result; } /*! * \brief Offsets (increases the area of) a polygons object in multiple steps to ensure that it does not lag through over a given obstacle. - * \param me[in] Polygons object that has to be offset. + * \param me[in] Shape object that has to be offset. * \param distance[in] The distance by which me should be offset. Expects values >=0. * \param collision[in] The area representing obstacles. * \param last_step_offset_without_check[in] The most it is allowed to offset in one step. - * \param min_amount_offset[in] How many steps have to be done at least. As this uses round offset this increases the amount of vertices, which may be required if Polygons get + * \param min_amount_offset[in] How many steps have to be done at least. As this uses round offset this increases the amount of vertices, which may be required if Shape get * very small. Required as arcTolerance is not exposed in offset, which should result with a similar result, benefit may be eliminated by simplifying. \param * min_offset_per_step Don't get below this amount of offset per step taken. Fine-tune tradeoff between speed and accuracy. \param simplifier[in] Pointer to Simplify object if - * the offset operation also simplify the Polygon. Improves performance. \return The resulting Polygons object. + * the offset operation also simplify the Polygon. Improves performance. \return The resulting Shape object. */ - [[nodiscard]] static Polygons safeOffsetInc( - const Polygons& me, + [[nodiscard]] static Shape safeOffsetInc( + const Shape& me, coord_t distance, - const Polygons& collision, + const Shape& collision, coord_t safe_step_size, coord_t last_step_offset_without_check, size_t min_amount_offset, @@ -221,7 +219,7 @@ class TreeSupportUtils Simplify* simplifier) { bool do_final_difference = last_step_offset_without_check == 0; - Polygons ret = safeUnion(me); // Ensure sane input. + Shape ret = safeUnion(me); // Ensure sane input. if (distance == 0) { return (do_final_difference ? ret.difference(collision) : ret).unionPolygons(); @@ -291,14 +289,14 @@ class TreeSupportUtils * \param max_allowed_distance[in] The maximum distance a point may be moved. If not possible the point will be moved as far as possible in the direction of the outside of the * provided area. \return A Polyline object containing the moved points. */ - [[nodiscard]] static Polygons movePointsOutside(const Polygons& polylines, const Polygons& area, coord_t max_allowed_distance) + [[nodiscard]] static OpenLinesSet movePointsOutside(const OpenLinesSet& polylines, const Shape& area, coord_t max_allowed_distance) { - Polygons result; + OpenLinesSet result; - for (auto line : polylines) + for (const OpenPolyline& line : polylines) { - Polygon next_line; - for (Point2LL p : line) + OpenPolyline next_line; + for (const Point2LL& p : line) { if (area.inside(p)) { @@ -306,30 +304,30 @@ class TreeSupportUtils PolygonUtils::moveOutside(area, next_outside); if (vSize2(p - next_outside) < max_allowed_distance * max_allowed_distance) { - next_line.add(next_outside); + next_line.push_back(next_outside); } else // move point as far as allowed. { double max_partial_move_proportion = double(max_allowed_distance) / double(vSize(p - next_outside)); next_outside = p + (next_outside - p) * max_partial_move_proportion; - next_line.add(next_outside); + next_line.push_back(next_outside); } } else { - next_line.add(p); + next_line.push_back(p); } } if (next_line.size() > 0) { - result.add(next_line); + result.push_back(next_line); } } return result; } - [[nodiscard]] static VariableWidthLines polyLineToVWL(const Polygons& polylines, coord_t line_width) + [[nodiscard]] static VariableWidthLines polyLineToVWL(const Shape& polylines, coord_t line_width) { VariableWidthLines result; for (auto path : polylines) diff --git a/include/WallToolPaths.h b/include/WallToolPaths.h index 07b67ebfcd..ff1df0a63a 100644 --- a/include/WallToolPaths.h +++ b/include/WallToolPaths.h @@ -7,9 +7,9 @@ #include #include "BeadingStrategy/BeadingStrategyFactory.h" +#include "geometry/Polygon.h" #include "settings/Settings.h" #include "utils/ExtrusionLine.h" -#include "utils/polygon.h" #include "utils/section_type.h" namespace cura @@ -26,7 +26,7 @@ class WallToolPaths * \param settings The settings as provided by the user */ WallToolPaths( - const Polygons& outline, + const Shape& outline, const coord_t nominal_bead_width, const size_t inset_count, const coord_t wall_0_inset, @@ -44,7 +44,7 @@ class WallToolPaths * \param settings The settings as provided by the user */ WallToolPaths( - const Polygons& outline, + const Shape& outline, const coord_t bead_width_0, const coord_t bead_width_x, const size_t inset_count, @@ -90,7 +90,7 @@ class WallToolPaths * If there are no walls, the outline will be returned. * \return The inner contour of the generated walls. */ - const Polygons& getInnerContour(); + const Shape& getInnerContour(); /*! * Removes empty paths from the toolpaths @@ -122,7 +122,7 @@ class WallToolPaths static void simplifyToolPaths(std::vector& toolpaths, const Settings& settings); private: - const Polygons& outline_; // toolpaths_; // +#include #include //Loading JSON documents to get settings from them. #include //To store the command line arguments. +#include #include //To store the command line arguments. #include "Communication.h" //The class we're implementing. @@ -15,6 +17,9 @@ namespace cura { class Settings; +using setting_map = std::unordered_map; +using container_setting_map = std::unordered_map; + /* * \brief When slicing via the command line, interprets the command line * arguments to initiate a slice. @@ -107,14 +112,14 @@ class CommandLine : public Communication * * The command line doesn't show any layer view so this is ignored. */ - void sendPolygon(const PrintFeatureType&, const ConstPolygonRef&, const coord_t&, const coord_t&, const Velocity&) override; + void sendPolygon(const PrintFeatureType&, const Polygon&, const coord_t&, const coord_t&, const Velocity&) override; /* * \brief Send a polygon to show it in layer view. * * The command line doesn't show any layer view so this is ignored. */ - void sendPolygons(const PrintFeatureType&, const Polygons&, const coord_t&, const coord_t&, const Velocity&) override; + void sendPolygons(const PrintFeatureType&, const Shape&, const coord_t&, const coord_t&, const Velocity&) override; /* * \brief Show an estimate of how long the print would take and how much @@ -212,6 +217,20 @@ class CommandLine : public Communication * \return The first definition file that matches the definition ID. */ static std::string findDefinitionFile(const std::string& definition_id, const std::vector& search_directories); + + /* + * \brief Read the resolved JSON values from a file. + * \param element The path to the file to read the JSON values from. + * \return The resolved JSON values. + */ + static std::optional readResolvedJsonValues(const std::filesystem::path& json_filename); + + /* + * \brief Read the resolved JSON values from a document. + * \param document The document to read the JSON values from. + * \return The resolved JSON values. + */ + static std::optional readResolvedJsonValues(const rapidjson::Document& document); }; } // namespace cura diff --git a/include/communication/Communication.h b/include/communication/Communication.h index 0b8a509a5b..9bc990c11e 100644 --- a/include/communication/Communication.h +++ b/include/communication/Communication.h @@ -4,16 +4,16 @@ #ifndef COMMUNICATION_H #define COMMUNICATION_H +#include "geometry/Point2LL.h" #include "settings/types/LayerIndex.h" #include "settings/types/Velocity.h" -#include "utils/Point2LL.h" namespace cura { // Some forward declarations to increase compilation speed. enum class PrintFeatureType : unsigned char; -class Polygons; -class ConstPolygonRef; +class Shape; +class Polygon; class ExtruderTrain; /* @@ -76,7 +76,7 @@ class Communication * \param line_thickness The thickness (in the Z direction) of the polygons. * \param velocity The velocity of printing these polygons. */ - virtual void sendPolygons(const PrintFeatureType& type, const Polygons& polygons, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity) = 0; + virtual void sendPolygons(const PrintFeatureType& type, const Shape& polygons, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity) = 0; /* * \brief Send a polygon to the user to visualise. @@ -90,7 +90,7 @@ class Communication * \param line_thickness The thickness (in the Z direction) of the polygon. * \param velocity The velocity of printing this polygon. */ - virtual void sendPolygon(const PrintFeatureType& type, const ConstPolygonRef& polygon, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity) = 0; + virtual void sendPolygon(const PrintFeatureType& type, const Polygon& polygon, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity) = 0; /* * \brief Send a line to the user to visualise. diff --git a/include/gcodeExport.h b/include/gcodeExport.h index 5edd5dcdda..9d91325d97 100644 --- a/include/gcodeExport.h +++ b/include/gcodeExport.h @@ -11,21 +11,21 @@ #include // for stream.str() #include +#include "geometry/Point2LL.h" #include "settings/EnumSettings.h" #include "settings/Settings.h" //For MAX_EXTRUDERS. #include "settings/types/LayerIndex.h" #include "settings/types/Temperature.h" //Bed temperature. #include "settings/types/Velocity.h" -#include "sliceDataStorage.h" #include "timeEstimate.h" #include "utils/AABB3D.h" //To track the used build volume for the Griffin header. #include "utils/NoCopy.h" -#include "utils/Point2LL.h" namespace cura { class RetractionConfig; +class SliceDataStorage; struct WipeScriptConfig; // The GCodeExport class writes the actual GCode. This is the only class that knows how GCode looks and feels. diff --git a/include/geometry/ClosedLinesSet.h b/include/geometry/ClosedLinesSet.h new file mode 100644 index 0000000000..381628f767 --- /dev/null +++ b/include/geometry/ClosedLinesSet.h @@ -0,0 +1,22 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_CLOSED_LINES_SET_H +#define GEOMETRY_CLOSED_LINES_SET_H + +namespace cura +{ + +template +class LinesSet; +class ClosedPolyline; + +/*! + * \brief Convenience definition for a container that can hold only closed polylines. This makes it + * explicit what the lines actually represent. + */ +using ClosedLinesSet = LinesSet; + +} // namespace cura + +#endif // GEOMETRY_CLOSED_LINES_SET_H diff --git a/include/geometry/ClosedPolyline.h b/include/geometry/ClosedPolyline.h new file mode 100644 index 0000000000..8f4bc3e753 --- /dev/null +++ b/include/geometry/ClosedPolyline.h @@ -0,0 +1,140 @@ +// Copyright (c) 2023 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_CLOSED_POLYLINE_H +#define GEOMETRY_CLOSED_POLYLINE_H + +#include "geometry/Polyline.h" + +namespace cura +{ + +class OpenPolyline; + +/*! @brief This describes a polyline which forms a closed path. + * @sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#closedpolyline + * + * The path may be closed: + * * Explicitely, which means the last point is at the same position as the first point. + * In this case, in order to iterate over the segments, you just have to iterate over + * the actual points. + * * Implicitely, which means the last and first point are at different positions. In this + * case, to iterate over the segments, you have to consider an additional segment + * between the last and first point + * + * The difference is made because it is easier to iterate over segments when the path is + * explicitely closed, but ClipperLib uses implicitely closed paths. It is also a bit healthier + * to use implicitely closed because there is no risk that the first and last point become different + */ +class ClosedPolyline : public Polyline +{ +private: + bool explicitely_closed_{ false }; + +public: + ClosedPolyline() = default; + + /*! + * \brief Builds an empty closed polyline + * \param explicitely_closed Indicates whether the line will be explicitely closed + * \warning By default, the line is tagged as explicitely closed. We need this default + * constructor in various places, but be careful that the interpretation of the points + * added later will depend on this. + */ + explicit ClosedPolyline(const bool explicitely_closed) + : explicitely_closed_{ explicitely_closed } + { + } + + /*! \brief Creates a copy of the given polyline */ + ClosedPolyline(const ClosedPolyline& other) = default; + + /*! \brief Constructor that takes ownership of the inner points list from the given polyline */ + ClosedPolyline(ClosedPolyline&& other) = default; + + /*! + * \brief Constructor with a points initializer list, provided for convenience + * \param explicitely_closed Specify whether the given points form an explicitely closed line + */ + ClosedPolyline(const std::initializer_list& initializer, bool explicitely_closed) + : Polyline{ initializer } + , explicitely_closed_{ explicitely_closed } + { + } + + /*! + * \brief Constructor with an existing list of points + * \param explicitely_closed Specify whether the given points form an explicitely closed line + */ + explicit ClosedPolyline(const ClipperLib::Path& points, bool explicitely_closed) + : Polyline{ points } + , explicitely_closed_{ explicitely_closed } + { + } + + /*! + * \brief Constructor that takes ownership of the given list of points + * \param explicitely_closed Specify whether the given points form an explicitely closed line + */ + explicit ClosedPolyline(ClipperLib::Path&& points, bool explicitely_closed) + : Polyline{ std::move(points) } + , explicitely_closed_{ explicitely_closed } + { + } + + ~ClosedPolyline() override = default; + + /*! @see Polyline::hasClosingSegment() */ + [[nodiscard]] bool hasClosingSegment() const override + { + return ! explicitely_closed_; + } + + /*! @see Polyline::addClosingSegment() */ + [[nodiscard]] size_t segmentsCount() const override; + + /*! @see Polyline::isValid() */ + [[nodiscard]] bool isValid() const override; + + ClosedPolyline& operator=(const ClosedPolyline& other) = default; + + ClosedPolyline& operator=(ClosedPolyline&& other) = default; + + [[nodiscard]] bool isExplicitelyClosed() const + { + return explicitely_closed_; + } + + /*! + * \brief Sets whether the points set is to be treated as explicitely or implicitely closed + * \warning This does not actually changes the points set, only the interpretation of it will + * change. So use this method only if you really know what you are doing. + */ + void setExplicitelyClosed(bool explicitely_closed) + { + explicitely_closed_ = explicitely_closed; + } + + /*! + * Clipper function. + * Returns false if outside, true if inside; if the point lies exactly on the border, will return 'border_result'. + * + * http://www.angusj.com/delphi/clipper/documentation/Docs/Units/ClipperLib/Functions/PointInPolygon.htm + */ + [[nodiscard]] bool inside(const Point2LL& p, bool border_result = false) const; + + [[nodiscard]] bool inside(const ClipperLib::Path& polygon) const; + + /*! + * \brief Converts the closed polyline to an open polyline which happens to have its end and start points at the same + * position, making it a pseudo-closed polyline. Although this should never be required in practice, there + * are many places in the code where this is done because historically we wouldn't make a clear difference + * between open and closed polylines + * \return An open polyline instance, with the end point at the same position of the start point + */ + [[nodiscard]] OpenPolyline toPseudoOpenPolyline() const; +}; + +} // namespace cura + +#endif // GEOMETRY_CLOSED_POLYLINE_H diff --git a/include/geometry/LinesSet.h b/include/geometry/LinesSet.h new file mode 100644 index 0000000000..1e885cb89c --- /dev/null +++ b/include/geometry/LinesSet.h @@ -0,0 +1,299 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_LINES_SET_H +#define GEOMETRY_LINES_SET_H + +#include + +#include + +#include "geometry/OpenLinesSet.h" +#include "geometry/Point2LL.h" + +namespace cura +{ + +class Shape; +template +class LinesSet; +class OpenPolyline; + +enum class CheckNonEmptyParam +{ + OnlyIfValid, + OnlyIfNotEmpty, + EvenIfEmpty +}; + +/*! + * \brief Base class for all geometry containers representing a set of polylines. + * \sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#linesset + */ +template +class LinesSet +{ +private: + std::vector lines_; + +public: + // Required for some std calls as a container + using value_type = LineType; + using iterator = typename std::vector::iterator; + using const_iterator = typename std::vector::const_iterator; + + /*! \brief Builds an empty set */ + LinesSet() noexcept = default; + + virtual ~LinesSet() = default; + + /*! \brief Creates a copy of the given lines set */ + LinesSet(const LinesSet& other) = default; + + /*! \brief Constructor that takes the inner lines list from the given set */ + LinesSet(LinesSet&& other) = default; + + /*! \brief Constructor with an existing set of lines */ + explicit LinesSet(const std::vector& lines) + : lines_(lines) + { + } + + /*! \brief Constructor that takes ownership of the data from the given set of lines */ + explicit LinesSet(std::vector&& lines) + : lines_(std::move(lines)) + { + } + + /*! + * \brief Constructor that takes ownership of the data from the given set of lines + * \warning This constructor is actually only defined for a LinesSet containing OpenPolyline + * objects, because closed ones require an additional argument + */ + template + requires std::is_same_v + explicit LinesSet(ClipperLib::Paths&& paths) + { + reserve(paths.size()); + for (ClipperLib::Path& path : paths) + { + lines_.emplace_back(std::move(path)); + } + } + + const std::vector& getLines() const + { + return lines_; + } + + std::vector& getLines() + { + return lines_; + } + + void setLines(std::vector&& lines) + { + lines_ = lines; + } + + const_iterator begin() const + { + return lines_.begin(); + } + + iterator begin() + { + return lines_.begin(); + } + + const_iterator end() const + { + return lines_.end(); + } + + iterator end() + { + return lines_.end(); + } + + const LineType& back() const + { + return lines_.back(); + } + + LineType& back() + { + return lines_.back(); + } + + const LineType& front() const + { + return lines_.front(); + } + + LineType& front() + { + return lines_.front(); + } + + /*! + * \brief Pushes the given line at the end of the set + * \param check_non_empty Indicates whether we should check for the line to be non-empty before adding it + */ + void push_back(const LineType& line, CheckNonEmptyParam check_non_empty = CheckNonEmptyParam::EvenIfEmpty); + + /*! + * \brief Pushes the given line at the end of the set and takes ownership of the inner data + * \param check_non_empty Indicates whether we should check for the line to be non-empty before adding it + */ + void push_back(LineType&& line, CheckNonEmptyParam check_non_empty = CheckNonEmptyParam::EvenIfEmpty); + + /*! \brief Pushes an entier set at the end and takes ownership of the inner data */ + template + void push_back(LinesSet&& lines_set); + + /*! \brief Pushes an entier set at the end */ + void push_back(const LinesSet& other) + { + lines_.insert(lines_.end(), other.lines_.begin(), other.lines_.end()); + } + + void pop_back() + { + lines_.pop_back(); + } + + [[nodiscard]] size_t size() const + { + return lines_.size(); + } + + [[nodiscard]] bool empty() const + { + return lines_.empty(); + } + + void reserve(size_t size) + { + lines_.reserve(size); + } + + void resize(size_t size) + { + lines_.resize(size); + } + + void clear() + { + lines_.clear(); + } + + void emplace_back(auto&&... args) + { + lines_.emplace_back(std::forward(args)...); + } + + iterator erase(const_iterator first, const_iterator last) + { + return lines_.erase(first, last); + } + + LinesSet& operator=(const LinesSet& other) = default; + + LinesSet& operator=(LinesSet&& other) noexcept = default; + + LineType& operator[](size_t index) + { + return lines_[index]; + } + + const LineType& operator[](size_t index) const + { + return lines_[index]; + } + + LineType& newLine() + { + lines_.emplace_back(); + return lines_.back(); + } + + /*! \brief Return the amount of points in all lines */ + [[nodiscard]] size_t pointCount() const; + + /*! + * Remove a line from the list and move the last line to its place + * \warning changes the order of the lines! + */ + void removeAt(size_t index); + + /*! \brief Add a simple line consisting of two points */ + void addSegment(const Point2LL& from, const Point2LL& to); + + /*! \brief Get the total length of all the lines */ + [[nodiscard]] coord_t length() const; + + void splitIntoSegments(OpenLinesSet& result) const; + [[nodiscard]] OpenLinesSet splitIntoSegments() const; + + /*! \brief Removes overlapping consecutive line segments which don't delimit a positive area */ + void removeDegenerateVerts(); + + [[nodiscard]] Shape offset(coord_t distance, ClipperLib::JoinType join_type = ClipperLib::jtMiter, double miter_limit = 1.2) const; + + /*! + * Utility method for creating the tube (or 'donut') of a shape. + * + * \param inner_offset Offset relative to the original shape-outline towards the inside of the + * shape. Sort-of like a negative normal offset, except it's the offset part that's kept, + * not the shape. + * \param outer_offset Offset relative to the original shape-outline towards the outside of the + * shape. Comparable to normal offset. + * \return The resulting polygons. + */ + [[nodiscard]] Shape createTubeShape(const coord_t inner_offset, const coord_t outer_offset) const; + + void translate(const Point2LL& delta); + + /*! + * \brief Utility method to add all the lines to a ClipperLib::Clipper object + * \note This method needs to be public but you shouldn't need to use it from outside + */ + void addPaths(ClipperLib::Clipper& clipper, ClipperLib::PolyType poly_typ) const; + + /*! + * \brief Utility method to add all the lines to a ClipperLib::ClipperOffset object + * \note This method needs to be public but you shouldn't need to use it from outside + */ + void addPaths(ClipperLib::ClipperOffset& clipper, ClipperLib::JoinType joint_type, ClipperLib::EndType end_type) const; + + /*! + * \brief Display operator, useful for debugging/testing + */ + template + friend std::basic_ostream& operator<<(std::basic_ostream& os, const LinesSet& lines) + { + os << "LinesSet["; + + for (const LineType& line : lines | ranges::views::drop(1)) + { + os << line; + os << ", "; + } + + if (lines.size() > 1) + { + os << lines.back(); + } + + os << "]"; + return os; + } + +private: + bool checkAdd(const LineType& line, CheckNonEmptyParam check_non_empty); +}; + +} // namespace cura + +#endif // GEOMETRY_LINES_SET_H diff --git a/include/geometry/MixedLinesSet.h b/include/geometry/MixedLinesSet.h new file mode 100644 index 0000000000..f692f47661 --- /dev/null +++ b/include/geometry/MixedLinesSet.h @@ -0,0 +1,80 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_MIXED_LINES_SET_H +#define GEOMETRY_MIXED_LINES_SET_H + +#include + +#include "geometry/ClosedLinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "utils/Coord_t.h" + +namespace cura +{ + +class Polyline; +class ClosedPolyline; +class Polygon; +class Shape; + +using PolylinePtr = std::shared_ptr; +using OpenPolylinePtr = std::shared_ptr; + +/*! + * \brief Convenience definition for a container that can hold any type of polyline. + * \sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#mixedlinesset + */ +class MixedLinesSet : public std::vector +{ +public: + /*! + * \brief Computes the offset of all the polylines contained in the set. The polylines may + * be of different types, and polylines are polygons are treated differently. + * \param distance The distance to increase the polylines from, or decrase if negative + * \param join_type The type of tip joint to be used (for open polylines only) + * \return A shape containing the offsetted polylines. This may contain many unjoined polygons, + * but no overlapping ones. + */ + [[nodiscard]] Shape offset(coord_t distance, ClipperLib::JoinType join_type = ClipperLib::jtMiter, double miter_limit = 1.2) const; + + /*! @brief Adds a copy of the given polyline to the set */ + void push_back(const OpenPolyline& line); + + /*! @brief Adds a copy of the given polyline to the set */ + void push_back(const Polygon& line); + + /*! @brief Adds a copy of the given polyline to the set */ + void push_back(OpenPolyline&& line); + + /*! @brief Adds a copy of the given polyline to the set */ + void push_back(ClosedPolyline&& line); + + /*! @brief Adds the given shared pointer to the set. The pointer reference count will be incremeted but no data is actually copied. */ + void push_back(const OpenPolylinePtr& line); + + /*! @brief Adds the given shared pointer to the set. The pointer reference count will be incremeted but no data is actually copied. */ + void push_back(const PolylinePtr& line); + + /*! @brief Adds a copy of all the polygons contained in the shape */ + void push_back(const Shape& shape); + + /*! @brief Adds a copy of all the polygons contained in the set */ + void push_back(const LinesSet& lines_set); + + /*! @brief Adds a copy of all the polylines contained in the set */ + void push_back(const OpenLinesSet& lines_set); + + /*! @brief Adds a copy of all the polylines contained in the set */ + void push_back(OpenLinesSet&& lines_set); + + /*! @brief Adds a copy of all the polylines contained in the set */ + void push_back(ClosedLinesSet&& lines_set); + + /*! \brief Computes the total lenght of all the polylines in the set */ + [[nodiscard]] coord_t length() const; +}; + +} // namespace cura + +#endif // GEOMETRY_MIXED_LINES_SET_H diff --git a/include/geometry/OpenLinesSet.h b/include/geometry/OpenLinesSet.h new file mode 100644 index 0000000000..d3e18902f5 --- /dev/null +++ b/include/geometry/OpenLinesSet.h @@ -0,0 +1,22 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_OPEN_LINES_SET_H +#define GEOMETRY_OPEN_LINES_SET_H + +namespace cura +{ + +template +class LinesSet; +class OpenPolyline; + +/*! + * \brief Convenience definition for a container that can hold only open polylines. This makes it + * explicit what the lines actually represent. + */ +using OpenLinesSet = LinesSet; + +} // namespace cura + +#endif // GEOMETRY_OPEN_LINES_SET_H diff --git a/include/geometry/OpenPolyline.h b/include/geometry/OpenPolyline.h new file mode 100644 index 0000000000..2a6b0faef2 --- /dev/null +++ b/include/geometry/OpenPolyline.h @@ -0,0 +1,94 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_OPEN_POLYLINE_H +#define GEOMETRY_OPEN_POLYLINE_H + +#include "geometry/Polyline.h" + +namespace cura +{ + +/*! + * @brief Represents a polyline that is explicitely not closed + * @sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#openpolyline + * @note Open polylines are sometimes used to represent actually closed polylines. In this case + * the first and last point are at the very same position. This should not be done, but + * it exists all around the engine for historical reasons. The behavior is however deprecated + * and should not be used in the future + */ +class OpenPolyline : public Polyline +{ +public: + /*! @brief Builds an empty polyline */ + OpenPolyline() = default; + + /*! + * \brief Creates a copy of the given polyline + * \warning A copy of the points list is made, so this constructor is somehow "slow" + */ + OpenPolyline(const OpenPolyline& other) = default; + + /*! + * \brief Constructor that takes ownership of the inner points list from the given polyline + * \warning This constructor is fast because it does not allocate data, but it will clear + * the source object + */ + OpenPolyline(OpenPolyline&& other) = default; + + /*! + * \brief Constructor with a points initializer list, provided for convenience + * \warning A copy of the points list is made, so this constructor is somehow "slow" + */ + OpenPolyline(const std::initializer_list& initializer) + : Polyline{ initializer } + { + } + + /*! + * \brief Constructor with an existing list of points + * \warning A copy of the points list is made, so this constructor is somehow "slow" + */ + explicit OpenPolyline(const ClipperLib::Path& points) + : Polyline{ points } + { + } + + /*! + * \brief Constructor that takes ownership of the given list of points + * \warning This constructor is fast because it does not allocate data, but it will clear + * the source object + */ + explicit OpenPolyline(ClipperLib::Path&& points) + : Polyline{ std::move(points) } + { + } + + ~OpenPolyline() override = default; + + /*! @see Polyline::hasClosingSegment() */ + [[nodiscard]] bool hasClosingSegment() const override + { + return false; // Definitely not + } + + /*! @see Polyline::segmentsCount() */ + [[nodiscard]] size_t segmentsCount() const override + { + return size() > 1 ? size() - 1 : 0; + } + + /*! @see Polyline::isValid() */ + [[nodiscard]] bool isValid() const override + { + return size() >= 2; + } + + OpenPolyline& operator=(OpenPolyline&& other) noexcept = default; + + OpenPolyline& operator=(const OpenPolyline& other) = default; +}; + +} // namespace cura + +#endif // GEOMETRY_OPEN_POLYLINE_H diff --git a/include/geometry/PartsView.h b/include/geometry/PartsView.h new file mode 100644 index 0000000000..6cfb9bf4c6 --- /dev/null +++ b/include/geometry/PartsView.h @@ -0,0 +1,68 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_PARTS_VIEW_H +#define GEOMETRY_PARTS_VIEW_H + +#include + +namespace cura +{ + +class Shape; +class SingleShape; + +/*! + * Extension of vector> which is similar to a vector of PolygonParts, except the base of the container is indices to polygons into the original Polygons, + * instead of the polygons themselves + */ +class PartsView : public std::vector> +{ +public: + Shape& polygons_; + + PartsView() = delete; + + explicit PartsView(Shape& polygons) + : polygons_{ polygons } + { + } + + PartsView(PartsView&& parts_view) = default; + PartsView(const PartsView& parts_view) = default; + + ~PartsView() = default; + + PartsView& operator=(PartsView&& parts_view) = delete; + PartsView& operator=(const PartsView& parts_view) = delete; + + /*! + * Get the index of the SingleShape of which the polygon with index \p poly_idx is part. + * + * \param poly_idx The index of the polygon in \p polygons + * \param boundary_poly_idx Optional output parameter: The index of the boundary polygon of the part in \p polygons + * \return The SingleShape containing the polygon with index \p poly_idx + */ + size_t getPartContaining(size_t poly_idx, size_t* boundary_poly_idx = nullptr) const; + + /*! + * Assemble the SingleShape of which the polygon with index \p poly_idx is part. + * + * \param poly_idx The index of the polygon in \p polygons + * \param boundary_poly_idx Optional output parameter: The index of the boundary polygon of the part in \p polygons + * \return The SingleShape containing the polygon with index \p poly_idx + */ + SingleShape assemblePartContaining(size_t poly_idx, size_t* boundary_poly_idx = nullptr) const; + + /*! + * Assemble the SingleShape of which the polygon with index \p poly_idx is part. + * + * \param part_idx The index of the part + * \return The SingleShape with index \p poly_idx + */ + [[nodiscard]] SingleShape assemblePart(size_t part_idx) const; +}; + +} // namespace cura + +#endif // GEOMETRY_PARTS_VIEW_H diff --git a/include/geometry/Point2LL.h b/include/geometry/Point2LL.h new file mode 100644 index 0000000000..08a556a961 --- /dev/null +++ b/include/geometry/Point2LL.h @@ -0,0 +1,262 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#ifndef UTILS_INT_POINT_H +#define UTILS_INT_POINT_H + +/** +The integer point classes are used as soon as possible and represent microns in 2D or 3D space. +Integer points are used to avoid floating point rounding errors, and because ClipperLib uses them. +*/ +#define INLINE static inline + +#include +#include +#include + +#include "geometry/Point3LL.h" +#include "utils/types/generic.h" + +#ifdef __GNUC__ +#define DEPRECATED(func) func __attribute__((deprecated)) +#elif defined(_MSC_VER) +#define DEPRECATED(func) __declspec(deprecated) func +#else +#pragma message("WARNING: You need to implement DEPRECATED for this compiler") +#define DEPRECATED(func) func +#endif + + +namespace cura +{ + +/* 64bit Points are used mostly throughout the code, these are the 2D points from ClipperLib */ +using Point2LL = ClipperLib::IntPoint; + +#define POINT_MIN std::numeric_limits::min() +#define POINT_MAX std::numeric_limits::max() + +static Point2LL no_point(std::numeric_limits::min(), std::numeric_limits::min()); + +/* Extra operators to make it easier to do math with the 64bit Point objects */ +INLINE Point2LL operator-(const Point2LL& p0) +{ + return { -p0.X, -p0.Y }; +} + +INLINE Point2LL operator+(const Point2LL& p0, const Point2LL& p1) +{ + return { p0.X + p1.X, p0.Y + p1.Y }; +} + +INLINE Point2LL operator-(const Point2LL& p0, const Point2LL& p1) +{ + return { p0.X - p1.X, p0.Y - p1.Y }; +} + +INLINE Point2LL operator*(const Point2LL& p0, const coord_t i) +{ + return { p0.X * i, p0.Y * i }; +} + +template // Use only for numeric types. +INLINE Point2LL operator*(const Point2LL& p0, const T i) +{ + return { std::llrint(static_cast(p0.X) * i), std::llrint(static_cast(p0.Y) * i) }; +} + +template +INLINE Point2LL operator*(const T i, const Point2LL& p0) +{ + return p0 * i; +} + +INLINE Point2LL operator/(const Point2LL& p0, const coord_t i) +{ + return { p0.X / i, p0.Y / i }; +} + +INLINE Point2LL operator/(const Point2LL& p0, const Point2LL& p1) +{ + return { p0.X / p1.X, p0.Y / p1.Y }; +} + +INLINE Point2LL operator%(const Point2LL& p0, const coord_t i) +{ + return { p0.X % i, p0.Y % i }; +} + +INLINE Point2LL& operator+=(Point2LL& p0, const Point2LL& p1) +{ + p0.X += p1.X; + p0.Y += p1.Y; + return p0; +} + +INLINE Point2LL& operator-=(Point2LL& p0, const Point2LL& p1) +{ + p0.X -= p1.X; + p0.Y -= p1.Y; + return p0; +} + +INLINE bool operator<(const Point2LL& p0, const Point2LL& p1) +{ + return p0.X < p1.X || (p0.X == p1.X && p0.Y < p1.Y); +} + +/* ***** NOTE ***** + TL;DR: DO NOT implement operators *= and /= because of the default values in ClipperLib::IntPoint's constructor. + + We DO NOT implement operators *= and /= because the class Point is essentially a ClipperLib::IntPoint and it has a + constructor IntPoint(int x = 0, int y = 0), and this causes problems. If you implement *= as *=(int) and when you + do "Point a = a * 5", you probably intend to do "a.x *= 5" and "a.y *= 5", but with that constructor, it will create + an IntPoint(5, y = 0) and you end up with wrong results. + */ + +// INLINE bool operator==(const Point& p0, const Point& p1) { return p0.X==p1.X&&p0.Y==p1.Y; } +// INLINE bool operator!=(const Point& p0, const Point& p1) { return p0.X!=p1.X||p0.Y!=p1.Y; } + +INLINE coord_t vSize2(const Point2LL& p0) +{ + return p0.X * p0.X + p0.Y * p0.Y; +} + +INLINE double vSize2f(const Point2LL& p0) +{ + return static_cast(p0.X) * static_cast(p0.X) + static_cast(p0.Y) * static_cast(p0.Y); +} + +INLINE bool shorterThen(const Point2LL& p0, const coord_t len) +{ + if (p0.X > len || p0.X < -len) + { + return false; + } + if (p0.Y > len || p0.Y < -len) + { + return false; + } + return vSize2(p0) <= len * len; +} + +INLINE bool shorterThan(const Point2LL& p0, const coord_t len) +{ + return shorterThen(p0, len); +} + +INLINE coord_t vSize(const Point2LL& p0) +{ + return std::llrint(sqrt(static_cast(vSize2(p0)))); +} + +INLINE double vSizeMM(const Point2LL& p0) +{ + double fx = INT2MM(p0.X); + double fy = INT2MM(p0.Y); + return std::sqrt(fx * fx + fy * fy); +} + +INLINE Point2LL normal(const Point2LL& p0, coord_t length) +{ + const coord_t len{ vSize(p0) }; + if (len < 1) + { + return { length, 0 }; + } + return p0 * length / len; +} + +INLINE Point2LL turn90CCW(const Point2LL& p0) +{ + return { -p0.Y, p0.X }; +} + +INLINE Point2LL rotate(const Point2LL& p0, double angle) +{ + const double cos_component = std::cos(angle); + const double sin_component = std::sin(angle); + const auto x = static_cast(p0.X); + const auto y = static_cast(p0.Y); + return { std::llrint(cos_component * x - sin_component * y), std::llrint(sin_component * x + cos_component * y) }; +} + +INLINE coord_t dot(const Point2LL& p0, const Point2LL& p1) +{ + return p0.X * p1.X + p0.Y * p1.Y; +} + +INLINE coord_t cross(const Point2LL& p0, const Point2LL& p1) +{ + return p0.X * p1.Y - p0.Y * p1.X; +} + +INLINE double angle(const Point2LL& p) +{ + double angle = std::atan2(p.X, p.Y) / std::numbers::pi * 180.0; + if (angle < 0.0) + { + angle += 360.0; + } + return angle; +} + +// Identity function, used to be able to make templated algorithms where the input is sometimes points, sometimes things that contain or can be converted to points. +INLINE const Point2LL& make_point(const Point2LL& p) +{ + return p; +} + +inline Point3LL operator+(const Point3LL& p3, const Point2LL& p2) +{ + return { p3.x_ + p2.X, p3.y_ + p2.Y, p3.z_ }; +} + +inline Point3LL& operator+=(Point3LL& p3, const Point2LL& p2) +{ + p3.x_ += p2.X; + p3.y_ += p2.Y; + return p3; +} + +inline Point2LL operator+(const Point2LL& p2, const Point3LL& p3) +{ + return { p3.x_ + p2.X, p3.y_ + p2.Y }; +} + +inline Point3LL operator-(const Point3LL& p3, const Point2LL& p2) +{ + return { p3.x_ - p2.X, p3.y_ - p2.Y, p3.z_ }; +} + +inline Point3LL& operator-=(Point3LL& p3, const Point2LL& p2) +{ + p3.x_ -= p2.X; + p3.y_ -= p2.Y; + return p3; +} + +inline Point2LL operator-(const Point2LL& p2, const Point3LL& p3) +{ + return { p2.X - p3.x_, p2.Y - p3.y_ }; +} + +} // namespace cura + +namespace std +{ +template<> +struct hash +{ + size_t operator()(const cura::Point2LL& pp) const noexcept + { + static int prime = 31; + int result = 89; + result = static_cast(result * prime + pp.X); + result = static_cast(result * prime + pp.Y); + return static_cast(result); + } +}; +} // namespace std + +#endif // UTILS_INT_POINT_H diff --git a/include/utils/Point3LL.h b/include/geometry/Point3LL.h similarity index 64% rename from include/utils/Point3LL.h rename to include/geometry/Point3LL.h index c992bcb3d5..fd220b05ef 100644 --- a/include/utils/Point3LL.h +++ b/include/geometry/Point3LL.h @@ -1,17 +1,17 @@ // Copyright (c) 2018 Ultimaker B.V. // CuraEngine is released under the terms of the AGPLv3 or higher. -#ifndef UTILS_POINT3_H -#define UTILS_POINT3_H +#ifndef GEOMETRY_POINT3LL_H +#define GEOMETRY_POINT3LL_H #include #include //For sqrt. #include //Auto-serialization. #include //For numeric_limits::min and max. -#include //For int32_t and int64_t. #include // for operations on any arithmetic number type -#include "Coord_t.h" +#include "utils/Coord_t.h" +#include "utils/types/generic.h" namespace cura @@ -20,10 +20,12 @@ namespace cura class Point3LL { public: - coord_t x_, y_, z_; - Point3LL() - { - } + coord_t x_{}; + coord_t y_{}; + coord_t z_{}; + + Point3LL() = default; + Point3LL(const coord_t x, const coord_t y, const coord_t z) : x_(x) , y_(y) @@ -31,41 +33,53 @@ class Point3LL { } + Point3LL(Point3LL&& point) = default; + Point3LL(const Point3LL& point) = default; + Point3LL& operator=(const Point3LL& point) = default; + Point3LL& operator=(Point3LL&& point) = default; + + virtual ~Point3LL() = default; + Point3LL operator+(const Point3LL& p) const; Point3LL operator-() const; Point3LL operator-(const Point3LL& p) const; Point3LL operator*(const Point3LL& p) const; //!< Element-wise multiplication. For dot product, use .dot()! Point3LL operator/(const Point3LL& p) const; - template::value, num_t>::type> - Point3LL operator*(const num_t i) const + + template + Point3LL operator*(const T& i) const { - return Point3LL(std::llround(static_cast(x_) * i), std::llround(static_cast(y_) * i), std::llround(static_cast(z_) * i)); + return { std::llround(static_cast(x_) * i), std::llround(static_cast(y_) * i), std::llround(static_cast(z_) * i) }; } - template::value, num_t>::type> - Point3LL operator/(const num_t i) const + + template + Point3LL operator/(const T& i) const { - return Point3LL(x_ / i, y_ / i, z_ / i); + return { x_ / i, y_ / i, z_ / i }; } - template::value, num_t>::type> - Point3LL operator%(const num_t i) const + + template + Point3LL operator%(const T& i) const { - return Point3LL(x_ % i, y_ % i, z_ % i); + return { x_ % i, y_ % i, z_ % i }; } Point3LL& operator+=(const Point3LL& p); Point3LL& operator-=(const Point3LL& p); Point3LL& operator*=(const Point3LL& p); Point3LL& operator/=(const Point3LL& p); - template::value, num_t>::type> - Point3LL& operator*=(const num_t i) + + template + Point3LL& operator*=(const T i) { x_ *= i; y_ *= i; z_ *= i; return *this; } - template::value, num_t>::type> - Point3LL& operator/=(const num_t i) + + template + Point3LL& operator/=(const T i) { x_ /= i; y_ /= i; @@ -73,9 +87,7 @@ class Point3LL return *this; } - bool operator==(const Point3LL& p) const; - bool operator!=(const Point3LL& p) const; - + auto operator<=>(const Point3LL&) const = default; template friend std::basic_ostream& operator<<(std::basic_ostream& os, const Point3LL& p) @@ -83,38 +95,47 @@ class Point3LL return os << "(" << p.x_ << ", " << p.y_ << ", " << p.z_ << ")"; } - - coord_t max() const + [[nodiscard]] coord_t max() const { if (x_ > y_ && x_ > z_) + { return x_; + } if (y_ > z_) + { return y_; + } return z_; } - bool testLength(coord_t len) const + [[nodiscard]] bool testLength(coord_t len) const { if (x_ > len || x_ < -len) + { return false; + } if (y_ > len || y_ < -len) + { return false; + } if (z_ > len || z_ < -len) + { return false; + } return vSize2() <= len * len; } - coord_t vSize2() const + [[nodiscard]] coord_t vSize2() const { return x_ * x_ + y_ * y_ + z_ * z_; } - coord_t vSize() const + [[nodiscard]] coord_t vSize() const { return std::llrint(sqrt(static_cast(vSize2()))); } - double vSizeMM() const + [[nodiscard]] double vSizeMM() const { double fx = INT2MM(x_); double fy = INT2MM(y_); @@ -122,7 +143,7 @@ class Point3LL return sqrt(fx * fx + fy * fy + fz * fz); } - coord_t dot(const Point3LL& p) const + [[nodiscard]] coord_t dot(const Point3LL& p) const { return x_ * p.x_ + y_ * p.y_ + z_ * p.z_; } @@ -153,8 +174,8 @@ class Point3LL */ static Point3LL no_point3(std::numeric_limits::min(), std::numeric_limits::min(), std::numeric_limits::min()); -template::value, num_t>::type> -inline Point3LL operator*(const num_t i, const Point3LL& rhs) +template +inline Point3LL operator*(const T i, const Point3LL& rhs) { return rhs * i; } @@ -167,7 +188,7 @@ namespace std template<> struct hash { - size_t operator()(const cura::Point3LL& pp) const + size_t operator()(const cura::Point3LL& pp) const noexcept { static int prime = 31; int result = 89; @@ -180,4 +201,4 @@ struct hash } // namespace std -#endif // UTILS_POINT3_H +#endif // GEOMETRY_POINT3LL_H diff --git a/include/geometry/Point3Matrix.h b/include/geometry/Point3Matrix.h new file mode 100644 index 0000000000..bbe11d6918 --- /dev/null +++ b/include/geometry/Point3Matrix.h @@ -0,0 +1,92 @@ +// Copyright (c) 2020 Ultimaker B.V. +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#ifndef GEOMETRY_POINT3_MATRIX_H +#define GEOMETRY_POINT3_MATRIX_H + +#include + +#include "geometry/Point3LL.h" +#include "geometry/PointMatrix.h" + +namespace cura +{ + +class Point3Matrix +{ +public: + std::array matrix{ 1, 0, 0, 0, 1, 0, 0, 0, 1 }; + + Point3Matrix() noexcept = default; + + /*! + * Initializes the top left corner with the values of \p b + * and the rest as if it's a unit matrix + */ + explicit Point3Matrix(const PointMatrix& b) + { + matrix.at(0) = b.matrix.at(0); + matrix.at(1) = b.matrix.at(1); + matrix.at(2) = 0; + matrix.at(3) = b.matrix.at(2); + matrix.at(4) = b.matrix.at(3); + matrix.at(5) = 0; + matrix.at(6) = 0; + matrix.at(7) = 0; + matrix.at(8) = 1; + } + + Point3Matrix(Point3Matrix&& point3_matrix) = default; + Point3Matrix(const Point3Matrix& point3_matrix) = default; + virtual ~Point3Matrix() = default; + + Point3Matrix& operator=(Point3Matrix&& point3_matrix) = default; + Point3Matrix& operator=(const Point3Matrix& point3_matrix) = default; + + [[nodiscard]] Point3LL apply(const Point3LL& p) const + { + const auto x = static_cast(p.x_); + const auto y = static_cast(p.y_); + const auto z = static_cast(p.z_); + return { std::llrint(x * matrix.at(0) + y * matrix.at(1) + z * matrix.at(2)), + std::llrint(x * matrix.at(3) + y * matrix.at(4) + z * matrix.at(5)), + std::llrint(x * matrix.at(6) + y * matrix.at(7) + z * matrix.at(8)) }; + } + + /*! + * Apply matrix to vector as homogeneous coordinates. + */ + [[nodiscard]] Point2LL apply(const Point2LL& p) const + { + Point3LL result = apply(Point3LL(p.X, p.Y, 1)); + return { result.x_ / result.z_, result.y_ / result.z_ }; + } + + static Point3Matrix translate(const Point2LL& p) + { + Point3Matrix ret; // uniform matrix + ret.matrix.at(2) = static_cast(p.X); + ret.matrix.at(5) = static_cast(p.Y); + return ret; + } + + [[nodiscard]] Point3Matrix compose(const Point3Matrix& b) const + { + Point3Matrix ret; + for (int outx = 0; outx < 3; outx++) + { + for (int outy = 0; outy < 3; outy++) + { + ret.matrix.at(outy * 3 + outx) = 0; + for (int in = 0; in < 3; in++) + { + ret.matrix.at(outy * 3 + outx) += matrix.at(outy * 3 + in) * b.matrix.at(in * 3 + outx); + } + } + } + return ret; + } +}; + +} // namespace cura +#endif // GEOMETRY_POINT3_MATRIX_H diff --git a/include/geometry/PointMatrix.h b/include/geometry/PointMatrix.h new file mode 100644 index 0000000000..1a22a74f06 --- /dev/null +++ b/include/geometry/PointMatrix.h @@ -0,0 +1,81 @@ +// Copyright (c) 2020 Ultimaker B.V. +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#ifndef GEOMETRY_POINT_MATRIX_H +#define GEOMETRY_POINT_MATRIX_H + +#include +#include + +#include "geometry/Point2LL.h" + + +namespace cura +{ + +class PointMatrix +{ +public: + std::array matrix{ 1, 0, 0, 1 }; + + PointMatrix() noexcept = default; + + explicit PointMatrix(double rotation) + { + rotation = rotation / 180 * std::numbers::pi; + matrix.at(0) = std::cos(rotation); + matrix.at(1) = -std::sin(rotation); + matrix.at(2) = -matrix.at(1); + matrix.at(3) = matrix.at(0); + } + + explicit PointMatrix(const Point2LL& p) + { + matrix.at(0) = static_cast(p.X); + matrix.at(1) = static_cast(p.Y); + double f = std::sqrt((matrix.at(0) * matrix.at(0)) + (matrix.at(1) * matrix.at(1))); + matrix.at(0) /= f; + matrix.at(1) /= f; + matrix.at(2) = -matrix.at(1); + matrix.at(3) = matrix.at(0); + } + + static PointMatrix scale(double s) + { + PointMatrix ret; + ret.matrix.at(0) = s; + ret.matrix.at(3) = s; + return ret; + } + + [[nodiscard]] Point2LL apply(const Point2LL& p) const + { + const auto x = static_cast(p.X); + const auto y = static_cast(p.Y); + return { std::llrint(x * matrix.at(0) + y * matrix.at(1)), std::llrint(x * matrix.at(2) + y * matrix.at(3)) }; + } + + /*! + * \warning only works on a rotation matrix! Output is incorrect for other types of matrix + */ + [[nodiscard]] Point2LL unapply(const Point2LL& p) const + { + const auto x = static_cast(p.X); + const auto y = static_cast(p.Y); + return { std::llrint(x * matrix.at(0) + y * matrix.at(2)), std::llrint(x * matrix.at(1) + y * matrix.at(3)) }; + } + + [[nodiscard]] PointMatrix inverse() const + { + PointMatrix ret; + double det = matrix.at(0) * matrix.at(3) - matrix.at(1) * matrix.at(2); + ret.matrix.at(0) = matrix.at(3) / det; + ret.matrix.at(1) = -matrix.at(1) / det; + ret.matrix.at(2) = -matrix.at(2) / det; + ret.matrix.at(3) = matrix.at(0) / det; + return ret; + } +}; + +} // namespace cura +#endif // GEOMETRY_POINT_MATRIX_H diff --git a/include/geometry/PointsSet.h b/include/geometry/PointsSet.h new file mode 100644 index 0000000000..f61329929d --- /dev/null +++ b/include/geometry/PointsSet.h @@ -0,0 +1,220 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_POINTS_SET_H +#define GEOMETRY_POINTS_SET_H + +#include "geometry/Point2LL.h" +#include "utils/Coord_t.h" + +namespace cura +{ + +class PointMatrix; +class Point3Matrix; + +const static int clipper_init = (0); +#define NO_INDEX (std::numeric_limits::max()) + +/*! + * \brief Base class for all geometry containers representing a set of points. + * \sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#pointsset + */ +class PointsSet +{ +private: + ClipperLib::Path points_; + +public: + // Required for some std calls as a container + using value_type = Point2LL; + using iterator = typename std::vector::iterator; + using const_iterator = typename std::vector::const_iterator; + using reverse_iterator = typename std::vector::reverse_iterator; + using const_reverse_iterator = typename std::vector::const_reverse_iterator; + + /*! \brief Builds an empty set */ + PointsSet() = default; + + /*! \brief Creates a copy of the given points set */ + PointsSet(const PointsSet& other) = default; + + /*! \brief Constructor that takes ownership of the inner points from the given set */ + PointsSet(PointsSet&& other) = default; + + /*! \brief Constructor with a points initializer list, provided for convenience" */ + PointsSet(const std::initializer_list& initializer); + + virtual ~PointsSet() = default; + + /*! \brief Constructor with an existing list of points */ + explicit PointsSet(const ClipperLib::Path& points); + + /*! \brief Constructor that takes ownership of the given list of points */ + explicit PointsSet(ClipperLib::Path&& points); + + [[nodiscard]] const ClipperLib::Path& getPoints() const + { + return points_; + } + + ClipperLib::Path& getPoints() + { + return points_; + } + + void setPoints(ClipperLib::Path&& points) + { + points_ = points; + } + + [[nodiscard]] size_t size() const + { + return points_.size(); + } + + void push_back(const Point2LL& point) + { + points_.push_back(point); + } + + void emplace_back(auto&&... args) + { + points_.emplace_back(std::forward(args)...); + } + + void pop_back() + { + points_.pop_back(); + } + + void insert(auto&&... args) + { + points_.insert(std::forward(args)...); + } + + [[nodiscard]] const_iterator begin() const + { + return points_.begin(); + } + + iterator begin() + { + return points_.begin(); + } + + [[nodiscard]] const_iterator end() const + { + return points_.end(); + } + + iterator end() + { + return points_.end(); + } + + [[nodiscard]] const_reverse_iterator rbegin() const + { + return points_.rbegin(); + } + + reverse_iterator rbegin() + { + return points_.rbegin(); + } + + [[nodiscard]] const_reverse_iterator rend() const + { + return points_.rend(); + } + + reverse_iterator rend() + { + return points_.rend(); + } + + [[nodiscard]] const Point2LL& front() const + { + return points_.front(); + } + + Point2LL& front() + { + return points_.front(); + } + + [[nodiscard]] const Point2LL& back() const + { + return points_.back(); + } + + Point2LL& back() + { + return points_.back(); + } + + [[nodiscard]] const Point2LL& at(const size_t pos) const + { + return points_.at(pos); + } + + Point2LL& at(const size_t pos) + { + return points_.at(pos); + } + + [[nodiscard]] bool empty() const + { + return points_.empty(); + } + + void resize(const size_t size) + { + points_.resize(size); + } + + void reserve(const size_t size) + { + points_.reserve(size); + } + + void clear() + { + points_.clear(); + } + + Point2LL& operator[](size_t index) + { + return points_[index]; + } + + const Point2LL& operator[](size_t index) const + { + return points_[index]; + } + + PointsSet& operator=(const PointsSet& other) = default; + + PointsSet& operator=(PointsSet&& other) = default; + + /*! + * \brief Translate all the points in some direction. + * \param translation The direction in which to move the points + */ + void translate(const Point2LL& translation); + + /*! \brief Apply a matrix to each vertex in this set */ + void applyMatrix(const PointMatrix& matrix); + void applyMatrix(const Point3Matrix& matrix); + + /*! \brief Display operator, useful for debugging/testing */ + template + friend std::basic_ostream& operator<<(std::basic_ostream& os, const PointsSet& polygon) + { + return os << "PointsSet(" << polygon.getPoints() << ")"; + } +}; + +} // namespace cura + +#endif // GEOMETRY_POINTS_SET_H diff --git a/include/geometry/Polygon.h b/include/geometry/Polygon.h new file mode 100644 index 0000000000..987851ab8d --- /dev/null +++ b/include/geometry/Polygon.h @@ -0,0 +1,199 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_POLYGON_H +#define GEOMETRY_POLYGON_H + +#include "geometry/ClosedPolyline.h" + +namespace cura +{ + +class Shape; +class ListPolyIt; +class AngleDegrees; + +/*! + * \brief A Polygon is a specific type of polyline, for which we consider that the "inside" part of + * the line forms a surface + * \sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#polygon + */ +class Polygon : public ClosedPolyline +{ +public: + Polygon() = default; + + /*! + * \brief Builds an empty polygon + * \param explicitely_closed Indicates whether the contour line will be explicitely closed + * \warning By default, the contour line is tagged as explicitely closed. We need this default + * constructor in various places, but be careful that the interpretation of the points + * added later will depend on this. + */ + explicit Polygon(const bool explicitely_closed) + : ClosedPolyline{ explicitely_closed } + { + } + + /*! \brief Creates a copy of the given polygon */ + Polygon(const Polygon& other) = default; + + /*! \brief Constructor that takes ownership of the inner points list from the given polygon */ + Polygon(Polygon&& other) = default; + + /*! + * \brief Constructor with a points initializer list, provided for convenience + * \param explicitely_closed Specify whether the given points form an explicitely closed line + */ + Polygon(const std::initializer_list& initializer, const bool explicitely_closed) + : ClosedPolyline{ initializer, explicitely_closed } + { + } + + /*! + * \brief Constructor with an existing list of points + * \param explicitely_closed Specify whether the given points form an explicitely closed line + */ + explicit Polygon(const ClipperLib::Path& points, const bool explicitely_closed) + : ClosedPolyline{ points, explicitely_closed } + { + } + + /*! + * \brief Constructor that takes ownership of the given list of points + * \param explicitely_closed Specify whether the given points form an explicitely closed line + */ + explicit Polygon(ClipperLib::Path&& points, const bool explicitely_closed) + : ClosedPolyline{ std::move(points), explicitely_closed } + { + } + + ~Polygon() override = default; + + Polygon& operator=(const Polygon& other) = default; + + Polygon& operator=(Polygon&& other) noexcept = default; + + /*! + * \brief Compute the morphological intersection between this polygon and another. + * \param other The polygon with which to intersect this polygon. + * \note The result may consist of multiple polygons, if you have bad luck. + */ + [[nodiscard]] Shape intersection(const Polygon& other) const; + + [[nodiscard]] double area() const + { + return ClipperLib::Area(getPoints()); + } + + [[nodiscard]] Point2LL centerOfMass() const; + + [[nodiscard]] Shape offset(int distance, ClipperLib::JoinType join_type = ClipperLib::jtMiter, double miter_limit = 1.2) const; + + /*! + * Smooth out small perpendicular segments and store the result in \p result. + * Smoothing is performed by removing the inner most vertex of a line segment smaller than \p remove_length + * which has an angle with the next and previous line segment smaller than roughly 150* + * + * Note that in its current implementation this function doesn't remove line segments with an angle smaller than 30* + * Such would be the case for an N shape. + * + * \param remove_length The length of the largest segment removed + * \param result (output) The result polygon, assumed to be empty + */ + void smooth(int remove_length, Polygon& result) const; + + /*! + * Smooth out sharp inner corners, by taking a shortcut which bypasses the corner + * + * \param angle The maximum angle of inner corners to be smoothed out + * \param shortcut_length The desired length of the shortcut line segment introduced (shorter shortcuts may be unavoidable) + * \param result The resulting polygon + */ + void smoothOutward(const AngleDegrees angle, int shortcut_length, Polygon& result) const; + + /*! + * Smooth out the polygon and store the result in \p result. + * Smoothing is performed by removing vertices for which both connected line segments are smaller than \p remove_length + * + * \param remove_length The length of the largest segment removed + * \param result (output) The result polygon, assumed to be empty + */ + void smooth2(int remove_length, Polygon& result) const; + + /*! + * Smooth out a simple corner consisting of two linesegments. + * + * Auxiliary function for \ref smooth_outward + * + * \param p0 The point before the corner + * \param p1 The corner + * \param p2 The point after the corner + * \param p0_it Iterator to the point before the corner + * \param p1_it Iterator to the corner + * \param p2_it Iterator to the point after the corner + * \param v10 Vector from \p p1 to \p p0 + * \param v12 Vector from \p p1 to \p p2 + * \param v02 Vector from \p p0 to \p p2 + * \param shortcut_length The desired length ofthe shortcutting line + * \param cos_angle The cosine on the angle in L 012 + */ + static void smoothCornerSimple( + const Point2LL& p0, + const Point2LL& p1, + const Point2LL& p2, + const ListPolyIt& p0_it, + const ListPolyIt& p1_it, + const ListPolyIt& p2_it, + const Point2LL& v10, + const Point2LL& v12, + const Point2LL& v02, + const int64_t shortcut_length, + double cos_angle); + + /*! + * Smooth out a complex corner where the shortcut bypasses more than two line segments + * + * Auxiliary function for \ref smooth_outward + * + * \warning This function might try to remove the whole polygon + * Error code -1 means the whole polygon should be removed (which means it is a hole polygon) + * + * \param p1 The corner point + * \param[in,out] p0_it Iterator to the last point checked before \p p1 to consider cutting off + * \param[in,out] p2_it Iterator to the last point checked after \p p1 to consider cutting off + * \param shortcut_length The desired length ofthe shortcutting line + * \return Whether this whole polygon whould be removed by the smoothing + */ + static bool smoothCornerComplex(const Point2LL& p1, ListPolyIt& p0_it, ListPolyIt& p2_it, const int64_t shortcut_length); + + /*! + * Try to take a step away from the corner point in order to take a bigger shortcut. + * + * Try to take the shortcut from a place as far away from the corner as the place we are taking the shortcut to. + * + * Auxiliary function for \ref smooth_outward + * + * \param[in] p1 The corner point + * \param[in] shortcut_length2 The square of the desired length ofthe shortcutting line + * \param[in,out] p0_it Iterator to the previously checked point somewhere beyond \p p1. Updated for the next iteration. + * \param[in,out] p2_it Iterator to the previously checked point somewhere before \p p1. Updated for the next iteration. + * \param[in,out] forward_is_blocked Whether trying another step forward is blocked by the smoothing outward condition. Updated for the next iteration. + * \param[in,out] backward_is_blocked Whether trying another step backward is blocked by the smoothing outward condition. Updated for the next iteration. + * \param[in,out] forward_is_too_far Whether trying another step forward is blocked by the shortcut length condition. Updated for the next iteration. + * \param[in,out] backward_is_too_far Whether trying another step backward is blocked by the shortcut length condition. Updated for the next iteration. + */ + static void smoothOutwardStep( + const Point2LL& p1, + const int64_t shortcut_length2, + ListPolyIt& p0_it, + ListPolyIt& p2_it, + bool& forward_is_blocked, + bool& backward_is_blocked, + bool& forward_is_too_far, + bool& backward_is_too_far); +}; + +} // namespace cura + +#endif // GEOMETRY_POLYGON_H diff --git a/include/geometry/Polyline.h b/include/geometry/Polyline.h new file mode 100644 index 0000000000..a26c942fcf --- /dev/null +++ b/include/geometry/Polyline.h @@ -0,0 +1,160 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_POLYLINE_H +#define GEOMETRY_POLYLINE_H + +#include "geometry/OpenLinesSet.h" +#include "geometry/PointsSet.h" +#include "geometry/SegmentIterator.h" + +namespace cura +{ + +template +class LinesSet; +class AngleRadians; +class OpenPolyline; + +/*! + * \brief Base class for various types of polylines. A polyline is basically a set of points, but + * we geometrically interpret them forming a chain of segments between each other. + * \sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#pointsset + * + * * Open Polyline : this represents a line that does not close, i.e. the last point is different + * from the initial point (think of the U letter) + * * Closed Polyline : a closed polyline has a final segment joining the last point and the + * initial one (think of the O letter) + * * Polygon : this is a particular type of closed polyline, for which we consider that the + * "inside" part of the line forms a surface + * + * \note Historically, the open and closed polylines were not explicitely differenciated, so + * sometimes we would use an open polyline with an extra point at the end, which virtually + * closes the line. This behaviour is now deprecated and should be removed over time. + */ +class Polyline : public PointsSet +{ +public: + using segments_iterator = SegmentIterator; + using const_segments_iterator = SegmentIterator; + + /*! \brief Builds an empty polyline */ + Polyline() = default; + + /*! \brief Creates a copy of the given polyline */ + Polyline(const Polyline& other) = default; + + /*! \brief Constructor that takes ownership of the inner points list from the given polyline */ + Polyline(Polyline&& other) = default; + + /*! \brief Constructor with a points initializer list, provided for convenience */ + Polyline(const std::initializer_list& initializer) + : PointsSet(initializer) + { + } + + /*! \brief Constructor with an existing list of points */ + explicit Polyline(const ClipperLib::Path& points) + : PointsSet(points) + { + } + + /*! \brief Constructor that takes ownership of the given list of points */ + explicit Polyline(ClipperLib::Path&& points) + : PointsSet{ std::move(points) } + { + } + + ~Polyline() override = default; + + /*! + * \brief Indicates whether this polyline has an additional closing segment between the last + * point in the set and the first one + * \return True if a segment between the last and first point should be considered + */ + [[nodiscard]] virtual bool hasClosingSegment() const = 0; + + /*! + * \brief Gets the total number of "full" segments in the polyline. Calling this is also safe if + * there are not enough points to make a valid polyline, so it can also be a good + * indicator of a "valid" polyline. + */ + [[nodiscard]] virtual size_t segmentsCount() const = 0; + + /*! + * \brief Indicates whether the points set form a valid polyline, i.e. if it has enough points + * according to its type. + */ + [[nodiscard]] virtual bool isValid() const = 0; + + Polyline& operator=(const Polyline& other) = default; + + Polyline& operator=(Polyline&& other) = default; + + /*! \brief Provides a begin iterator to iterate over all the segments of the line */ + [[nodiscard]] const_segments_iterator beginSegments() const; + + /*! \brief Provides an end iterator to iterate over all the segments of the line */ + [[nodiscard]] const_segments_iterator endSegments() const; + + /*! \brief Provides a begin iterator to iterate over all the segments of the line */ + segments_iterator beginSegments(); + + /*! \brief Provides an end iterator to iterate over all the segments of the line */ + segments_iterator endSegments(); + + /*! + * Split these poly line objects into several line segment objects consisting of only two verts + * and store them in the \p result + */ + void splitIntoSegments(OpenLinesSet& result) const; + [[nodiscard]] OpenLinesSet splitIntoSegments() const; + + /*! + * On Y-axis positive upward displays, Orientation will return true if the polygon's orientation is counter-clockwise. + * + * from http://www.angusj.com/delphi/clipper/documentation/Docs/Units/ClipperLib/Functions/Orientation.htm + */ + [[nodiscard]] bool orientation() const + { + return ClipperLib::Orientation(getPoints()); + } + + [[nodiscard]] coord_t length() const; + + [[nodiscard]] bool shorterThan(const coord_t check_length) const; + + void reverse() + { + ClipperLib::ReversePath(getPoints()); + } + + void removeColinearEdges(const AngleRadians max_deviation_angle); + + /*! + * Removes consecutive line segments with same orientation and changes this polygon. + * + * 1. Removes verts which are connected to line segments which are too small. + * 2. Removes verts which detour from a direct line from the previous and next vert by a too small amount. + * 3. Moves a vert when a small line segment is connected to a much longer one. in order to maintain the outline of the object. + * 4. Don't remove a vert when the impact on the outline of the object is too great. + * + * Note that the simplify is a best effort algorithm. It does not guarantee that no lines below the provided smallest_line_segment_squared are left. + * + * The following example (Two very long line segments (" & , respectively) that are connected by a very small line segment (i) is unsimplifable by this + * function, even though the actual area change of removing line segment i is very small. The reason for this is that in the case of long lines, even a small + * deviation from it's original direction is very noticeable in the final result, especially if the polygons above make a slightly different choice. + * + * """"""""""""""""""""""""""""""""i,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,, + + * + * \param smallest_line_segment_squared maximal squared length of removed line segments + * \param allowed_error_distance_squared The square of the distance of the middle point to the line segment of the consecutive and previous point for which the middle point is + removed + */ + void simplify(const coord_t smallest_line_segment_squared = MM2INT(0.01) * MM2INT(0.01), const coord_t allowed_error_distance_squared = 25); +}; + +} // namespace cura + +#endif // GEOMETRY_POLYLINE_H diff --git a/include/geometry/SegmentIterator.h b/include/geometry/SegmentIterator.h new file mode 100644 index 0000000000..5164c7d32f --- /dev/null +++ b/include/geometry/SegmentIterator.h @@ -0,0 +1,85 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_SEGMENT_ITERATOR_H +#define GEOMETRY_SEGMENT_ITERATOR_H + +#include "geometry/Point2LL.h" + +namespace cura +{ + +enum class ConstnessType +{ + Const, + Modifiable, +}; + +/*! @brief Custom iterator to loop over the segments of a polyline/polygon */ +template +struct SegmentIterator +{ + /*! @brief Transitory structure used to iterate over segments within a polyline */ + struct Segment + { + using PointType = std::conditional_t; + + PointType& start; + PointType& end; + }; + + // Define type values so that std library methods can use this iterator + using iterator_category = std::random_access_iterator_tag; + using value_type = Segment; + using difference_type = std::ptrdiff_t; + using pointer = Segment*; + using reference = Segment&; + using source_iterator_type = std::conditional_t::const_iterator, typename std::vector::iterator>; + +private: + source_iterator_type current_pos_; + source_iterator_type begin_; + source_iterator_type before_end_; + +public: + SegmentIterator(source_iterator_type pos, source_iterator_type begin, source_iterator_type end) + : current_pos_(pos) + , begin_(begin) + , before_end_(end != begin ? std::prev(end) : end) + { + } + + Segment operator*() const + { + if (current_pos_ == before_end_) + { + return Segment{ *current_pos_, *begin_ }; + } + return Segment{ *current_pos_, *std::next(current_pos_) }; + } + + SegmentIterator& operator++() + { + current_pos_++; + return *this; + } + + bool operator==(const SegmentIterator& other) const + { + return current_pos_ == other.current_pos_; + } + + bool operator!=(const SegmentIterator& other) const + { + return ! (*this == other); + } + + friend difference_type operator-(const SegmentIterator& iterator1, const SegmentIterator& iterator2) + { + return iterator1.current_pos_ - iterator2.current_pos_; + } +}; + +} // namespace cura + +#endif // GEOMETRY_SEGMENT_ITERATOR_H diff --git a/include/geometry/Shape.h b/include/geometry/Shape.h new file mode 100644 index 0000000000..aaa373f4e1 --- /dev/null +++ b/include/geometry/Shape.h @@ -0,0 +1,282 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_SHAPE_H +#define GEOMETRY_SHAPE_H + +#include "geometry/LinesSet.h" +#include "geometry/Polygon.h" +#include "settings/types/Angle.h" + +namespace cura +{ + +class Polygon; +class Ratio; +class SingleShape; +class PartsView; +class PointMatrix; +class Point3Matrix; + +/*! + * @brief A Shape is a set of polygons that together form a complex shape. Some of the polygons may + * be contained inside others, being actually "holes" of the shape. For example, if you + * wanted to represent the "8" digit with polygons, you would need 1 for the outline and 2 + * for the "holes" so the shape would contain a total of 3 polygons. + * @sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#shape + */ +class Shape : public LinesSet +{ +public: + // Clipper expects and returns implicitely closed polygons + static constexpr bool clipper_explicitely_closed_ = false; + + /*! \brief Constructor of an empty shape */ + Shape() = default; + + /*! \brief Creates a copy of the given shape */ + Shape(const Shape& other) = default; + + /*! \brief Constructor that takes the inner polygons list from the given shape */ + Shape(Shape&& other) = default; + + /*! \brief Constructor with an existing set of polygons */ + Shape(const std::vector& polygons); + + /*! + * \brief Constructor that takes ownership of the given list of points + * \param explicitely_closed Specify whether the given points form an explicitely closed line + */ + explicit Shape(ClipperLib::Paths&& paths, bool explicitely_closed = clipper_explicitely_closed_); + + Shape& operator=(const Shape& other) = default; + + Shape& operator=(Shape&& other) noexcept = default; + + ~Shape() override = default; + + void emplace_back(ClipperLib::Paths&& paths, bool explicitely_closed = clipper_explicitely_closed_); + + void emplace_back(ClipperLib::Path&& path, bool explicitely_closed = clipper_explicitely_closed_); + + void emplace_back(auto&&... args) + { + LinesSet::emplace_back(std::forward(args)...); + } + + [[nodiscard]] Shape difference(const Shape& other) const; + + [[nodiscard]] Shape unionPolygons(const Shape& other, ClipperLib::PolyFillType fill_type = ClipperLib::pftNonZero) const; + + /*! + * Union all polygons with each other (When polygons.add(polygon) has been called for overlapping polygons) + */ + [[nodiscard]] Shape unionPolygons() const; + + [[nodiscard]] Shape intersection(const Shape& other) const; + + /*! + * @brief Overridden definition of LinesSet::offset() + * @note The behavior of this method is exactly the same, but it just exists because it allows + * for a performance optimization + */ + [[nodiscard]] Shape offset(coord_t distance, ClipperLib::JoinType join_type = ClipperLib::jtMiter, double miter_limit = 1.2) const; + + /*! + * Intersect polylines with the area covered by the shape. + * + * \note Due to a clipper bug with polylines with nearly collinear segments, the polylines are cut up into separate polylines, and restitched back together at the end. + * + * \param polylines The polylines to limit to the area of this Polygons object + * \param restitch Whether to stitch the resulting segments into longer polylines, or leave every segment as a single segment + * \param max_stitch_distance The maximum distance for two polylines to be stitched together with a segment + * \return The resulting polylines limited to the area of this Polygons object + * \todo This should technically return a MixedLinesSet, because it can definitely contain open and closed polylines, but that is a heavy change + */ + template + OpenLinesSet intersection(const LinesSet& polylines, bool restitch = true, const coord_t max_stitch_distance = 10_mu) const; + + [[nodiscard]] Shape xorPolygons(const Shape& other, ClipperLib::PolyFillType pft = ClipperLib::pftEvenOdd) const; + + [[nodiscard]] Shape execute(ClipperLib::PolyFillType pft = ClipperLib::pftEvenOdd) const; + + /*! + * Check if we are inside the polygon. + * + * We do this by counting the number of polygons inside which this point lies. + * An odd number is inside, while an even number is outside. + * + * Returns false if outside, true if inside; if the point lies exactly on the border, will return \p border_result. + * + * \param p The point for which to check if it is inside this polygon + * \param border_result What to return when the point is exactly on the border + * \return Whether the point \p p is inside this polygon (or \p border_result when it is on the border) + */ + [[nodiscard]] bool inside(const Point2LL& p, bool border_result = false) const; + + /*! + * Find the polygon inside which point \p p resides. + * + * We do this by tracing from the point towards the positive X direction, + * every line we cross increments the crossings counter. If we have an even number of crossings then we are not inside the polygon. + * We then find the polygon with an uneven number of crossings which is closest to \p p. + * + * If \p border_result, we return the first polygon which is exactly on \p p. + * + * \param p The point for which to check in which polygon it is. + * \param border_result Whether a point exactly on a polygon counts as inside + * \return The index of the polygon inside which the point \p p resides + */ + [[nodiscard]] size_t findInside(const Point2LL& p, bool border_result = false) const; + + /*! + * \brief Approximates the convex hull of the polygons. + * \p extra_outset Extra offset outward + * \return the convex hull (approximately) + * + */ + [[nodiscard]] Shape approxConvexHull(int extra_outset = 0) const; + + /*! \brief Make each of the polygons convex */ + void makeConvex(); + + /*! + * Compute the area enclosed within the polygons (minus holes) + * + * \return The area in square micron + */ + [[nodiscard]] double area() const; + + /*! + * Smooth out small perpendicular segments + * Smoothing is performed by removing the inner most vertex of a line segment smaller than \p remove_length + * which has an angle with the next and previous line segment smaller than roughly 150* + * + * Note that in its current implementation this function doesn't remove line segments with an angle smaller than 30* + * Such would be the case for an N shape. + * + * \param remove_length The length of the largest segment removed + * \return The smoothed polygon + */ + [[nodiscard]] Shape smooth(int remove_length) const; + + /*! + * Smooth out sharp inner corners, by taking a shortcut which bypasses the corner + * + * \param angle The maximum angle of inner corners to be smoothed out + * \param shortcut_length The desired length of the shortcut line segment introduced (shorter shortcuts may be unavoidable) + * \return The resulting polygons + */ + [[nodiscard]] Shape smoothOutward(const AngleDegrees angle, int shortcut_length) const; + + [[nodiscard]] Shape smooth2(int remove_length, int min_area) const; //!< removes points connected to small lines + + void removeColinearEdges(const AngleRadians max_deviation_angle = AngleRadians(0.0005)); + + /*! + * Remove all but the polygons on the very outside. + * Exclude holes and parts within holes. + * \return the resulting polygons. + */ + [[nodiscard]] Shape getOutsidePolygons() const; + + /*! + * Split up the polygons into groups according to the even-odd rule. + * Each SingleShape in the result has an outline as first polygon, whereas the rest are holes. + */ + [[nodiscard]] std::vector splitIntoParts(bool union_all = false) const; + + /*! + * Sort the polygons into bins where each bin has polygons which are contained within one of the polygons in the previous bin. + * + * \warning When polygons are crossing each other the result is undefined. + */ + [[nodiscard]] std::vector sortByNesting() const; + + /*! + * Split up the polygons into groups according to the even-odd rule. + * Each vector in the result has the index to an outline as first index, whereas the rest are indices to holes. + * + * \warning Note that this function reorders the polygons! + */ + PartsView splitIntoPartsView(bool union_all = false); + + /*! + * Removes polygons with area smaller than \p min_area_size (note that min_area_size is in mm^2, not in micron^2). + * Unless \p remove_holes is true, holes are not removed even if their area is below \p min_area_size. + * However, holes that are contained within outlines whose area is below the threshold are removed though. + */ + void removeSmallAreas(const double min_area_size, const bool remove_holes = false); + + /*! + * Removes the same polygons from this set (and also empty polygons). + * Shape are considered the same if all points lie within [same_distance] of their counterparts. + */ + [[nodiscard]] Shape removePolygon(const Shape& to_be_removed, int same_distance = 0) const; + + [[nodiscard]] Shape processEvenOdd(ClipperLib::PolyFillType poly_fill_type = ClipperLib::PolyFillType::pftEvenOdd) const; + + /*! + * Ensure the polygon is manifold, by removing small areas where the polygon touches itself. + * ____ ____ + * | | | | + * | |____ ==> | / ____ + * """"| | """ / | + * |____| |____| + * + */ + void ensureManifold(); + + void applyMatrix(const PointMatrix& matrix); + + void applyMatrix(const Point3Matrix& matrix); + + [[nodiscard]] Shape offsetMulti(const std::vector& offset_dists) const; + + /*! + * @brief Remove self-intersections from the polygons + * _note_: this function uses wagyu to remove the self intersections. + * since wagyu uses a different internal representation of the polygons + * we need to convert back and forward between data structures which + * might impact performance, use wisely! + * + * @return Polygons - the cleaned polygons + */ + [[nodiscard]] Shape removeNearSelfIntersections() const; + + /*! + * \brief Simplify the polygon lines using ClipperLib::SimplifyPolygons + */ + void simplify(ClipperLib::PolyFillType fill_type = ClipperLib::pftEvenOdd); + +#ifdef BUILD_TESTS + /*! + * @brief Import the polygon from a WKT string + * @param wkt The WKT string to read from + * @return Polygons The polygons read from the stream + */ + [[maybe_unused]] static Shape fromWkt(const std::string& wkt); + + /*! + * @brief Export the polygon to a WKT string + * @param stream The stream to write to + */ + [[maybe_unused]] void writeWkt(std::ostream& stream) const; +#endif + +private: + /*! + * recursive part of \ref Polygons::removeEmptyHoles and \ref Polygons::getEmptyHoles + * \param node The node of the polygons part to process + * \param remove_holes Whether to remove empty holes or everything but the empty holes + * \param ret Where to store polygons which are not empty holes + */ + void removeEmptyHolesProcessPolyTreeNode(const ClipperLib::PolyNode& node, const bool remove_holes, Shape& ret) const; + void splitIntoPartsProcessPolyTreeNode(ClipperLib::PolyNode* node, std::vector& ret) const; + void sortByNestingProcessPolyTreeNode(ClipperLib::PolyNode* node, const size_t nesting_idx, std::vector& ret) const; + void splitIntoPartsViewProcessPolyTreeNode(PartsView& parts_view, Shape& reordered, ClipperLib::PolyNode* node) const; +}; + +} // namespace cura + +#endif // GEOMETRY_SHAPE_H diff --git a/include/geometry/SingleShape.h b/include/geometry/SingleShape.h new file mode 100644 index 0000000000..a962672a04 --- /dev/null +++ b/include/geometry/SingleShape.h @@ -0,0 +1,44 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher + +#ifndef GEOMETRY_SINGLE_SHAPE_H +#define GEOMETRY_SINGLE_SHAPE_H + +#include "geometry/Shape.h" + +namespace cura +{ + +class Polygon; + +/*! + * @brief A single area with holes. The first polygon is the outline, while the rest are holes within this outline. + * @sa https://github.com/Ultimaker/CuraEngine/wiki/Geometric-Base-Types#singleshape + * + * This class has little more functionality than Shape, but serves to show that a specific instance + * is ordered such that the first Polygon is the outline and the rest are holes. + */ +class SingleShape : public Shape +{ +public: + SingleShape() = default; + + explicit SingleShape(Shape&& shape) + : Shape{ std::move(shape) } {}; + + Polygon& outerPolygon(); + + [[nodiscard]] const Polygon& outerPolygon() const; + + /*! + * Tests whether the given point is inside this polygon part. + * \param p The point to test whether it is insisinglehde. + * \param border_result If the point is exactly on the border, this will be + * returned instead. + */ + [[nodiscard]] bool inside(const Point2LL& p, bool border_result = false) const; +}; + +} // namespace cura + +#endif // GEOMETRY_SINGLE_SHAPE_H diff --git a/include/infill.h b/include/infill.h index 3f0cc19943..081652d68f 100644 --- a/include/infill.h +++ b/include/infill.h @@ -8,6 +8,9 @@ #include +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/Point2LL.h" #include "infill/LightningGenerator.h" #include "infill/ZigzagConnectorProcessor.h" #include "settings/EnumSettings.h" //For infill types. @@ -15,7 +18,6 @@ #include "settings/types/Angle.h" #include "utils/AABB.h" #include "utils/ExtrusionLine.h" -#include "utils/Point2LL.h" #include "utils/section_type.h" namespace cura @@ -35,8 +37,8 @@ class Infill // We skip ZigZag, Cross and Cross3D because they have their own algorithms. Eventually we want to replace all that with the new algorithm. // Cubic Subdivision ends lines in the center of the infill so it won't be effective. bool connect_polygons_{}; //!< Whether to connect as much polygons together into a single path - Polygons outer_contour_{}; //!< The area that originally needs to be filled with infill. The input of the algorithm. - Polygons inner_contour_{}; //!< The part of the contour that will get filled with an infill pattern. Equals outer_contour minus the extra infill walls. + Shape outer_contour_{}; //!< The area that originally needs to be filled with infill. The input of the algorithm. + Shape inner_contour_{}; //!< The part of the contour that will get filled with an infill pattern. Equals outer_contour minus the extra infill walls. coord_t infill_line_width_{}; //!< The line width of the infill lines to generate coord_t line_distance_{}; //!< The distance between two infill lines / polygons coord_t infill_overlap_{}; //!< the distance by which to overlap with the actual area within which to generate infill @@ -74,7 +76,7 @@ class Infill EFillMethod pattern, bool zig_zaggify, bool connect_polygons, - Polygons in_outline, + Shape in_outline, coord_t infill_line_width, coord_t line_distance, coord_t infill_overlap, @@ -104,7 +106,7 @@ class Infill EFillMethod pattern, bool zig_zaggify, bool connect_polygons, - Polygons in_outline, + Shape in_outline, coord_t infill_line_width, coord_t line_distance, coord_t infill_overlap, @@ -142,7 +144,7 @@ class Infill EFillMethod pattern, bool zig_zaggify, bool connect_polygons, - Polygons in_outline, + Shape in_outline, coord_t infill_line_width, coord_t line_distance, coord_t infill_overlap, @@ -202,15 +204,15 @@ class Infill */ void generate( std::vector& toolpaths, - Polygons& result_polygons, - Polygons& result_lines, + Shape& result_polygons, + OpenLinesSet& result_lines, const Settings& settings, int layer_idx, SectionType section_type, const std::shared_ptr& cross_fill_provider = nullptr, const std::shared_ptr& lightning_layer = nullptr, const SliceMeshStorage* mesh = nullptr, - const Polygons& prevent_small_exposed_to_air = Polygons()); + const Shape& prevent_small_exposed_to_air = Shape()); /*! * Generate the wall toolpaths of an infill area. It will return the inner contour and set the inner-contour. @@ -224,9 +226,9 @@ class Infill * \param settings [in] A settings storage to use for generating variable-width walls. * \return The inner contour of the wall toolpaths */ - static Polygons generateWallToolPaths( + static Shape generateWallToolPaths( std::vector& toolpaths, - Polygons& outer_contour, + Shape& outer_contour, const size_t wall_line_count, const coord_t line_width, const coord_t infill_overlap, @@ -358,7 +360,7 @@ class Infill * * \param include_start Wether to include the start point or not, useful when tracing a poly-line. */ - void appendTo(PolygonRef& result_polyline, const bool include_start = true); + void appendTo(OpenPolyline& result_polyline, const bool include_start = true); }; /*! @@ -373,8 +375,8 @@ class Infill */ void _generate( std::vector& toolpaths, - Polygons& result_polygons, - Polygons& result_lines, + Shape& result_polygons, + OpenLinesSet& result_lines, const Settings& settings, const std::shared_ptr& cross_fill_pattern = nullptr, const std::shared_ptr& lightning_layer = nullptr, @@ -391,21 +393,21 @@ class Infill * \param[in,out] result_polygons The polygons to be multiplied (input and output) * \param[in,out] result_lines The lines to be multiplied (input and output) */ - void multiplyInfill(Polygons& result_polygons, Polygons& result_lines); + void multiplyInfill(Shape& result_polygons, OpenLinesSet& result_lines); /*! * Generate gyroid infill * \param result_polylines (output) The resulting polylines * \param result_polygons (output) The resulting polygons, if zigzagging accidentally happened to connect gyroid lines in a circle. */ - void generateGyroidInfill(Polygons& result_polylines, Polygons& result_polygons); + void generateGyroidInfill(OpenLinesSet& result_polylines, Shape& result_polygons); /*! * Generate lightning fill aka minfill aka 'Ribbed Support Vault Infill', see Tricard,Claux,Lefebvre/'Ribbed Support Vaults for 3D Printing of Hollowed Objects' * see https://hal.archives-ouvertes.fr/hal-02155929/document * \param result (output) The resulting polygons */ - void generateLightningInfill(const std::shared_ptr& lightning_layer, Polygons& result_lines); + void generateLightningInfill(const std::shared_ptr& lightning_layer, OpenLinesSet& result_lines); /*! * Generate sparse concentric infill @@ -419,25 +421,25 @@ class Infill * Generate a rectangular grid of infill lines * \param[out] result (output) The resulting lines */ - void generateGridInfill(Polygons& result); + void generateGridInfill(OpenLinesSet& result); /*! * Generate a shifting triangular grid of infill lines, which combine with consecutive layers into a cubic pattern * \param[out] result (output) The resulting lines */ - void generateCubicInfill(Polygons& result); + void generateCubicInfill(OpenLinesSet& result); /*! * Generate a double shifting square grid of infill lines, which combine with consecutive layers into a tetrahedral pattern * \param[out] result (output) The resulting lines */ - void generateTetrahedralInfill(Polygons& result); + void generateTetrahedralInfill(OpenLinesSet& result); /*! * Generate a double shifting square grid of infill lines, which combine with consecutive layers into a quarter cubic pattern * \param[out] result (output) The resulting lines */ - void generateQuarterCubicInfill(Polygons& result); + void generateQuarterCubicInfill(OpenLinesSet& result); /*! * Generate a single shifting square grid of infill lines. @@ -447,26 +449,26 @@ class Infill * \param angle_shift The angle to add to the infill_angle * \param[out] result (output) The resulting lines */ - void generateHalfTetrahedralInfill(double pattern_z_shift, int angle_shift, Polygons& result); + void generateHalfTetrahedralInfill(double pattern_z_shift, int angle_shift, OpenLinesSet& result); /*! * Generate a triangular grid of infill lines * \param[out] result (output) The resulting lines */ - void generateTriangleInfill(Polygons& result); + void generateTriangleInfill(OpenLinesSet& result); /*! * Generate a triangular grid of infill lines * \param[out] result (output) The resulting lines */ - void generateTrihexagonInfill(Polygons& result); + void generateTrihexagonInfill(OpenLinesSet& result); /*! * Generate a 3d pattern of subdivided cubes on their points * \param[out] result The resulting lines * \param[in] mesh Where the Cubic Subdivision Infill precomputation is stored */ - void generateCubicSubDivInfill(Polygons& result, const SliceMeshStorage& mesh); + void generateCubicSubDivInfill(OpenLinesSet& result, const SliceMeshStorage& mesh); /*! * Generate a 3d pattern of subdivided cubes on their points @@ -474,7 +476,7 @@ class Infill * \param[out] result_polygons The resulting polygons * \param[out] result_lines The resulting lines */ - void generateCrossInfill(const SierpinskiFillProvider& cross_fill_provider, Polygons& result_polygons, Polygons& result_lines); + void generateCrossInfill(const SierpinskiFillProvider& cross_fill_provider, Shape& result_polygons, OpenLinesSet& result_lines); /*! * Convert a mapping from scanline to line_segment-scanline-intersections (\p cut_list) into line segments, using the even-odd rule @@ -487,7 +489,7 @@ class Infill * \param total_shift total shift of the scanlines in the direction perpendicular to the fill_angle. */ void addLineInfill( - Polygons& result, + OpenLinesSet& result, const PointMatrix& rotation_matrix, const int scanline_min_idx, const int line_distance, @@ -506,7 +508,7 @@ class Infill * \param infill_rotation The angle of the generated lines * \param extra_shift extra shift of the scanlines in the direction perpendicular to the infill_rotation */ - void generateLineInfill(Polygons& result, int line_distance, const double& infill_rotation, coord_t extra_shift); + void generateLineInfill(OpenLinesSet& result, int line_distance, const double& infill_rotation, coord_t extra_shift); /*! * Function for creating linear based infill types (Lines, ZigZag). @@ -524,7 +526,7 @@ class Infill * \param extra_shift extra shift of the scanlines in the direction perpendicular to the fill_angle */ void generateLinearBasedInfill( - Polygons& result, + OpenLinesSet& result, const int line_distance, const PointMatrix& rotation_matrix, ZigzagConnectorProcessor& zigzag_connector_processor, @@ -578,7 +580,7 @@ class Infill * \param line_distance The distance between two lines which are in the same direction * \param infill_rotation The angle of the generated lines */ - void generateZigZagInfill(Polygons& result, const coord_t line_distance, const double& infill_rotation); + void generateZigZagInfill(OpenLinesSet& result, const coord_t line_distance, const double& infill_rotation); /*! * determine how far the infill pattern should be shifted based on the values of infill_origin and \p infill_rotation @@ -620,7 +622,7 @@ class Infill * border of the infill area, similar to the zigzag pattern. * \param[in/out] result_lines The lines to connect together. */ - void connectLines(Polygons& result_lines); + void connectLines(OpenLinesSet& result_lines); }; static_assert(concepts::semiregular, "Infill should be semiregular"); diff --git a/include/infill/GyroidInfill.h b/include/infill/GyroidInfill.h index 697ee26bc1..00c2e3b273 100644 --- a/include/infill/GyroidInfill.h +++ b/include/infill/GyroidInfill.h @@ -1,12 +1,13 @@ -//Copyright (c) 2020 Ultimaker B.V. -//CuraEngine is released under the terms of the AGPLv3 or higher. +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. -#include "../utils/Coord_t.h" +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "utils/Coord_t.h" namespace cura { - -class Polygons; +class Shape; class GyroidInfill { @@ -33,11 +34,8 @@ class GyroidInfill * \param z The Z coordinate of this layer. Different Z coordinates cause the pattern to vary, producing a 3D * pattern. */ - static void generateTotalGyroidInfill(Polygons& result_lines, bool zig_zaggify, coord_t line_distance, const Polygons& in_outline, coord_t z); - -private: + static void generateTotalGyroidInfill(OpenLinesSet& result_lines, bool zig_zaggify, coord_t line_distance, const Shape& in_outline, coord_t z); +private: }; - } // namespace cura - diff --git a/include/infill/LightningDistanceField.h b/include/infill/LightningDistanceField.h index 3624fd72c7..1968e40e8a 100644 --- a/include/infill/LightningDistanceField.h +++ b/include/infill/LightningDistanceField.h @@ -5,7 +5,7 @@ #define LIGHTNING_DISTANCE_FIELD_H #include "../utils/SquareGrid.h" //Tracking for each location the distance to overhang. -#include "../utils/polygon.h" //Using outlines to fill and tracking overhang. +#include "geometry/Polygon.h" //Using outlines to fill and tracking overhang. namespace cura { @@ -29,7 +29,7 @@ class LightningDistanceField * \param current_overhang The overhang that needs to be supported on this * layer. */ - LightningDistanceField(const coord_t& radius, const Polygons& current_outline, const Polygons& current_overhang); + LightningDistanceField(const coord_t& radius, const Shape& current_outline, const Shape& current_overhang); /*! * Gets the next unsupported location to be supported by a new branch. @@ -76,13 +76,13 @@ class LightningDistanceField /*! * The total infill area on the current layer. */ - const Polygons& current_outline_; + const Shape& current_outline_; /*! * The overhang that gets introduced on this layer, which the infill will * need to support. */ - const Polygons& current_overhang_; + const Shape& current_overhang_; /*! * Represents a small discrete area of infill that needs to be supported. diff --git a/include/infill/LightningGenerator.h b/include/infill/LightningGenerator.h index 30a36e5b02..0d6ff88552 100644 --- a/include/infill/LightningGenerator.h +++ b/include/infill/LightningGenerator.h @@ -1,18 +1,17 @@ -//Copyright (c) 2021 Ultimaker B.V. -//CuraEngine is released under the terms of the AGPLv3 or higher. +// Copyright (c) 2021 Ultimaker B.V. +// CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef LIGHTNING_GENERATOR_H #define LIGHTNING_GENERATOR_H -#include "LightningLayer.h" - -#include "../utils/polygonUtils.h" - #include #include #include -namespace cura +#include "../utils/polygonUtils.h" +#include "LightningLayer.h" + +namespace cura { class SliceMeshStorage; @@ -33,7 +32,7 @@ class SliceMeshStorage; * Printing of Hollowed Objects" by Tricard, Claux and Lefebvre: * https://www.researchgate.net/publication/333808588_Ribbed_Support_Vaults_for_3D_Printing_of_Hollowed_Objects */ -class LightningGenerator // "Just like Nicola used to make!" +class LightningGenerator // "Just like Nicola used to make!" { public: /*! @@ -111,7 +110,7 @@ class LightningGenerator // "Just like Nicola used to make!" * * This is generated by \ref generateInitialInternalOverhangs. */ - std::vector overhang_per_layer; + std::vector overhang_per_layer; /*! * For each layer, the generated lightning paths. diff --git a/include/infill/LightningLayer.h b/include/infill/LightningLayer.h index fe5a079e43..d62b5267a4 100644 --- a/include/infill/LightningLayer.h +++ b/include/infill/LightningLayer.h @@ -1,4 +1,4 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #ifndef LIGHTNING_LAYER_H @@ -9,9 +9,11 @@ #include #include -#include "../utils/SquareGrid.h" -#include "../utils/polygonUtils.h" +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" #include "infill/LightningTreeNode.h" +#include "utils/SquareGrid.h" +#include "utils/polygonUtils.h" namespace cura { @@ -21,7 +23,7 @@ using SparseLightningTreeNodeGrid = SparsePointGridInclusive boundary_location; //!< in case the gounding location is on the boundary + std::optional boundary_location; //!< in case the gounding location is on the boundary Point2LL p() const; }; @@ -36,8 +38,8 @@ class LightningLayer std::vector tree_roots; void generateNewTrees( - const Polygons& current_overhang, - const Polygons& current_outlines, + const Shape& current_overhang, + const Shape& current_outlines, const LocToLineGrid& outline_locator, const coord_t supporting_radius, const coord_t wall_supporting_radius); @@ -47,7 +49,7 @@ class LightningLayer */ GroundingLocation getBestGroundingLocation( const Point2LL& unsupported_location, - const Polygons& current_outlines, + const Shape& current_outlines, const LocToLineGrid& outline_locator, const coord_t supporting_radius, const coord_t wall_supporting_radius, @@ -63,18 +65,17 @@ class LightningLayer void reconnectRoots( std::vector& to_be_reconnected_tree_roots, - const Polygons& current_outlines, + const Shape& current_outlines, const LocToLineGrid& outline_locator, const coord_t supporting_radius, const coord_t wall_supporting_radius); - Polygons convertToLines(const Polygons& limit_to_outline, const coord_t line_width) const; + OpenLinesSet convertToLines(const Shape& limit_to_outline, const coord_t line_width) const; coord_t getWeightedDistance(const Point2LL& boundary_loc, const Point2LL& unsupported_location); void fillLocator(SparseLightningTreeNodeGrid& tree_node_locator); }; - } // namespace cura #endif // LIGHTNING_LAYER_H diff --git a/include/infill/LightningTreeNode.h b/include/infill/LightningTreeNode.h index 0ebd5503a8..fa73e10fb9 100644 --- a/include/infill/LightningTreeNode.h +++ b/include/infill/LightningTreeNode.h @@ -9,8 +9,11 @@ #include #include -#include "../utils/polygon.h" -#include "../utils/polygonUtils.h" +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" +#include "utils/polygonUtils.h" namespace cura { @@ -98,7 +101,7 @@ class LightningTreeNode : public std::enable_shared_from_this */ void propagateToNextLayer( std::vector& next_trees, - const Polygons& next_outlines, + const Shape& next_outlines, const LocToLineGrid& outline_locator, const coord_t prune_distance, const coord_t smooth_magnitude, @@ -200,7 +203,7 @@ class LightningTreeNode : public std::enable_shared_from_this /*! Reconnect trees from the layer above to the new outlines of the lower layer. * \return Wether or not the root is kept (false is no, true is yes). */ - bool realign(const Polygons& outlines, const LocToLineGrid& outline_locator, std::vector& rerooted_parts); + bool realign(const Shape& outlines, const LocToLineGrid& outline_locator, std::vector& rerooted_parts); struct RectilinearJunction { @@ -239,7 +242,7 @@ class LightningTreeNode : public std::enable_shared_from_this * * \param output all branches in this tree connected into polylines */ - void convertToPolylines(Polygons& output, const coord_t line_width) const; + void convertToPolylines(OpenLinesSet& output, const coord_t line_width) const; /*! If this was ever a direct child of the root, it'll have a previous grounding location. * @@ -258,9 +261,9 @@ class LightningTreeNode : public std::enable_shared_from_this * \param long_line a reference to a polyline in \p output which to continue building on in the recursion * \param output all branches in this tree connected into polylines */ - void convertToPolylines(size_t long_line_idx, Polygons& output) const; + void convertToPolylines(size_t long_line_idx, OpenLinesSet& output) const; - void removeJunctionOverlap(Polygons& polylines, const coord_t line_width) const; + void removeJunctionOverlap(OpenLinesSet& polylines, const coord_t line_width) const; bool is_root_; Point2LL p_; diff --git a/include/infill/NoZigZagConnectorProcessor.h b/include/infill/NoZigZagConnectorProcessor.h index 002fb0fb7b..c456297223 100644 --- a/include/infill/NoZigZagConnectorProcessor.h +++ b/include/infill/NoZigZagConnectorProcessor.h @@ -5,11 +5,15 @@ #define INFILL_NO_ZIGZAG_CONNECTOR_PROCESSOR_H #include "ZigzagConnectorProcessor.h" +#include "geometry/OpenLinesSet.h" namespace cura { -class Polygons; +class OpenPolyline; + +template +class LinesSet; /*! * This processor adds no connection. This is for line infill pattern. @@ -17,7 +21,7 @@ class Polygons; class NoZigZagConnectorProcessor : public ZigzagConnectorProcessor { public: - NoZigZagConnectorProcessor(const PointMatrix& rotation_matrix, Polygons& result) + NoZigZagConnectorProcessor(const PointMatrix& rotation_matrix, OpenLinesSet& result) : ZigzagConnectorProcessor( rotation_matrix, result, @@ -29,7 +33,7 @@ class NoZigZagConnectorProcessor : public ZigzagConnectorProcessor } void registerVertex(const Point2LL& vertex); - void registerScanlineSegmentIntersection(const Point2LL& intersection, int scanline_index); + void registerScanlineSegmentIntersection(const Point2LL& intersection, int scanline_index, coord_t min_distance_to_scanline); void registerPolyFinished(); }; diff --git a/include/infill/SubDivCube.h b/include/infill/SubDivCube.h index bfb10f487a..e16b4a2149 100644 --- a/include/infill/SubDivCube.h +++ b/include/infill/SubDivCube.h @@ -4,15 +4,18 @@ #ifndef INFILL_SUBDIVCUBE_H #define INFILL_SUBDIVCUBE_H +#include "geometry/OpenLinesSet.h" +#include "geometry/Point2LL.h" +#include "geometry/Point3LL.h" +#include "geometry/Point3Matrix.h" +#include "geometry/PointMatrix.h" #include "settings/types/LayerIndex.h" #include "settings/types/Ratio.h" -#include "utils/Point2LL.h" -#include "utils/Point3LL.h" namespace cura { -class Polygons; +class Polygon; class SliceMeshStorage; class SubDivCube @@ -37,7 +40,7 @@ class SubDivCube * \param z the specified layer height * \param result (output) The resulting lines */ - void generateSubdivisionLines(const coord_t z, Polygons& result); + void generateSubdivisionLines(const coord_t z, OpenLinesSet& result); private: /*! @@ -46,7 +49,7 @@ class SubDivCube * \param result (output) The resulting lines * \param directional_line_groups Array of 3 times a polylines. Used to keep track of line segments that are all pointing the same direction for line segment combining */ - void generateSubdivisionLines(const coord_t z, Polygons (&directional_line_groups)[3]); + void generateSubdivisionLines(const coord_t z, OpenLinesSet (&directional_line_groups)[3]); struct CubeProperties { @@ -92,7 +95,7 @@ class SubDivCube * Adds the defined line to the specified polygons. It assumes that the specified polygons are all parallel lines. Combines line segments with touching ends closer than * epsilon. \param[out] group the polygons to add the line to \param from the first endpoint of the line \param to the second endpoint of the line */ - void addLineAndCombine(Polygons& group, Point2LL from, Point2LL to); + void addLineAndCombine(OpenLinesSet& group, Point2LL from, Point2LL to); size_t depth_; //!< the recursion depth of the cube (0 is most recursed) Point3LL center_; //!< center location of the cube in absolute coordinates diff --git a/include/infill/ZigzagConnectorProcessor.h b/include/infill/ZigzagConnectorProcessor.h index 513ad9d8f1..a7ff59cfc1 100644 --- a/include/infill/ZigzagConnectorProcessor.h +++ b/include/infill/ZigzagConnectorProcessor.h @@ -4,11 +4,18 @@ #ifndef INFILL_ZIGZAG_CONNECTOR_PROCESSOR_H #define INFILL_ZIGZAG_CONNECTOR_PROCESSOR_H -#include "../utils/polygon.h" //TODO: We have implementation in this header file! +#include + +#include "geometry/OpenLinesSet.h" +#include "geometry/Point2LL.h" namespace cura { +class Polygon; +class Shape; +class PointMatrix; + /*! * Processor class for processing the connections between lines which makes the infill a zigzag pattern. * @@ -108,7 +115,7 @@ class ZigzagConnectorProcessor * \param skip_some_zags Whether to skip some zags * \param zag_skip_count Skip 1 zag in every N zags */ - ZigzagConnectorProcessor(const PointMatrix& rotation_matrix, Polygons& result, bool use_endpieces, bool connected_endpieces, bool skip_some_zags, int zag_skip_count) + ZigzagConnectorProcessor(const PointMatrix& rotation_matrix, OpenLinesSet& result, bool use_endpieces, bool connected_endpieces, bool skip_some_zags, int zag_skip_count) : rotation_matrix_(rotation_matrix) , result_(result) , use_endpieces_(use_endpieces) @@ -137,7 +144,7 @@ class ZigzagConnectorProcessor * \param intersection The intersection * \param scanline_index Index of the current scanline */ - virtual void registerScanlineSegmentIntersection(const Point2LL& intersection, int scanline_index); + virtual void registerScanlineSegmentIntersection(const Point2LL& intersection, int scanline_index, coord_t min_distance_to_scanline); /*! * Handle the end of a polygon and prepare for the next. @@ -156,7 +163,7 @@ class ZigzagConnectorProcessor * * \param polyline The polyline to add */ - void addPolyline(PolygonRef polyline); + void addPolyline(const OpenPolyline& polyline); /*! * Checks whether the current connector should be added or not. @@ -166,17 +173,6 @@ class ZigzagConnectorProcessor */ bool shouldAddCurrentConnector(int start_scanline_idx, int end_scanline_idx) const; - /*! - * Checks whether two points are separated at least by "threshold" microns. - * If they are far away from each other enough, the line represented by the two points - * will be added; In case they are close, the second point will be set to be the same - * as the first and this line won't be added. - * - * \param first_point The first of the points - * \param second_point The second of the points - */ - void checkAndAddZagConnectorLine(Point2LL* first_point, Point2LL* second_point); - /*! * Adds a Zag connector represented by the given points. The last line of the connector will not be * added if the given connector is an end piece and "connected_endpieces" is not enabled. @@ -186,9 +182,11 @@ class ZigzagConnectorProcessor */ void addZagConnector(std::vector& points, bool is_endpiece); + bool handleConnectorTooCloseToSegment(const coord_t scanline_x, const coord_t min_distance_to_scanline); + protected: const PointMatrix& rotation_matrix_; //!< The rotation matrix used to enforce the infill angle - Polygons& result_; //!< The result of the computation + OpenLinesSet& result_; //!< The result of the computation const bool use_endpieces_; //!< Whether to include end pieces or not const bool connected_endpieces_; //!< Whether the end pieces should be connected with the rest part of the infill @@ -212,29 +210,6 @@ class ZigzagConnectorProcessor std::vector current_connector_; }; -// -// Inline functions -// - -inline void ZigzagConnectorProcessor::reset() -{ - is_first_connector_ = true; - first_connector_end_scanline_index_ = 0; - last_connector_index_ = 0; - first_connector_.clear(); - current_connector_.clear(); -} - -inline void ZigzagConnectorProcessor::addPolyline(PolygonRef polyline) -{ - result_.emplace_back(polyline); - for (Point2LL& p : result_.back()) - { - p = rotation_matrix_.unapply(p); - } -} - - } // namespace cura diff --git a/include/pathPlanning/Comb.h b/include/pathPlanning/Comb.h index 289a5a1f5d..35bca5a480 100644 --- a/include/pathPlanning/Comb.h +++ b/include/pathPlanning/Comb.h @@ -7,9 +7,11 @@ #include // To find the maximum for coord_t. #include // shared_ptr -#include "../settings/types/LayerIndex.h" // To store the layer on which we comb. -#include "../utils/polygon.h" -#include "../utils/polygonUtils.h" +#include "geometry/PartsView.h" +#include "geometry/Polygon.h" +#include "geometry/SingleShape.h" +#include "settings/types/LayerIndex.h" // To store the layer on which we comb. +#include "utils/polygonUtils.h" namespace cura { @@ -36,7 +38,7 @@ class SliceDataStorage; * perpendicular to its boundary. * * As an optimization, the combing paths inside are calculated on specifically - * those PolygonsParts within which to comb, while the boundary_outside isn't + * those SingleShapes within which to comb, while the boundary_outside isn't * split into outside parts, because generally there is only one outside part; * encapsulated holes occur less often. */ @@ -56,9 +58,9 @@ class Comb bool dest_is_inside_; //!< Whether the startPoint or endPoint is inside the inside boundary Point2LL in_or_mid_; //!< The point on the inside boundary, or in between the inside and outside boundary if the start/end point isn't inside the inside boudary Point2LL out_; //!< The point on the outside boundary - PolygonsPart dest_part_; //!< The assembled inside-boundary PolygonsPart in which the dest_point lies. (will only be initialized when Crossing::dest_is_inside holds) - std::optional dest_crossing_poly_; //!< The polygon of the part in which dest_point lies, which will be crossed (often will be the outside polygon) - const Polygons& boundary_inside_; //!< The inside boundary as in \ref Comb::boundary_inside + SingleShape dest_part_; //!< The assembled inside-boundary SingleShape in which the dest_point lies. (will only be initialized when Crossing::dest_is_inside holds) + std::optional dest_crossing_poly_; //!< The polygon of the part in which dest_point lies, which will be crossed (often will be the outside polygon) + const Shape& boundary_inside_; //!< The inside boundary as in \ref Comb::boundary_inside const LocToLineGrid& inside_loc_to_line_; //!< The loc to line grid \ref Comb::inside_loc_to_line /*! @@ -75,7 +77,7 @@ class Comb const bool dest_is_inside, const unsigned int dest_part_idx, const unsigned int dest_part_boundary_crossing_poly_idx, - const Polygons& boundary_inside, + const Shape& boundary_inside, const LocToLineGrid& inside_loc_to_line); /*! @@ -98,7 +100,7 @@ class Comb * \param comber[in] The combing calculator which has references to the * offsets and boundaries to use in combing. */ - bool findOutside(const ExtruderTrain& train, const Polygons& outside, const Point2LL close_to, const bool fail_on_unavoidable_obstacles, Comb& comber); + bool findOutside(const ExtruderTrain& train, const Shape& outside, const Point2LL close_to, const bool fail_on_unavoidable_obstacles, Comb& comber); private: const Point2LL dest_point_; //!< Either the eventual startPoint or the eventual endPoint of this combing move @@ -116,8 +118,8 @@ class Comb * \param comber[in] The combing calculator which has references to the offsets and boundaries to use in combing. * \return A pair of which the first is the crossing point on the inside boundary and the second the crossing point on the outside boundary */ - std::shared_ptr> - findBestCrossing(const ExtruderTrain& train, const Polygons& outside, ConstPolygonRef from, const Point2LL estimated_start, const Point2LL estimated_end, Comb& comber); + std::shared_ptr> + findBestCrossing(const ExtruderTrain& train, const Shape& outside, const Polygon& from, const Point2LL estimated_start, const Point2LL estimated_end, Comb& comber); }; @@ -134,15 +136,15 @@ class Comb static constexpr coord_t offset_dist_to_get_from_on_the_polygon_to_outside_ = 40; //!< in order to prevent on-boundary vs crossing boundary confusions (precision thing) static constexpr coord_t offset_extra_start_end_ = 100; //!< Distance to move start point and end point toward eachother to extra avoid collision with the boundaries. - Polygons boundary_inside_minimum_; //!< The boundary within which to comb. (Will be reordered by the partsView_inside_minimum) - Polygons boundary_inside_optimal_; //!< The boundary within which to comb. (Will be reordered by the partsView_inside_optimal) + Shape boundary_inside_minimum_; //!< The boundary within which to comb. (Will be reordered by the partsView_inside_minimum) + Shape boundary_inside_optimal_; //!< The boundary within which to comb. (Will be reordered by the partsView_inside_optimal) const PartsView parts_view_inside_minimum_; //!< Structured indices onto boundary_inside_minimum which shows which polygons belong to which part. const PartsView parts_view_inside_optimal_; //!< Structured indices onto boundary_inside_optimal which shows which polygons belong to which part. std::unique_ptr inside_loc_to_line_minimum_; //!< The SparsePointGridInclusive mapping locations to line segments of the inner boundary. std::unique_ptr inside_loc_to_line_optimal_; //!< The SparsePointGridInclusive mapping locations to line segments of the inner boundary. - std::unordered_map boundary_outside_; //!< The boundary outside of which to stay to avoid collision with other layer parts. This is a pointer cause we only - //!< compute it when we move outside the boundary (so not when there is only a single part in the layer) - std::unordered_map model_boundary_; //!< The boundary of the model itself + std::unordered_map boundary_outside_; //!< The boundary outside of which to stay to avoid collision with other layer parts. This is a pointer cause we only + //!< compute it when we move outside the boundary (so not when there is only a single part in the layer) + std::unordered_map model_boundary_; //!< The boundary of the model itself std::unordered_map> outside_loc_to_line_; //!< The SparsePointGridInclusive mapping locations to line segments of the outside boundary. std::unordered_map> model_boundary_loc_to_line_; //!< The SparsePointGridInclusive mapping locations to line segments of the model boundary @@ -157,7 +159,7 @@ class Comb /*! * Get the boundary_outside, which is an offset from the outlines of all meshes in the layer. Calculate it when it hasn't been calculated yet. */ - Polygons& getBoundaryOutside(const ExtruderTrain& train); + Shape& getBoundaryOutside(const ExtruderTrain& train); /*! * Get the SparsePointGridInclusive mapping locations to line segments of the model boundary. Calculate it when it hasn't been calculated yet. @@ -167,7 +169,7 @@ class Comb /*! * Get the boundary_outside, which is an offset from the outlines of all meshes in the layer. Calculate it when it hasn't been calculated yet. */ - Polygons& getModelBoundary(const ExtruderTrain& train); + Shape& getModelBoundary(const ExtruderTrain& train); /*! * Move the startPoint or endPoint inside when it should be inside @@ -177,9 +179,9 @@ class Comb * \param start_inside_poly[out] The polygon in which the point has been moved * \return Whether we have moved the point inside */ - bool moveInside(Polygons& boundary_inside, bool is_inside, LocToLineGrid* inside_loc_to_line, Point2LL& dest_point, size_t& start_inside_poly); + bool moveInside(Shape& boundary_inside, bool is_inside, LocToLineGrid* inside_loc_to_line, Point2LL& dest_point, size_t& start_inside_poly); - void moveCombPathInside(Polygons& boundary_inside, Polygons& boundary_inside_optimal, CombPath& comb_path_input, CombPath& comb_path_output); + void moveCombPathInside(Shape& boundary_inside, Shape& boundary_inside_optimal, CombPath& comb_path_input, CombPath& comb_path_output); public: /*! @@ -206,8 +208,8 @@ class Comb Comb( const SliceDataStorage& storage, const LayerIndex layer_nr, - const Polygons& comb_boundary_inside_minimum, - const Polygons& comb_boundary_inside_optimal, + const Shape& comb_boundary_inside_minimum, + const Shape& comb_boundary_inside_optimal, coord_t offset_from_outlines, coord_t travel_avoid_distance, coord_t move_inside_distance); diff --git a/include/pathPlanning/CombPath.h b/include/pathPlanning/CombPath.h index 6559e8c7ee..0b904d2a5a 100644 --- a/include/pathPlanning/CombPath.h +++ b/include/pathPlanning/CombPath.h @@ -4,7 +4,7 @@ #ifndef PATH_PLANNING_COMB_PATH_H #define PATH_PLANNING_COMB_PATH_H -#include "../utils/Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/pathPlanning/GCodePath.h b/include/pathPlanning/GCodePath.h index 8ac4286082..09160202d9 100644 --- a/include/pathPlanning/GCodePath.h +++ b/include/pathPlanning/GCodePath.h @@ -10,9 +10,9 @@ #include "GCodePathConfig.h" #include "SpaceFillType.h" #include "TimeMaterialEstimates.h" +#include "geometry/Point2LL.h" #include "settings/types/Ratio.h" #include "sliceDataStorage.h" -#include "utils/Point2LL.h" namespace cura { diff --git a/include/pathPlanning/LinePolygonsCrossings.h b/include/pathPlanning/LinePolygonsCrossings.h index 5fb641dd9b..d937a35c03 100644 --- a/include/pathPlanning/LinePolygonsCrossings.h +++ b/include/pathPlanning/LinePolygonsCrossings.h @@ -4,9 +4,10 @@ #ifndef PATH_PLANNING_LINE_POLYGONS_CROSSINGS_H #define PATH_PLANNING_LINE_POLYGONS_CROSSINGS_H -#include "../utils/polygon.h" -#include "../utils/polygonUtils.h" #include "CombPath.h" +#include "geometry/PointMatrix.h" +#include "geometry/Polygon.h" +#include "utils/polygonUtils.h" namespace cura { @@ -47,7 +48,7 @@ class LinePolygonsCrossings std::vector crossings_; //!< All crossings of polygons in the LinePolygonsCrossings::boundary with the scanline. - const Polygons& boundary_; //!< The boundary not to cross during combing. + const Shape& boundary_; //!< The boundary not to cross during combing. LocToLineGrid& loc_to_line_grid_; //!< Mapping from locations to line segments of \ref LinePolygonsCrossings::boundary Point2LL start_point_; //!< The start point of the scanline. Point2LL end_point_; //!< The end point of the scanline. @@ -124,7 +125,7 @@ class LinePolygonsCrossings * \param end the end point * \param dist_to_move_boundary_point_outside Distance used to move a point from a boundary so that it doesn't intersect with it anymore. (Precision issue) */ - LinePolygonsCrossings(const Polygons& boundary, LocToLineGrid& loc_to_line_grid, Point2LL& start, Point2LL& end, int64_t dist_to_move_boundary_point_outside) + LinePolygonsCrossings(const Shape& boundary, LocToLineGrid& loc_to_line_grid, Point2LL& start, Point2LL& end, int64_t dist_to_move_boundary_point_outside) : boundary_(boundary) , loc_to_line_grid_(loc_to_line_grid) , start_point_(start) @@ -145,7 +146,7 @@ class LinePolygonsCrossings * \return Whether combing succeeded, i.e. we didn't cross any gaps/other parts */ static bool comb( - const Polygons& boundary, + const Shape& boundary, LocToLineGrid& loc_to_line_grid, Point2LL startPoint, Point2LL endPoint, diff --git a/include/PathOrdering.h b/include/path_ordering.h similarity index 96% rename from include/PathOrdering.h rename to include/path_ordering.h index 0ec185e700..5a5de96e17 100644 --- a/include/PathOrdering.h +++ b/include/path_ordering.h @@ -50,7 +50,7 @@ struct PathOrdering * Vertex data, converted into a Polygon so that the orderer knows how * to deal with this data. */ - ConstPolygonPointer converted_; + const PointsSet* converted_{ nullptr }; /*! * Which vertex along the path to start printing with. @@ -76,7 +76,6 @@ struct PathOrdering */ bool backwards_; - /*! * Get vertex data from the custom path type. * @@ -89,7 +88,7 @@ struct PathOrdering * for each different type that this class is used with. See the .cpp file * for examples and where to add a new specialization. */ - ConstPolygonRef getVertexData(); + const PointsSet& getVertexData(); protected: /*! @@ -100,7 +99,7 @@ struct PathOrdering * For example, if the ``PathType`` is a list of ``ExtrusionJunction``s, * this will store the coordinates of those junctions. */ - std::optional cached_vertices_; + std::optional cached_vertices_; }; } // namespace cura diff --git a/include/plugins/converters.h b/include/plugins/converters.h index 9af4e7cbae..de947a850c 100644 --- a/include/plugins/converters.h +++ b/include/plugins/converters.h @@ -27,13 +27,15 @@ #include "cura/plugins/slots/postprocess/v0/modify.pb.h" #include "cura/plugins/slots/simplify/v0/modify.grpc.pb.h" #include "cura/plugins/slots/simplify/v0/modify.pb.h" +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/Polygon.h" #include "pathPlanning/GCodePath.h" #include "pathPlanning/SpeedDerivatives.h" #include "plugins/metadata.h" #include "plugins/types.h" #include "settings/Settings.h" #include "settings/types/LayerIndex.h" -#include "utils/polygon.h" namespace cura::plugins @@ -81,12 +83,12 @@ struct handshake_response : public details::converter +struct simplify_request : public details::converter { value_type operator()(const native_value_type& polygons, const coord_t max_resolution, const coord_t max_deviation, const coord_t max_area_deviation) const; }; -struct simplify_response : public details::converter +struct simplify_response : public details::converter { native_value_type operator()([[maybe_unused]] const native_value_type& original_value, const value_type& message) const; }; @@ -101,13 +103,13 @@ struct postprocess_response : public details::converter +struct infill_generate_request : public details::converter { value_type operator()(const native_value_type& inner_contour, const std::string& pattern, const Settings& settings) const; }; struct infill_generate_response - : public details::converter>, Polygons, Polygons>> + : public details::converter>, Shape, OpenLinesSet>> { native_value_type operator()(const value_type& message) const; }; diff --git a/include/plugins/slotproxy.h b/include/plugins/slotproxy.h index 037db10a19..82dd57d2a5 100644 --- a/include/plugins/slotproxy.h +++ b/include/plugins/slotproxy.h @@ -4,20 +4,19 @@ #ifndef PLUGINS_SLOTPROXY_H #define PLUGINS_SLOTPROXY_H -#include "plugins/converters.h" -#include "plugins/pluginproxy.h" -#include "plugins/types.h" -#include "plugins/validator.h" -#include "utils/types/char_range_literal.h" - -#include - #include #include #include #include #include +#include + +#include "plugins/pluginproxy.h" +#include "plugins/types.h" +#include "plugins/validator.h" +#include "utils/types/char_range_literal.h" + namespace cura::plugins { diff --git a/include/plugins/slots.h b/include/plugins/slots.h index d313b266cf..30740be3c4 100644 --- a/include/plugins/slots.h +++ b/include/plugins/slots.h @@ -10,22 +10,16 @@ #include #include -#include "WallToolPaths.h" #include "cura/plugins/slots/broadcast/v0/broadcast.grpc.pb.h" #include "cura/plugins/slots/gcode_paths/v0/modify.grpc.pb.h" #include "cura/plugins/slots/infill/v0/generate.grpc.pb.h" #include "cura/plugins/slots/postprocess/v0/modify.grpc.pb.h" #include "cura/plugins/slots/simplify/v0/modify.grpc.pb.h" #include "cura/plugins/v0/slot_id.pb.h" -#include "infill.h" -#include "plugins/converters.h" #include "plugins/slotproxy.h" #include "plugins/types.h" #include "plugins/validator.h" -#include "utils/Point2LL.h" #include "utils/Simplify.h" // TODO: Remove once the simplify slot has been removed -#include "utils/polygon.h" -#include "utils/types/char_range_literal.h" namespace cura { @@ -52,7 +46,7 @@ struct simplify_default struct infill_generate_default { - std::tuple, Polygons, Polygons> operator()([[maybe_unused]] auto&&... args) + std::tuple, Shape, OpenLinesSet> operator()([[maybe_unused]] auto&&... args) { // this code is only reachable when no slot is registered while the infill type is requested to be // generated by a plugin; this should not be possible to set up in the first place. Return an empty diff --git a/include/plugins/types.h b/include/plugins/types.h index eb443d0b27..a8fa18fad9 100644 --- a/include/plugins/types.h +++ b/include/plugins/types.h @@ -11,8 +11,6 @@ #include #include "cura/plugins/v0/slot_id.pb.h" -#include "utils/Point2LL.h" -#include "utils/polygon.h" namespace fmt { diff --git a/include/settings/EnumSettings.h b/include/settings/EnumSettings.h index 18bd091569..bc6ea72cee 100644 --- a/include/settings/EnumSettings.h +++ b/include/settings/EnumSettings.h @@ -275,7 +275,7 @@ enum class BrimLocation /*! * Convenience binary operator to allow testing brim location easily, like (actual_location & BrimLocation::OUTSIDE) */ -static int operator&(BrimLocation location1, BrimLocation location2) +[[maybe_unused]] static int operator&(BrimLocation location1, BrimLocation location2) { return static_cast(location1) & static_cast(location2); } diff --git a/include/settings/ZSeamConfig.h b/include/settings/ZSeamConfig.h index db1fb49ef9..9626786690 100644 --- a/include/settings/ZSeamConfig.h +++ b/include/settings/ZSeamConfig.h @@ -4,8 +4,8 @@ #ifndef ZSEAMCONFIG_H #define ZSEAMCONFIG_H -#include "../utils/Point2LL.h" //To store the preferred seam position. #include "EnumSettings.h" //For EZSeamType and EZSeamCornerPrefType. +#include "geometry/Point2LL.h" //To store the preferred seam position. namespace cura { diff --git a/include/settings/types/Angle.h b/include/settings/types/Angle.h index f6a7665b00..f2814ee3c8 100644 --- a/include/settings/types/Angle.h +++ b/include/settings/types/Angle.h @@ -1,12 +1,11 @@ -// Copyright (c) 2021 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef ANGLE_H #define ANGLE_H #include //For fmod. - -#include "../../utils/math.h" //For PI. +#include #define TAU (2.0 * std::numbers::pi) @@ -27,10 +26,7 @@ class AngleDegrees /* * \brief Default constructor setting the angle to 0. */ - AngleDegrees() - : value_(0.0) - { - } + AngleDegrees() noexcept = default; /* * \brief Converts radians to degrees. @@ -41,7 +37,7 @@ class AngleDegrees * \brief Casts a double to an AngleDegrees instance. */ AngleDegrees(double value) - : value_(std::fmod(std::fmod(value, 360) + 360, 360)) + : value_{ std::fmod(std::fmod(value, 360) + 360, 360) } { } @@ -60,11 +56,13 @@ class AngleDegrees { return std::fmod(std::fmod(value_ + other.value_, 360) + 360, 360); } + template AngleDegrees operator+(const T& other) const { return operator+(AngleDegrees(static_cast(other))); } + AngleDegrees& operator+=(const AngleDegrees& other) { value_ = std::fmod(std::fmod(value_ + other.value_, 360) + 360, 360); @@ -74,11 +72,13 @@ class AngleDegrees { return std::fmod(std::fmod(value_ - other.value_, 360) + 360, 360); } + template AngleDegrees operator-(const T& other) const { return operator-(AngleDegrees(static_cast(other))); } + AngleDegrees& operator-=(const AngleDegrees& other) { value_ = std::fmod(std::fmod(value_ - other.value_, 360) + 360, 360); @@ -90,7 +90,7 @@ class AngleDegrees * * This value should always be between 0 and 360. */ - double value_ = 0; + double value_{ 0 }; }; /* @@ -105,10 +105,7 @@ class AngleRadians /* * \brief Default constructor setting the angle to 0. */ - AngleRadians() - : value_(0.0) - { - } + AngleRadians() noexcept = default; /*! * \brief Converts an angle from degrees into radians. @@ -138,15 +135,18 @@ class AngleRadians { return std::fmod(std::fmod(value_ + other.value_, TAU) + TAU, TAU); } + AngleRadians& operator+=(const AngleRadians& other) { value_ = std::fmod(std::fmod(value_ + other.value_, TAU) + TAU, TAU); return *this; } + AngleRadians operator-(const AngleRadians& other) const { return std::fmod(std::fmod(value_ - other.value_, TAU) + TAU, TAU); } + AngleRadians& operator-=(const AngleRadians& other) { value_ = std::fmod(std::fmod(value_ - other.value_, TAU) + TAU, TAU); @@ -158,15 +158,16 @@ class AngleRadians * * This value should always be between 0 and 2 * pi. */ - double value_ = 0; + double value_{ 0 }; }; inline AngleDegrees::AngleDegrees(const AngleRadians& value) : value_(value * 360 / TAU) { } + inline AngleRadians::AngleRadians(const AngleDegrees& value) - : value_(double(value) * TAU / 360.0) + : value_(static_cast(value) * TAU / 360.0) { } diff --git a/include/skin.h b/include/skin.h index e0367cbf0c..fe7d555ef7 100644 --- a/include/skin.h +++ b/include/skin.h @@ -10,7 +10,7 @@ namespace cura { -class Polygons; +class Shape; class SkinPart; class SliceLayerPart; class SliceMeshStorage; @@ -96,7 +96,7 @@ class SkinInfillAreaComputation * above. The input is the area within the inner walls (or an empty Polygons * object). */ - void calculateTopSkin(const SliceLayerPart& part, Polygons& upskin); + void calculateTopSkin(const SliceLayerPart& part, Shape& upskin); /*! * \brief Calculate the basic areas which have air below. @@ -105,7 +105,7 @@ class SkinInfillAreaComputation * layers above. The input is the area within the inner walls (or an empty * Polygons object). */ - void calculateBottomSkin(const SliceLayerPart& part, Polygons& downskin); + void calculateBottomSkin(const SliceLayerPart& part, Shape& downskin); /*! * Apply skin expansion: @@ -116,7 +116,7 @@ class SkinInfillAreaComputation * \param[in,out] upskin The top skin areas to grow * \param[in,out] downskin The bottom skin areas to grow */ - void applySkinExpansion(const Polygons& original_outline, Polygons& upskin, Polygons& downskin); + void applySkinExpansion(const Shape& original_outline, Shape& upskin, Shape& downskin); /*! * Generate infill of a given part @@ -150,7 +150,7 @@ class SkinInfillAreaComputation * \param part Where to get the SkinParts to get the outline info from * \param roofing_layer_count The number of layers above the layer which we are looking into */ - Polygons generateFilledAreaAbove(SliceLayerPart& part, size_t roofing_layer_count); + Shape generateFilledAreaAbove(SliceLayerPart& part, size_t roofing_layer_count); /*! * Helper function to calculate and return the areas which are 'directly' above air. @@ -158,7 +158,7 @@ class SkinInfillAreaComputation * \param part Where to get the SkinParts to get the outline info from * \param flooring_layer_count The number of layers below the layer which we are looking into */ - Polygons generateFilledAreaBelow(SliceLayerPart& part, size_t flooring_layer_count); + Shape generateFilledAreaBelow(SliceLayerPart& part, size_t flooring_layer_count); protected: LayerIndex layer_nr_; //!< The index of the layer for which to generate the skins and infill. @@ -191,7 +191,7 @@ class SkinInfillAreaComputation * \param part_here The part for which to check. * \param layer2_nr The layer index from which to gather the outlines. */ - Polygons getOutlineOnLayer(const SliceLayerPart& part_here, const LayerIndex layer2_nr); + Shape getOutlineOnLayer(const SliceLayerPart& part_here, const LayerIndex layer2_nr); }; } // namespace cura diff --git a/include/sliceDataStorage.h b/include/sliceDataStorage.h index ebdf20430b..669e6829d0 100644 --- a/include/sliceDataStorage.h +++ b/include/sliceDataStorage.h @@ -13,14 +13,19 @@ #include "SupportInfillPart.h" #include "TopSurface.h" #include "WipeScriptConfig.h" +#include "geometry/LinesSet.h" +#include "geometry/MixedLinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Point2LL.h" +#include "geometry/Polygon.h" +#include "geometry/SingleShape.h" #include "settings/Settings.h" //For MAX_EXTRUDERS. #include "settings/types/Angle.h" //Infill angles. #include "settings/types/LayerIndex.h" #include "utils/AABB.h" #include "utils/AABB3D.h" #include "utils/NoCopy.h" -#include "utils/Point2LL.h" -#include "utils/polygon.h" // libArachne #include "utils/ExtrusionLine.h" @@ -40,12 +45,12 @@ class LightningGenerator; class SkinPart { public: - PolygonsPart outline; //!< The skinOutline is the area which needs to be 100% filled to generate a proper top&bottom filling. It's filled by the "skin" module. Includes both - //!< roofing and non-roofing. - Polygons skin_fill; //!< The part of the skin which is not roofing. - Polygons roofing_fill; //!< The inner infill which has air directly above - Polygons top_most_surface_fill; //!< The inner infill of the uppermost top layer which has air directly above. - Polygons bottom_most_surface_fill; //!< The inner infill of the bottommost bottom layer which has air directly below. + SingleShape outline; //!< The skinOutline is the area which needs to be 100% filled to generate a proper top&bottom filling. It's filled by the "skin" module. Includes both + //!< roofing and non-roofing. + Shape skin_fill; //!< The part of the skin which is not roofing. + Shape roofing_fill; //!< The inner infill which has air directly above + Shape top_most_surface_fill; //!< The inner infill of the uppermost top layer which has air directly above. + Shape bottom_most_surface_fill; //!< The inner infill of the bottommost bottom layer which has air directly below. }; /*! @@ -59,13 +64,12 @@ class SliceLayerPart AABB boundaryBox; //!< The boundaryBox is an axis-aligned boundary box which is used to quickly check for possible //!< collision between different parts on different layers. It's an optimization used during //!< skin calculations. - PolygonsPart outline; //!< The outline is the first member that is filled, and it's filled with polygons that match - //!< a cross-section of the 3D model. The first polygon is the outer boundary polygon and the - //!< rest are holes. - Polygons print_outline; //!< An approximation to the outline of what's actually printed, based on the outer wall. - //!< Too small parts will be omitted compared to the outline. - Polygons spiral_wall; //!< The centerline of the wall used by spiralize mode. Only computed if spiralize mode is enabled. - Polygons inner_area; //!< The area of the outline, minus the walls. This will be filled with either skin or infill. + SingleShape outline; //!< The outline is the first member that is filled, and it's filled with polygons that match + //!< a cross-section of the 3D model. + Shape print_outline; //!< An approximation to the outline of what's actually printed, based on the outer wall. + //!< Too small parts will be omitted compared to the outline. + Shape spiral_wall; //!< The centerline of the wall used by spiralize mode. Only computed if spiralize mode is enabled. + Shape inner_area; //!< The area of the outline, minus the walls. This will be filled with either skin or infill. std::vector skin_parts; //!< The skin parts which are filled for 100% with lines and/or insets. std::vector wall_toolpaths; //!< toolpaths for walls, will replace(?) the insets. Binned by inset_idx. std::vector infill_wall_toolpaths; //!< toolpaths for the walls of the infill areas. Binned by inset_idx. @@ -75,7 +79,7 @@ class SliceLayerPart * Like SliceLayerPart::outline, this class member is not used to actually determine the feature area, * but is used to compute the inside comb boundary. */ - Polygons infill_area; + Shape infill_area; /*! * The areas which need to be filled with sparse (0-99%) infill. @@ -86,7 +90,7 @@ class SliceLayerPart * * If these polygons are not initialized, simply use the normal infill area. */ - std::optional infill_area_own; + std::optional infill_area_own; /*! * The areas which need to be filled with sparse (0-99%) infill for different thicknesses. @@ -133,21 +137,21 @@ class SliceLayerPart * infill_area[x] will lie fully inside infill_area[x+1]. * infill_area_per_combine_per_density.back()[0] == part.infill area initially */ - std::vector> infill_area_per_combine_per_density; + std::vector> infill_area_per_combine_per_density; /*! * Get the infill_area_own (or when it's not instantiated: the normal infill_area) * \see SliceLayerPart::infill_area_own * \return the own infill area */ - Polygons& getOwnInfillArea(); + Shape& getOwnInfillArea(); /*! * Get the infill_area_own (or when it's not instantiated: the normal infill_area) * \see SliceLayerPart::infill_area_own * \return the own infill area */ - const Polygons& getOwnInfillArea() const; + const Shape& getOwnInfillArea() const; /*! * Searches whether the part has any walls in the specified inset index @@ -166,7 +170,7 @@ class SliceLayer coord_t printZ; //!< The height at which this layer needs to be printed. Can differ from sliceZ due to the raft. coord_t thickness; //!< The thickness of this layer. Can be different when using variable layer heights. std::vector parts; //!< An array of LayerParts which contain the actual data. The parts are printed one at a time to minimize travel outside of the 3D model. - Polygons openPolyLines; //!< A list of lines which were never hooked up into a 2D polygon. (Currently unused in normal operation) + OpenLinesSet open_polylines; //!< A list of lines which were never hooked up into a 2D polygon. (Currently unused in normal operation) /*! * \brief The parts of the model that are exposed at the very top of the @@ -181,7 +185,7 @@ class SliceLayer * * Note: Filled only when needed. */ - Polygons bottom_surface; + Shape bottom_surface; /*! * Get the all outlines of all layer parts in this layer. @@ -189,7 +193,7 @@ class SliceLayer * \param external_polys_only Whether to only include the outermost outline of each layer part * \return A collection of all the outline polygons */ - Polygons getOutlines(bool external_polys_only = false) const; + Shape getOutlines(bool external_polys_only = false) const; /*! * Get the all outlines of all layer parts in this layer. @@ -198,7 +202,7 @@ class SliceLayer * \param external_polys_only Whether to only include the outermost outline of each layer part * \param result The result: a collection of all the outline polygons */ - void getOutlines(Polygons& result, bool external_polys_only = false) const; + void getOutlines(Shape& result, bool external_polys_only = false) const; ~SliceLayer(); }; @@ -210,14 +214,14 @@ class SupportLayer { public: std::vector support_infill_parts; //!< a list of support infill parts - Polygons support_bottom; //!< Piece of support below the support and above the model. This must not overlap with any of the support_infill_parts or support_roof. - Polygons support_roof; //!< Piece of support above the support and below the model. This must not overlap with any of the support_infill_parts or support_bottom. - // NOTE: This is _all_ of the support_roof, and as such, overlaps with support_fractional_roof! - Polygons support_fractional_roof; //!< If the support distance is not exactly a multiple of the layer height, - // the first part of support just underneath the model needs to be printed at a fracional layer height. - Polygons support_mesh_drop_down; //!< Areas from support meshes which should be supported by more support - Polygons support_mesh; //!< Areas from support meshes which should NOT be supported by more support - Polygons anti_overhang; //!< Areas where no overhang should be detected. + Shape support_bottom; //!< Piece of support below the support and above the model. This must not overlap with any of the support_infill_parts or support_roof. + Shape support_roof; //!< Piece of support above the support and below the model. This must not overlap with any of the support_infill_parts or support_bottom. + // NOTE: This is _all_ of the support_roof, and as such, overlaps with support_fractional_roof! + Shape support_fractional_roof; //!< If the support distance is not exactly a multiple of the layer height, + // the first part of support just underneath the model needs to be printed at a fracional layer height. + Shape support_mesh_drop_down; //!< Areas from support meshes which should be supported by more support + Shape support_mesh; //!< Areas from support meshes which should NOT be supported by more support + Shape anti_overhang; //!< Areas where no overhang should be detected. /*! * Exclude the given polygons from the support infill areas and update the SupportInfillParts. @@ -225,7 +229,7 @@ class SupportLayer * \param exclude_polygons The polygons to exclude * \param exclude_polygons_boundary_box The boundary box for the polygons to exclude */ - void excludeAreasFromSupportInfillAreas(const Polygons& exclude_polygons, const AABB& exclude_polygons_boundary_box); + void excludeAreasFromSupportInfillAreas(const Shape& exclude_polygons, const AABB& exclude_polygons_boundary_box); /* Fill up the infill parts for the support with the given support polygons. The support polygons will be split into parts. * @@ -238,14 +242,14 @@ class SupportLayer * \param custom_line_distance (optional, default to 0) Distance between lines of the infill pattern. custom_line_distance of 0 means use the default instead. */ void fillInfillParts( - const Polygons& area, + const Shape& area, const coord_t support_line_width, const coord_t wall_line_count, const bool use_fractional_config = false, const bool unionAll = false, const coord_t custom_line_distance = 0) { - for (const PolygonsPart& island_outline : area.splitIntoParts(unionAll)) + for (const SingleShape& island_outline : area.splitIntoParts(unionAll)) { support_infill_parts.emplace_back(island_outline, support_line_width, use_fractional_config, wall_line_count, custom_line_distance); } @@ -266,7 +270,7 @@ class SupportLayer */ void fillInfillParts( const LayerIndex layer_nr, - const std::vector& support_fill_per_layer, + const std::vector& support_fill_per_layer, const coord_t infill_layer_height, const std::vector>& meshes, const coord_t support_line_width, @@ -310,11 +314,11 @@ class SliceMeshStorage std::vector infill_angles; //!< a list of angle values which is cycled through to determine the infill angle of each layer std::vector roofing_angles; //!< a list of angle values which is cycled through to determine the roofing angle of each layer std::vector skin_angles; //!< a list of angle values which is cycled through to determine the skin angle of each layer - std::vector overhang_areas; //!< For each layer the areas that are classified as overhang on this mesh. - std::vector full_overhang_areas; //!< For each layer the full overhang without the tangent of the overhang angle removed, such that the overhang area adjoins the - //!< areas of the next layers. - std::vector> overhang_points; //!< For each layer a list of points where point-overhang is detected. This is overhang that hasn't got any surface area, - //!< such as a corner pointing downwards. + std::vector overhang_areas; //!< For each layer the areas that are classified as overhang on this mesh. + std::vector full_overhang_areas; //!< For each layer the full overhang without the tangent of the overhang angle removed, such that the overhang area adjoins the + //!< areas of the next layers. + std::vector> overhang_points; //!< For each layer a list of points where point-overhang is detected. This is overhang that hasn't got any surface area, + //!< such as a corner pointing downwards. AABB3D bounding_box; //!< the mesh's bounding box std::shared_ptr base_subdiv_cube; @@ -359,15 +363,6 @@ class SliceMeshStorage Point2LL getZSeamHint() const; }; -/*! - * Class to store all open polylines or closed polygons related to one outset index of brim/skirt. - */ -struct SkirtBrimLine -{ - Polygons open_polylines; - Polygons closed_polygons; -}; - class SliceDataStorage : public NoCopy { public: @@ -381,25 +376,25 @@ class SliceDataStorage : public NoCopy SupportStorage support; - std::vector skirt_brim[MAX_EXTRUDERS]; //!< Skirt/brim polygons per extruder, ordered from inner to outer polygons. - Polygons support_brim; //!< brim lines for support, going from the edge of the support inward. \note Not ordered by inset. + std::vector skirt_brim[MAX_EXTRUDERS]; //!< Skirt/brim polygons per extruder, ordered from inner to outer polygons. + ClosedLinesSet support_brim; //!< brim lines for support, going from the edge of the support inward. \note Not ordered by inset. // Storage for the outline of the raft-parts. Will be filled with lines when the GCode is generated. - Polygons raftBaseOutline; - Polygons raftInterfaceOutline; - Polygons raftSurfaceOutline; + Shape raft_base_outline; + Shape raft_interface_outline; + Shape raft_surface_outline; int max_print_height_second_to_last_extruder; //!< Used in multi-extrusion: the layer number beyond which all models are printed with the same extruder std::vector max_print_height_per_extruder; //!< For each extruder the highest layer number at which it is used. std::vector max_print_height_order; //!< Ordered indices into max_print_height_per_extruder: back() will return the extruder number with the highest print height. std::vector spiralize_seam_vertex_indices; //!< the index of the seam vertex for each layer - std::vector spiralize_wall_outlines; //!< the wall outline polygons for each layer + std::vector spiralize_wall_outlines; //!< the wall outline polygons for each layer PrimeTower primeTower; - std::vector oozeShield; // oozeShield per layer - Polygons draft_protection_shield; //!< The polygons for a heightened skirt which protects from warping by gusts of wind and acts as a heated chamber. + std::vector ooze_shield; // oozeShield per layer + Shape draft_protection_shield; //!< The polygons for a heightened skirt which protects from warping by gusts of wind and acts as a heated chamber. /*! * \brief Creates a new slice data storage that stores the slice data of the @@ -422,7 +417,7 @@ class SliceDataStorage : public NoCopy * \param external_polys_only Whether to disregard all hole polygons. * \param extruder_nr (optional) only give back outlines for this extruder (where the walls are printed with this extruder) */ - Polygons getLayerOutlines( + Shape getLayerOutlines( const LayerIndex layer_nr, const bool include_support, const bool include_prime_tower, @@ -458,9 +453,9 @@ class SliceDataStorage : public NoCopy * Gets the border of the usable print area for this machine. * * \param extruder_nr The extruder for which to return the allowed areas. -1 if the areas allowed for all extruders should be returned. - * \return the Polygons representing the usable area of the print bed. + * \return the Shape representing the usable area of the print bed. */ - Polygons getMachineBorder(int extruder_nr = -1) const; + Shape getMachineBorder(int extruder_nr = -1) const; private: /*! diff --git a/include/slicer.h b/include/slicer.h index 37cd45f122..ddafd67ca3 100644 --- a/include/slicer.h +++ b/include/slicer.h @@ -1,4 +1,4 @@ -// Copyright (c) 2018 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef SLICER_H @@ -8,8 +8,11 @@ #include #include +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Shape.h" #include "settings/EnumSettings.h" -#include "utils/polygon.h" /* The Slicer creates layers of polygons from an optimized 3D model. @@ -56,12 +59,12 @@ class GapCloserResult class SlicerLayer { public: - std::vector segments; - std::unordered_map face_idx_to_segment_idx; // topology + std::vector segments_; + std::unordered_map face_idx_to_segment_idx_; // topology - int z = -1; - Polygons polygons; - Polygons openPolylines; + int z_ = -1; + Shape polygons_; + OpenLinesSet open_polylines_; /*! * \brief Connect the segments into polygons for this layer of this \p mesh. @@ -76,7 +79,7 @@ class SlicerLayer * * \param[in,out] open_polylines The polylines which are stiched, but couldn't be closed into a loop */ - void makeBasicPolygonLoops(Polygons& open_polylines); + void makeBasicPolygonLoops(OpenLinesSet& open_polylines); /*! * Connect the segments into a loop, starting from the segment with index \p start_segment_idx @@ -84,7 +87,7 @@ class SlicerLayer * \param[in,out] open_polylines The polylines which are stiched, but couldn't be closed into a loop * \param[in] start_segment_idx The index into SlicerLayer::segments for the first segment from which to start the polygon loop */ - void makeBasicPolygonLoop(Polygons& open_polylines, const size_t start_segment_idx); + void makeBasicPolygonLoop(OpenLinesSet& open_polylines, const size_t start_segment_idx); /*! * Get the next segment connected to the end of \p segment. @@ -104,7 +107,7 @@ class SlicerLayer * * \param[in,out] open_polylines The polylines which are stiched, but couldn't be closed into a loop */ - void connectOpenPolylines(Polygons& open_polylines); + void connectOpenPolylines(OpenLinesSet& open_polylines); /*! * Link up all the missing ends, closing up the smallest gaps first. This is an inefficient implementation which can run in O(n*n*n) time. @@ -113,7 +116,7 @@ class SlicerLayer * * \param[in,out] open_polylines The polylines which are stiched, but couldn't be closed into a loop yet */ - void stitch(Polygons& open_polylines); + void stitch(OpenLinesSet& open_polylines); std::optional findPolygonGapCloser(Point2LL ip0, Point2LL ip1); @@ -126,7 +129,7 @@ class SlicerLayer * * \param[in,out] open_polylines The polylines which are stiched, but couldn't be closed into a loop yet */ - void stitch_extensive(Polygons& open_polylines); + void stitch_extensive(OpenLinesSet& open_polylines); private: /*! @@ -417,7 +420,7 @@ class SlicerLayer * the order of a polyline. * \return The stitches that are allowed in order from best to worst. */ - std::priority_queue findPossibleStitches(const Polygons& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse) const; + std::priority_queue findPossibleStitches(const OpenLinesSet& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse) const; /*! Plans the best way to perform a stitch. * @@ -435,7 +438,7 @@ class SlicerLayer * \param[in,out] terminus_1 the Terminus on polyline_1 to join at. * \param[out] reverse Whether the polylines need to be reversed. */ - void planPolylineStitch(const Polygons& open_polylines, Terminus& terminus_0, Terminus& terminus_1, bool reverse[2]) const; + void planPolylineStitch(const OpenLinesSet& open_polylines, Terminus& terminus_0, Terminus& terminus_1, bool reverse[2]) const; /*! Joins polyline_1 onto polyline_0. * @@ -453,7 +456,7 @@ class SlicerLayer * polyline_0 and reverse[1] indicates whether to reverse * polyline_1 */ - void joinPolylines(PolygonRef& polyline_0, PolygonRef& polyline_1, const bool reverse[2]) const; + static void joinPolylines(OpenPolyline& polyline_0, OpenPolyline& polyline_1, const bool reverse[2]); /*! * Connecting polylines that are not closed yet. @@ -473,7 +476,7 @@ class SlicerLayer * \param[in] allow_reverse If true, then this function is allowed * to reverse edge directions to merge polylines. */ - void connectOpenPolylinesImpl(Polygons& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse); + void connectOpenPolylinesImpl(OpenLinesSet& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse); }; class Slicer diff --git a/include/support.h b/include/support.h index 81b1218929..0d6eaeffa6 100644 --- a/include/support.h +++ b/include/support.h @@ -8,7 +8,7 @@ #include #include "settings/types/LayerIndex.h" -#include "utils/polygon.h" +#include "utils/Coord_t.h" namespace cura { @@ -17,6 +17,8 @@ class Settings; class SliceDataStorage; class SliceMeshStorage; class Slicer; +class Polygon; +class Shape; class AreaSupport { @@ -74,8 +76,7 @@ class AreaSupport * \param global_support_areas_per_layer the global support areas per layer * \param total_layer_count total number of layers */ - static void - splitGlobalSupportAreasIntoSupportInfillParts(SliceDataStorage& storage, const std::vector& global_support_areas_per_layer, unsigned int total_layer_count); + static void splitGlobalSupportAreasIntoSupportInfillParts(SliceDataStorage& storage, const std::vector& global_support_areas_per_layer, unsigned int total_layer_count); /*! * Generate gradual support on the already generated support areas. This must be called after generateSupportAreas(). @@ -158,7 +159,7 @@ class AreaSupport const Settings& bottom_settings, const size_t mesh_idx, const size_t layer_count, - std::vector& support_areas); + std::vector& support_areas); /*! * Generate support bottom areas for a given mesh. @@ -172,7 +173,7 @@ class AreaSupport * \param mesh The mesh to generate support for. * \param global_support_areas_per_layer the global support areas on each layer. */ - static void generateSupportBottom(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer); + static void generateSupportBottom(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer); /*! * Generate support roof areas for a given mesh. @@ -186,7 +187,7 @@ class AreaSupport * \param mesh The mesh to generate support roof for. * \param global_support_areas_per_layer the global support areas on each layer. */ - static void generateSupportRoof(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer); + static void generateSupportRoof(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer); /*! * \brief Generate a single layer of support interface. @@ -206,12 +207,12 @@ class AreaSupport * \param[out] interface_polygons The resulting interface layer. Do not use `interface` in windows! */ static void generateSupportInterfaceLayer( - Polygons& support_areas, - const Polygons mesh_outlines, + Shape& support_areas, + const Shape mesh_outlines, const coord_t safety_offset, const coord_t outline_offset, const double minimum_interface_area, - Polygons& interface_polygons); + Shape& interface_polygons); /*! * \brief Join current support layer with the support of the layer above, @@ -221,7 +222,7 @@ class AreaSupport * \param supportLayer_this The overhang areas of the current layer at hand. * \return The joined support areas for this layer. */ - static Polygons join(const SliceDataStorage& storage, const Polygons& supportLayer_up, Polygons& supportLayer_this); + static Shape join(const SliceDataStorage& storage, const Shape& supportLayer_up, Shape& supportLayer_this); /*! * Move the support up from model (cut away polygons to ensure bottom z distance) @@ -239,9 +240,9 @@ class AreaSupport */ static void moveUpFromModel( const SliceDataStorage& storage, - Polygons& stair_removal, - Polygons& sloped_areas, - Polygons& support_areas, + Shape& stair_removal, + Shape& sloped_areas, + Shape& support_areas, const size_t layer_idx, const size_t bottom_empty_layer_count, const size_t bottom_stair_step_layer_count, @@ -272,7 +273,7 @@ class AreaSupport * \param layer_idx The layer for which to compute the overhang. * \return A pair of basic overhang and full overhang. */ - static std::pair computeBasicAndFullOverhang(const SliceDataStorage& storage, const SliceMeshStorage& mesh, const LayerIndex& layer_idx); + static std::pair computeBasicAndFullOverhang(const SliceDataStorage& storage, const SliceMeshStorage& mesh, const LayerIndex& layer_idx); /*! * \brief Adds tower pieces to the current support layer. @@ -290,10 +291,10 @@ class AreaSupport */ static void handleTowers( const Settings& settings, - const Polygons& xy_disallowed_area, - Polygons& supportLayer_this, - std::vector& tower_roofs, - std::vector>& overhang_points, + const Shape& xy_disallowed_area, + Shape& supportLayer_this, + std::vector& tower_roofs, + std::vector>& overhang_points, LayerIndex layer_idx, size_t layer_count); @@ -303,7 +304,7 @@ class AreaSupport * \param supportLayer_this The areas of the layer for which to handle the * wall struts. */ - static void handleWallStruts(const Settings& settings, Polygons& supportLayer_this); + static void handleWallStruts(const Settings& settings, Shape& supportLayer_this); /*! * Clean up the SupportInfillParts. @@ -323,7 +324,7 @@ class AreaSupport * \param layer_idx The layer for which the disallowed areas are to be calcualted * */ - static Polygons generateVaryingXYDisallowedArea(const SliceMeshStorage& storage, const LayerIndex layer_idx); + static Shape generateVaryingXYDisallowedArea(const SliceMeshStorage& storage, const LayerIndex layer_idx); }; diff --git a/include/utils/AABB.h b/include/utils/AABB.h index b69d8c2c57..e3a93c678b 100644 --- a/include/utils/AABB.h +++ b/include/utils/AABB.h @@ -4,14 +4,13 @@ #ifndef UTILS_AABB_H #define UTILS_AABB_H -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { -class ConstPolygonRef; class Polygon; -class Polygons; +class Shape; /* Axis aligned boundary box */ class AABB @@ -21,11 +20,11 @@ class AABB AABB(); //!< initializes with invalid min and max AABB(const Point2LL& min, const Point2LL& max); //!< initializes with given min and max - AABB(const Polygons& polys); //!< Computes the boundary box for the given polygons - AABB(ConstPolygonRef poly); //!< Computes the boundary box for the given polygons + AABB(const Shape& shape); //!< Computes the boundary box for the given shape + AABB(const Polygon& poly); //!< Computes the boundary box for the given polygons - void calculate(const Polygons& polys); //!< Calculates the aabb for the given polygons (throws away old min and max data of this aabb) - void calculate(ConstPolygonRef poly); //!< Calculates the aabb for the given polygon (throws away old min and max data of this aabb) + void calculate(const Shape& shape); //!< Calculates the aabb for the given shape (throws away old min and max data of this aabb) + void calculate(const Polygon& poly); //!< Calculates the aabb for the given polygon (throws away old min and max data of this aabb) /*! * Whether the bounding box contains the specified point. @@ -79,7 +78,9 @@ class AABB * * \param point The point to include in the bounding box. */ - void include(Point2LL point); + void include(const Point2LL& point); + + void include(const Polygon& polygon); /*! * \brief Includes the specified bounding box in the bounding box. @@ -90,7 +91,7 @@ class AABB * * \param other The bounding box to include in this one. */ - void include(const AABB other); + void include(const AABB& other); /*! * Expand the borders of the bounding box in each direction with the given amount diff --git a/include/utils/AABB3D.h b/include/utils/AABB3D.h index 48142d24ce..402135be66 100644 --- a/include/utils/AABB3D.h +++ b/include/utils/AABB3D.h @@ -4,7 +4,7 @@ #ifndef UTILS_AABB3D_H #define UTILS_AABB3D_H -#include "Point2LL.h" +#include "geometry/Point2LL.h" #include "utils/AABB.h" namespace cura diff --git a/include/utils/ExtrusionJunction.h b/include/utils/ExtrusionJunction.h index 66cb46c9cc..f431a73f94 100644 --- a/include/utils/ExtrusionJunction.h +++ b/include/utils/ExtrusionJunction.h @@ -5,7 +5,7 @@ #ifndef UTILS_EXTRUSION_JUNCTION_H #define UTILS_EXTRUSION_JUNCTION_H -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/ExtrusionLine.h b/include/utils/ExtrusionLine.h index 4b3eb2c84b..987ce1b783 100644 --- a/include/utils/ExtrusionLine.h +++ b/include/utils/ExtrusionLine.h @@ -10,7 +10,8 @@ #include #include "ExtrusionJunction.h" -#include "polygon.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" namespace cura { @@ -208,11 +209,7 @@ struct ExtrusionLine /*! * Sum the total length of this path. */ - coord_t getLength() const; - coord_t polylineLength() const - { - return getLength(); - } + coord_t length() const; /*! * Put all junction locations into a polygon object. @@ -224,55 +221,17 @@ struct ExtrusionLine Polygon ret; for (const ExtrusionJunction& j : junctions_) - ret.add(j.p_); + ret.push_back(j.p_); return ret; } - /*! - * Create a true-extrusion area shape for the path; this means that each junction follows the bead-width - * set for that junction. - */ - [[maybe_unused]] Polygons toExtrusionPolygons() const - { - Polygon poly; - - const auto add_line_direction = [&poly](const auto iterator) - { - for (const auto& element : iterator | ranges::views::sliding(2)) - { - const ExtrusionJunction& j1 = element[0]; - const ExtrusionJunction& j2 = element[1]; - - const auto dir = j2.p_ - j1.p_; - const auto normal = turn90CCW(dir); - const auto mag = vSize(normal); - - if (mag <= 5) - { - continue; - } - - poly.emplace_back(j1.p_ + normal * j1.w_ / mag / 2); - poly.emplace_back(j2.p_ + normal * j2.w_ / mag / 2); - } - }; - - // forward pass - add_line_direction(junctions_); - // backward pass - add_line_direction(junctions_ | ranges::views::reverse); - - Polygons paths; - paths.emplace_back(poly.poly); - ClipperLib::SimplifyPolygons(paths.paths, ClipperLib::pftNonZero); - return paths; - } - /*! * Get the minimal width of this path */ coord_t getMinimalWidth() const; + + bool shorterThan(const coord_t check_length) const; }; using VariableWidthLines = std::vector; //; + +} // namespace cura +#endif // UTILS_EXTRUSION_LINE_STITCHER_H diff --git a/include/utils/ExtrusionSegment.h b/include/utils/ExtrusionSegment.h index 99430f8789..87ca8176cf 100644 --- a/include/utils/ExtrusionSegment.h +++ b/include/utils/ExtrusionSegment.h @@ -1,15 +1,14 @@ -// Copyright (c) 2020 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef UTILS_EXTRUSION_SEGMENT_H #define UTILS_EXTRUSION_SEGMENT_H -#include +#include #include "ExtrusionJunction.h" -#include "Point2LL.h" -#include "polygon.h" +#include "geometry/Polygon.h" #include "polygonUtils.h" namespace cura @@ -55,7 +54,7 @@ class ExtrusionSegment * Converts this segment to an outline of the area that the segment covers. * \return The area that would be covered by this extrusion segment. */ - Polygons toPolygons(); + Shape toShape(); /*! * Converts this segment to an outline of the area that the segment covers. @@ -63,7 +62,7 @@ class ExtrusionSegment * it will be included in the next extrusion move. Overrides class field * \ref is_reduced . */ - Polygons toPolygons(bool reduced); + Shape toShape(bool reduced); /*! * Discretize a variable-line-width extrusion segment into multiple diff --git a/include/utils/HalfEdge.h b/include/utils/HalfEdge.h index ae8736f589..4f6f3671d5 100644 --- a/include/utils/HalfEdge.h +++ b/include/utils/HalfEdge.h @@ -7,8 +7,8 @@ #include #include -#include "../utils/Point2LL.h" #include "Coord_t.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/HalfEdgeNode.h b/include/utils/HalfEdgeNode.h index 163b45d07b..118655324c 100644 --- a/include/utils/HalfEdgeNode.h +++ b/include/utils/HalfEdgeNode.h @@ -6,7 +6,7 @@ #include -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/ListPolyIt.h b/include/utils/ListPolyIt.h index 0847305bdf..ed56a1825a 100644 --- a/include/utils/ListPolyIt.h +++ b/include/utils/ListPolyIt.h @@ -7,13 +7,18 @@ #include #include -#include "Point2LL.h" -#include "polygon.h" +#include "geometry/Point2LL.h" namespace cura { +class Polygon; +class Shape; + +using ListPolygon = std::list; //!< A polygon represented by a linked list instead of a vector +using ListPolygons = std::vector; //!< Polygons represented by a vector of linked lists instead of a vector of vectors + /*! * A wrapper class for a ListPolygon::iterator and a reference to the containing ListPolygon */ @@ -110,7 +115,7 @@ class ListPolyIt * \param polys The polygons to convert * \param result The converted polygons */ - static void convertPolygonsToLists(const Polygons& polys, ListPolygons& result); + static void convertPolygonsToLists(const Shape& shape, ListPolygons& result); /*! * Convert Polygons to ListPolygons @@ -118,7 +123,7 @@ class ListPolyIt * \param polys The polygons to convert * \param result The converted polygons */ - static void convertPolygonToList(ConstPolygonRef poly, ListPolygon& result); + static void convertPolygonToList(const Polygon& poly, ListPolygon& result); /*! * Convert ListPolygons to Polygons @@ -126,7 +131,7 @@ class ListPolyIt * \param list_polygons The polygons to convert * \param polygons The converted polygons */ - static void convertListPolygonsToPolygons(const ListPolygons& list_polygons, Polygons& polygons); + static void convertListPolygonsToPolygons(const ListPolygons& list_polygons, Shape& polygons); /*! * Convert ListPolygons to Polygons @@ -134,7 +139,7 @@ class ListPolyIt * \param list_polygons The polygons to convert * \param polygons The converted polygons */ - static void convertListPolygonToPolygon(const ListPolygon& list_polygon, PolygonRef polygon); + static void convertListPolygonToPolygon(const ListPolygon& list_polygon, Polygon& polygon); /*! * Insert a point into a ListPolygon if it's not a duplicate of the point before or the point after. diff --git a/include/utils/MinimumSpanningTree.h b/include/utils/MinimumSpanningTree.h index 599ab2273b..866c0d17e6 100644 --- a/include/utils/MinimumSpanningTree.h +++ b/include/utils/MinimumSpanningTree.h @@ -8,7 +8,7 @@ #include #include -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/MixedPolylineStitcher.h b/include/utils/MixedPolylineStitcher.h new file mode 100644 index 0000000000..9c092e1bcd --- /dev/null +++ b/include/utils/MixedPolylineStitcher.h @@ -0,0 +1,24 @@ +// Copyright (c) 2024 Ultimaker B.V. +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#ifndef UTILS_MIXED_POLYLINE_STITCHER_H +#define UTILS_MIXED_POLYLINE_STITCHER_H + +#include "PolylineStitcher.h" +#include "geometry/ClosedLinesSet.h" +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" + +namespace cura +{ + +class MixedLinesSet; + +class MixedPolylineStitcher : public PolylineStitcher +{ +public: + static void stitch(const OpenLinesSet& lines, MixedLinesSet& result, coord_t max_stitch_distance = MM2INT(0.1), coord_t snap_distance = 10); +}; + +} // namespace cura +#endif // UTILS_MIXED_POLYLINE_STITCHER_H diff --git a/include/utils/OpenPolylineStitcher.h b/include/utils/OpenPolylineStitcher.h new file mode 100644 index 0000000000..fb2a532f0b --- /dev/null +++ b/include/utils/OpenPolylineStitcher.h @@ -0,0 +1,18 @@ +// Copyright (c) 2024 Ultimaker B.V. +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#ifndef UTILS_OPEN_POLYLINE_STITCHER_H +#define UTILS_OPEN_POLYLINE_STITCHER_H + +#include "PolylineStitcher.h" +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/Shape.h" + +namespace cura +{ + +using OpenPolylineStitcher = PolylineStitcher; + +} // namespace cura +#endif // UTILS_OPEN_POLYLINE_STITCHER_H diff --git a/include/utils/Point2LL.h b/include/utils/Point2LL.h deleted file mode 100644 index 8623282fc2..0000000000 --- a/include/utils/Point2LL.h +++ /dev/null @@ -1,407 +0,0 @@ -// Copyright (c) 2020 Ultimaker B.V. -// CuraEngine is released under the terms of the AGPLv3 or higher. - -#ifndef UTILS_INT_POINT_H -#define UTILS_INT_POINT_H - -/** -The integer point classes are used as soon as possible and represent microns in 2D or 3D space. -Integer points are used to avoid floating point rounding errors, and because ClipperLib uses them. -*/ -#define INLINE static inline - -// Include Clipper to get the ClipperLib::IntPoint definition, which we reuse as Point definition. -#include -#include // for hash function object -#include // auto-serialization / auto-toString() -#include -#include -#include - -#include "../utils/math.h" // for PI. Use relative path to avoid pulling -#include "Point3LL.h" //For applying Point3Matrices. - -#ifdef __GNUC__ -#define DEPRECATED(func) func __attribute__((deprecated)) -#elif defined(_MSC_VER) -#define DEPRECATED(func) __declspec(deprecated) func -#else -#pragma message("WARNING: You need to implement DEPRECATED for this compiler") -#define DEPRECATED(func) func -#endif - - -namespace cura -{ - -/* 64bit Points are used mostly throughout the code, these are the 2D points from ClipperLib */ -typedef ClipperLib::IntPoint Point2LL; - -#define POINT_MIN std::numeric_limits::min() -#define POINT_MAX std::numeric_limits::max() - -static Point2LL no_point(std::numeric_limits::min(), std::numeric_limits::min()); - -/* Extra operators to make it easier to do math with the 64bit Point objects */ -INLINE Point2LL operator-(const Point2LL& p0) -{ - return Point2LL(-p0.X, -p0.Y); -} -INLINE Point2LL operator+(const Point2LL& p0, const Point2LL& p1) -{ - return Point2LL(p0.X + p1.X, p0.Y + p1.Y); -} -INLINE Point2LL operator-(const Point2LL& p0, const Point2LL& p1) -{ - return Point2LL(p0.X - p1.X, p0.Y - p1.Y); -} -INLINE Point2LL operator*(const Point2LL& p0, const coord_t i) -{ - return Point2LL(p0.X * i, p0.Y * i); -} -template::value, T>::type> // Use only for numeric types. -INLINE Point2LL operator*(const Point2LL& p0, const T i) -{ - return Point2LL(std::llrint(static_cast(p0.X) * i), std::llrint(static_cast(p0.Y) * i)); -} -template::value, T>::type> // Use only for numeric types. -INLINE Point2LL operator*(const T i, const Point2LL& p0) -{ - return p0 * i; -} -INLINE Point2LL operator/(const Point2LL& p0, const coord_t i) -{ - return Point2LL(p0.X / i, p0.Y / i); -} -INLINE Point2LL operator/(const Point2LL& p0, const Point2LL& p1) -{ - return Point2LL(p0.X / p1.X, p0.Y / p1.Y); -} -INLINE Point2LL operator%(const Point2LL& p0, const coord_t i) -{ - return Point2LL(p0.X % i, p0.Y % i); -} - -INLINE Point2LL& operator+=(Point2LL& p0, const Point2LL& p1) -{ - p0.X += p1.X; - p0.Y += p1.Y; - return p0; -} -INLINE Point2LL& operator-=(Point2LL& p0, const Point2LL& p1) -{ - p0.X -= p1.X; - p0.Y -= p1.Y; - return p0; -} - -INLINE bool operator<(const Point2LL& p0, const Point2LL& p1) -{ - return p0.X < p1.X || (p0.X == p1.X && p0.Y < p1.Y); -} - -/* ***** NOTE ***** - TL;DR: DO NOT implement operators *= and /= because of the default values in ClipperLib::IntPoint's constructor. - - We DO NOT implement operators *= and /= because the class Point is essentially a ClipperLib::IntPoint and it has a - constructor IntPoint(int x = 0, int y = 0), and this causes problems. If you implement *= as *=(int) and when you - do "Point a = a * 5", you probably intend to do "a.x *= 5" and "a.y *= 5", but with that constructor, it will create - an IntPoint(5, y = 0) and you end up with wrong results. - */ - -// INLINE bool operator==(const Point& p0, const Point& p1) { return p0.X==p1.X&&p0.Y==p1.Y; } -// INLINE bool operator!=(const Point& p0, const Point& p1) { return p0.X!=p1.X||p0.Y!=p1.Y; } - -INLINE coord_t vSize2(const Point2LL& p0) -{ - return p0.X * p0.X + p0.Y * p0.Y; -} -INLINE double vSize2f(const Point2LL& p0) -{ - return static_cast(p0.X) * static_cast(p0.X) + static_cast(p0.Y) * static_cast(p0.Y); -} - -INLINE bool shorterThen(const Point2LL& p0, const coord_t len) -{ - if (p0.X > len || p0.X < -len) - { - return false; - } - if (p0.Y > len || p0.Y < -len) - { - return false; - } - return vSize2(p0) <= len * len; -} - -INLINE bool shorterThan(const Point2LL& p0, const coord_t len) -{ - return shorterThen(p0, len); -} - -INLINE coord_t vSize(const Point2LL& p0) -{ - return std::llrint(sqrt(static_cast(vSize2(p0)))); -} - -INLINE double vSizeMM(const Point2LL& p0) -{ - double fx = INT2MM(p0.X); - double fy = INT2MM(p0.Y); - return sqrt(fx * fx + fy * fy); -} - -INLINE Point2LL normal(const Point2LL& p0, coord_t len) -{ - coord_t _len = vSize(p0); - if (_len < 1) - return Point2LL(len, 0); - return p0 * len / _len; -} - -INLINE Point2LL turn90CCW(const Point2LL& p0) -{ - return Point2LL(-p0.Y, p0.X); -} - -INLINE Point2LL rotate(const Point2LL& p0, double angle) -{ - const double cos_component = std::cos(angle); - const double sin_component = std::sin(angle); - const double x = static_cast(p0.X); - const double y = static_cast(p0.Y); - return Point2LL(std::llrint(cos_component * x - sin_component * y), std::llrint(sin_component * x + cos_component * y)); -} - -INLINE coord_t dot(const Point2LL& p0, const Point2LL& p1) -{ - return p0.X * p1.X + p0.Y * p1.Y; -} - -INLINE coord_t cross(const Point2LL& p0, const Point2LL& p1) -{ - return p0.X * p1.Y - p0.Y * p1.X; -} - -INLINE double angle(const Point2LL& p) -{ - double angle = std::atan2(p.X, p.Y) / std::numbers::pi * 180.0; - if (angle < 0.0) - angle += 360.0; - return angle; -} - -// Identity function, used to be able to make templated algorithms where the input is sometimes points, sometimes things that contain or can be converted to points. -INLINE const Point2LL& make_point(const Point2LL& p) -{ - return p; -} - -} // namespace cura - -namespace std -{ -template<> -struct hash -{ - size_t operator()(const cura::Point2LL& pp) const - { - static int prime = 31; - int result = 89; - result = static_cast(result * prime + pp.X); - result = static_cast(result * prime + pp.Y); - return static_cast(result); - } -}; -} // namespace std - -namespace cura -{ - -class PointMatrix -{ -public: - double matrix[4]; - - PointMatrix() - { - matrix[0] = 1; - matrix[1] = 0; - matrix[2] = 0; - matrix[3] = 1; - } - - PointMatrix(double rotation) - { - rotation = rotation / 180 * std::numbers::pi; - matrix[0] = cos(rotation); - matrix[1] = -sin(rotation); - matrix[2] = -matrix[1]; - matrix[3] = matrix[0]; - } - - PointMatrix(const Point2LL p) - { - matrix[0] = static_cast(p.X); - matrix[1] = static_cast(p.Y); - double f = sqrt((matrix[0] * matrix[0]) + (matrix[1] * matrix[1])); - matrix[0] /= f; - matrix[1] /= f; - matrix[2] = -matrix[1]; - matrix[3] = matrix[0]; - } - - static PointMatrix scale(double s) - { - PointMatrix ret; - ret.matrix[0] = s; - ret.matrix[3] = s; - return ret; - } - - Point2LL apply(const Point2LL p) const - { - const double x = static_cast(p.X); - const double y = static_cast(p.Y); - return Point2LL(std::llrint(x * matrix[0] + y * matrix[1]), std::llrint(x * matrix[2] + y * matrix[3])); - } - - /*! - * \warning only works on a rotation matrix! Output is incorrect for other types of matrix - */ - Point2LL unapply(const Point2LL p) const - { - const double x = static_cast(p.X); - const double y = static_cast(p.Y); - return Point2LL(std::llrint(x * matrix[0] + y * matrix[2]), std::llrint(x * matrix[1] + y * matrix[3])); - } - - PointMatrix inverse() const - { - PointMatrix ret; - double det = matrix[0] * matrix[3] - matrix[1] * matrix[2]; - ret.matrix[0] = matrix[3] / det; - ret.matrix[1] = -matrix[1] / det; - ret.matrix[2] = -matrix[2] / det; - ret.matrix[3] = matrix[0] / det; - return ret; - } -}; - -class Point3Matrix -{ -public: - double matrix[9]; - - Point3Matrix() - { - matrix[0] = 1; - matrix[1] = 0; - matrix[2] = 0; - matrix[3] = 0; - matrix[4] = 1; - matrix[5] = 0; - matrix[6] = 0; - matrix[7] = 0; - matrix[8] = 1; - } - - /*! - * Initializes the top left corner with the values of \p b - * and the rest as if it's a unit matrix - */ - Point3Matrix(const PointMatrix& b) - { - matrix[0] = b.matrix[0]; - matrix[1] = b.matrix[1]; - matrix[2] = 0; - matrix[3] = b.matrix[2]; - matrix[4] = b.matrix[3]; - matrix[5] = 0; - matrix[6] = 0; - matrix[7] = 0; - matrix[8] = 1; - } - - Point3LL apply(const Point3LL p) const - { - const double x = static_cast(p.x_); - const double y = static_cast(p.y_); - const double z = static_cast(p.z_); - return Point3LL( - std::llrint(x * matrix[0] + y * matrix[1] + z * matrix[2]), - std::llrint(x * matrix[3] + y * matrix[4] + z * matrix[5]), - std::llrint(x * matrix[6] + y * matrix[7] + z * matrix[8])); - } - - /*! - * Apply matrix to vector as homogeneous coordinates. - */ - Point2LL apply(const Point2LL p) const - { - Point3LL result = apply(Point3LL(p.X, p.Y, 1)); - return Point2LL(result.x_ / result.z_, result.y_ / result.z_); - } - - static Point3Matrix translate(const Point2LL p) - { - Point3Matrix ret; // uniform matrix - ret.matrix[2] = static_cast(p.X); - ret.matrix[5] = static_cast(p.Y); - return ret; - } - - Point3Matrix compose(const Point3Matrix& b) - { - Point3Matrix ret; - for (int outx = 0; outx < 3; outx++) - { - for (int outy = 0; outy < 3; outy++) - { - ret.matrix[outy * 3 + outx] = 0; - for (int in = 0; in < 3; in++) - { - ret.matrix[outy * 3 + outx] += matrix[outy * 3 + in] * b.matrix[in * 3 + outx]; - } - } - } - return ret; - } -}; - - -inline Point3LL operator+(const Point3LL& p3, const Point2LL& p2) -{ - return Point3LL(p3.x_ + p2.X, p3.y_ + p2.Y, p3.z_); -} -inline Point3LL& operator+=(Point3LL& p3, const Point2LL& p2) -{ - p3.x_ += p2.X; - p3.y_ += p2.Y; - return p3; -} - -inline Point2LL operator+(const Point2LL& p2, const Point3LL& p3) -{ - return Point2LL(p3.x_ + p2.X, p3.y_ + p2.Y); -} - - -inline Point3LL operator-(const Point3LL& p3, const Point2LL& p2) -{ - return Point3LL(p3.x_ - p2.X, p3.y_ - p2.Y, p3.z_); -} -inline Point3LL& operator-=(Point3LL& p3, const Point2LL& p2) -{ - p3.x_ -= p2.X; - p3.y_ -= p2.Y; - return p3; -} - -inline Point2LL operator-(const Point2LL& p2, const Point3LL& p3) -{ - return Point2LL(p2.X - p3.x_, p2.Y - p3.y_); -} - -} // namespace cura -#endif // UTILS_INT_POINT_H diff --git a/include/utils/Point3D.h b/include/utils/Point3D.h index 8405d03c4d..6000d9be76 100644 --- a/include/utils/Point3D.h +++ b/include/utils/Point3D.h @@ -7,7 +7,7 @@ #include #include -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura diff --git a/include/utils/Point3F.h b/include/utils/Point3F.h index 44262ad2b0..e67702937c 100644 --- a/include/utils/Point3F.h +++ b/include/utils/Point3F.h @@ -7,8 +7,8 @@ #include #include -#include "Point2LL.h" #include "Point3D.h" +#include "geometry/Point2LL.h" namespace cura diff --git a/include/utils/PolygonConnector.h b/include/utils/PolygonConnector.h index 3dc7b75dec..38a3c1fe00 100644 --- a/include/utils/PolygonConnector.h +++ b/include/utils/PolygonConnector.h @@ -1,4 +1,4 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef UTILS_POLYGON_CONNECTOR_H @@ -9,10 +9,12 @@ #endif #include -#include "Point2LL.h" +#include "geometry/Point2LL.h" +#include "geometry/Polygon.h" #include "linearAlg2D.h" -#include "polygon.h" #include "polygonUtils.h" +#include "settings/types/Ratio.h" +#include "utils/math.h" namespace cura { @@ -75,7 +77,7 @@ class PolygonConnector /*! * Add polygons to be connected by a future call to \ref PolygonConnector::connect() */ - void add(const Polygons& input); + void add(const Shape& input); /*! * Add variable-width paths to be connected by a future call to @@ -98,7 +100,7 @@ class PolygonConnector * \param output_paths Paths that were connected as much as possible. These * are expected to be empty to start with. */ - void connect(Polygons& output_polygons, std::vector& output_paths); + void connect(Shape& output_polygons, std::vector& output_paths); protected: coord_t line_width_; //!< The distance between the line segments which connect two polygons. diff --git a/include/utils/PolygonsPointIndex.h b/include/utils/PolygonsPointIndex.h index c4ce020105..a9e94709f5 100644 --- a/include/utils/PolygonsPointIndex.h +++ b/include/utils/PolygonsPointIndex.h @@ -6,8 +6,9 @@ #include -#include "Point2LL.h" -#include "polygon.h" +#include "geometry/Point2LL.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" namespace cura @@ -79,7 +80,7 @@ class PathsPointIndex /*! * Get the polygon to which this PolygonsPointIndex refers */ - ConstPolygonRef getPolygon() const; + const Polygon& getPolygon() const; /*! * Test whether two iterators refer to the same polygon in the same polygon list. @@ -145,7 +146,7 @@ class PathsPointIndex } }; -using PolygonsPointIndex = PathsPointIndex; +using PolygonsPointIndex = PathsPointIndex; /*! @@ -153,14 +154,7 @@ using PolygonsPointIndex = PathsPointIndex; */ struct PolygonsPointIndexSegmentLocator { - std::pair operator()(const PolygonsPointIndex& val) const - { - ConstPolygonRef poly = (*val.polygons_)[val.poly_idx_]; - Point2LL start = poly[val.point_idx_]; - size_t next_point_idx = (val.point_idx_ + 1ul) % poly.size(); - Point2LL end = poly[next_point_idx]; - return std::pair(start, end); - } + std::pair operator()(const PolygonsPointIndex& val) const; }; @@ -176,7 +170,7 @@ struct PathsPointIndexLocator } }; -using PolygonsPointIndexLocator = PathsPointIndexLocator; +using PolygonsPointIndexLocator = PathsPointIndexLocator; } // namespace cura diff --git a/include/utils/PolygonsSegmentIndex.h b/include/utils/PolygonsSegmentIndex.h index ab37fa2f06..c0c5d395c9 100644 --- a/include/utils/PolygonsSegmentIndex.h +++ b/include/utils/PolygonsSegmentIndex.h @@ -18,7 +18,7 @@ class PolygonsSegmentIndex : public PolygonsPointIndex { public: PolygonsSegmentIndex(); - PolygonsSegmentIndex(const Polygons* polygons, unsigned int poly_idx, unsigned int point_idx); + PolygonsSegmentIndex(const Shape* polygons, unsigned int poly_idx, unsigned int point_idx); Point2LL from() const; diff --git a/include/utils/PolylineStitcher.h b/include/utils/PolylineStitcher.h index 7d608f1ee7..cddfb0effa 100644 --- a/include/utils/PolylineStitcher.h +++ b/include/utils/PolylineStitcher.h @@ -5,20 +5,19 @@ #define UTILS_POLYLINE_STITCHER_H #include -#include -#include "PolygonsPointIndex.h" #include "SparsePointGrid.h" -#include "SymmetricPair.h" -#include "polygon.h" namespace cura { +template +class PathsPointIndex; + /*! * Class for stitching polylines into longer polylines or into polygons */ -template +template class PolylineStitcher { public: @@ -52,191 +51,12 @@ class PolylineStitcher * \param snap_distance Points closer than this distance are considered to * be the same point. */ - static void stitch(const Paths& lines, Paths& result_lines, Paths& result_polygons, coord_t max_stitch_distance = MM2INT(0.1), coord_t snap_distance = 10) - { - if (lines.empty()) - { - return; - } - - SparsePointGrid, PathsPointIndexLocator> grid(max_stitch_distance, lines.size() * 2); - - // populate grid - for (size_t line_idx = 0; line_idx < lines.size(); line_idx++) - { - const auto line = lines[line_idx]; - grid.insert(PathsPointIndex(&lines, line_idx, 0)); - grid.insert(PathsPointIndex(&lines, line_idx, line.size() - 1)); - } - - std::vector processed(lines.size(), false); - - for (size_t line_idx = 0; line_idx < lines.size(); line_idx++) - { - if (processed[line_idx]) - { - continue; - } - processed[line_idx] = true; - const auto line = lines[line_idx]; - bool should_close = isOdd(line); - - Path chain = line; - bool closest_is_closing_polygon = false; - for (bool go_in_reverse_direction : { false, true }) // first go in the unreversed direction, to try to prevent the chain.reverse() operation. - { // NOTE: Implementation only works for this order; we currently only re-reverse the chain when it's closed. - if (go_in_reverse_direction) - { // try extending chain in the other direction - chain.reverse(); - } - coord_t chain_length = chain.polylineLength(); - - while (true) - { - Point2LL from = make_point(chain.back()); - - PathsPointIndex closest; - coord_t closest_distance = std::numeric_limits::max(); - grid.processNearby( - from, - max_stitch_distance, - std::function&)>( - [from, - &chain, - &closest, - &closest_is_closing_polygon, - &closest_distance, - &processed, - &chain_length, - go_in_reverse_direction, - max_stitch_distance, - snap_distance, - should_close](const PathsPointIndex& nearby) -> bool - { - bool is_closing_segment = false; - coord_t dist = vSize(nearby.p() - from); - if (dist > max_stitch_distance) - { - return true; // keep looking - } - if (vSize2(nearby.p() - make_point(chain.front())) < snap_distance * snap_distance) - { - if (chain_length + dist < 3 * max_stitch_distance // prevent closing of small poly, cause it might be able to continue making a larger polyline - || chain.size() <= 2) // don't make 2 vert polygons - { - return true; // look for a better next line - } - is_closing_segment = true; - if (! should_close) - { - dist += 10; // prefer continuing polyline over closing a polygon; avoids closed zigzags from being printed separately - // continue to see if closing segment is also the closest - // there might be a segment smaller than [max_stitch_distance] which closes the polygon better - } - else - { - dist -= 10; // Prefer closing the polygon if it's 100% even lines. Used to create closed contours. - // Continue to see if closing segment is also the closest. - } - } - else if (processed[nearby.poly_idx_]) - { // it was already moved to output - return true; // keep looking for a connection - } - bool nearby_would_be_reversed = nearby.point_idx_ != 0; - nearby_would_be_reversed - = nearby_would_be_reversed != go_in_reverse_direction; // flip nearby_would_be_reversed when searching in the reverse direction - if (! canReverse(nearby) && nearby_would_be_reversed) - { // connecting the segment would reverse the polygon direction - return true; // keep looking for a connection - } - if (! canConnect(chain, (*nearby.polygons_)[nearby.poly_idx_])) - { - return true; // keep looking for a connection - } - if (dist < closest_distance) - { - closest_distance = dist; - closest = nearby; - closest_is_closing_polygon = is_closing_segment; - } - if (dist < snap_distance) - { // we have found a good enough next line - return false; // stop looking for alternatives - } - return true; // keep processing elements - })); - - if (! closest.initialized() // we couldn't find any next line - || closest_is_closing_polygon // we closed the polygon - ) - { - break; - } - - - coord_t segment_dist = vSize(make_point(chain.back()) - closest.p()); - assert(segment_dist <= max_stitch_distance + 10); - const size_t old_size = chain.size(); - if (closest.point_idx_ == 0) - { - auto start_pos = (*closest.polygons_)[closest.poly_idx_].begin(); - if (segment_dist < snap_distance) - { - ++start_pos; - } - chain.insert(chain.end(), start_pos, (*closest.polygons_)[closest.poly_idx_].end()); - } - else - { - auto start_pos = (*closest.polygons_)[closest.poly_idx_].rbegin(); - if (segment_dist < snap_distance) - { - ++start_pos; - } - chain.insert(chain.end(), start_pos, (*closest.polygons_)[closest.poly_idx_].rend()); - } - for (size_t i = old_size; i < chain.size(); ++i) // Update chain length. - { - chain_length += vSize(chain[i] - chain[i - 1]); - } - should_close = should_close & ! isOdd((*closest.polygons_)[closest.poly_idx_]); // If we connect an even to an odd line, we should no longer try to close it. - assert(! processed[closest.poly_idx_]); - processed[closest.poly_idx_] = true; - } - - if (closest_is_closing_polygon) - { - if (go_in_reverse_direction) - { // re-reverse chain to retain original direction - // NOTE: not sure if this code could ever be reached, since if a polygon can be closed that should be already possible in the forward direction - chain.reverse(); - } - - break; // don't consider reverse direction - } - } - if (closest_is_closing_polygon) - { - result_polygons.emplace_back(chain); - } - else - { - PathsPointIndex ppi_here(&lines, line_idx, 0); - if (! canReverse(ppi_here)) - { // Since closest_is_closing_polygon is false we went through the second iterations of the for-loop, where go_in_reverse_direction is true - // the polyline isn't allowed to be reversed, so we re-reverse it. - chain.reverse(); - } - result_lines.emplace_back(chain); - } - } - } + static void stitch(const InputPaths& lines, InputPaths& result_lines, OutputPaths& result_polygons, coord_t max_stitch_distance = MM2INT(0.1), coord_t snap_distance = 10); /*! * Whether a polyline is allowed to be reversed. (Not true for wall polylines which are not odd) */ - static bool canReverse(const PathsPointIndex& polyline); + static bool canReverse(const PathsPointIndex& polyline); /*! * Whether two paths are allowed to be connected. @@ -245,6 +65,9 @@ class PolylineStitcher static bool canConnect(const Path& a, const Path& b); static bool isOdd(const Path& line); + +private: + static void pushToClosedResult(OutputPaths& result_polygons, const Path& polyline); }; } // namespace cura diff --git a/include/utils/SVG.h b/include/utils/SVG.h index a67b1d5944..96bcfa0a45 100644 --- a/include/utils/SVG.h +++ b/include/utils/SVG.h @@ -12,7 +12,7 @@ #include "AABB.h" #include "ExtrusionLine.h" //To accept variable-width paths. #include "NoCopy.h" -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { @@ -99,15 +99,15 @@ class SVG : NoCopy void writeComment(const std::string& comment) const; - void writeAreas(const Polygons& polygons, const ColorObject color = Color::GRAY, const ColorObject outline_color = Color::BLACK, const double stroke_width = 1.0) const; + void writeAreas(const Shape& polygons, const ColorObject color = Color::GRAY, const ColorObject outline_color = Color::BLACK, const double stroke_width = 1.0) const; - void writeAreas(ConstPolygonRef polygon, const ColorObject color = Color::GRAY, const ColorObject outline_color = Color::BLACK, const double stroke_width = 1.0) const; + void writeAreas(const Polygon& polygon, const ColorObject color = Color::GRAY, const ColorObject outline_color = Color::BLACK, const double stroke_width = 1.0) const; void writePoint(const Point2LL& p, const bool write_coords = false, const double size = 5.0, const ColorObject color = Color::BLACK) const; - void writePoints(ConstPolygonRef poly, const bool write_coords = false, const double size = 5.0, const ColorObject color = Color::BLACK) const; + void writePoints(const Polygon& poly, const bool write_coords = false, const double size = 5.0, const ColorObject color = Color::BLACK) const; - void writePoints(const Polygons& polygons, const bool write_coords = false, const double size = 5.0, const ColorObject color = Color::BLACK) const; + void writePoints(const Shape& polygons, const bool write_coords = false, const double size = 5.0, const ColorObject color = Color::BLACK) const; /*! * \brief Draws a polyline on the canvas. @@ -144,13 +144,13 @@ class SVG : NoCopy void writeText(const Point2LL& p, const std::string& txt, const ColorObject color = Color::BLACK, const double font_size = 10.0) const; - void writePolygons(const Polygons& polys, const ColorObject color = Color::BLACK, const double stroke_width = 1.0, const bool flush = true) const; + void writePolygons(const Shape& polys, const ColorObject color = Color::BLACK, const double stroke_width = 1.0, const bool flush = true) const; - void writePolygon(ConstPolygonRef poly, const ColorObject color = Color::BLACK, const double stroke_width = 1.0, const bool flush = true) const; + void writePolygon(Polygon poly, const ColorObject color = Color::BLACK, const double stroke_width = 1.0, const bool flush = true) const; - void writePolylines(const Polygons& polys, const ColorObject color = Color::BLACK, const double stroke_width = 1.0) const; + void writePolylines(const Shape& polys, const ColorObject color = Color::BLACK, const double stroke_width = 1.0) const; - void writePolyline(ConstPolygonRef poly, const ColorObject color = Color::BLACK, const double stroke_width = 1.0) const; + void writePolyline(const Polygon& poly, const ColorObject color = Color::BLACK, const double stroke_width = 1.0) const; /*! * Draw variable-width paths into the image. diff --git a/include/utils/Simplify.h b/include/utils/Simplify.h index 3ca5b0009d..d06c018568 100644 --- a/include/utils/Simplify.h +++ b/include/utils/Simplify.h @@ -4,14 +4,25 @@ #ifndef UTILS_SIMPLIFY_H #define UTILS_SIMPLIFY_H -#include "../settings/Settings.h" //To load the parameters from a Settings object. -#include "ExtrusionLine.h" -#include "linearAlg2D.h" //To calculate line deviations and intersecting lines. -#include "polygon.h" +#include "geometry/Point2LL.h" +#include "utils/Coord_t.h" + namespace cura { +template +class LinesSet; +struct ExtrusionLine; +struct ExtrusionJunction; +class MixedLinesSet; +class Settings; +class Shape; +class Polygon; +class OpenPolyline; +class ClosedPolyline; +class Polyline; + /*! * Utility class to reduce the resolution of polygons and polylines, under * certain constraints. @@ -82,7 +93,7 @@ class Simplify * \param polygons The polygons to simplify. * \return The simplified polygons. */ - Polygons polygon(const Polygons& polygons) const; + Shape polygon(const Shape& polygons) const; /*! * Simplify a polygon. @@ -105,7 +116,17 @@ class Simplify * \param polylines The polylines to simplify. * \return The simplified polylines. */ - Polygons polyline(const Polygons& polylines) const; + template + LinesSet polyline(const LinesSet& polylines) const; + + /*! + * Simplify a batch of polylines. + * + * The endpoints of each polyline cannot be altered. + * \param polylines The polylines to simplify. + * \return The simplified polylines. + */ + MixedLinesSet polyline(const MixedLinesSet& polylines) const; /*! * Simplify a polyline. @@ -114,7 +135,15 @@ class Simplify * \param polyline The polyline to simplify. * \return The simplified polyline. */ - Polygon polyline(const Polygon& polyline) const; + OpenPolyline polyline(const OpenPolyline& polyline) const; + + /*! + * Simplify a polyline. + * + * \param polyline The polyline to simplify. + * \return The simplified polyline. + */ + ClosedPolyline polyline(const ClosedPolyline& polyline) const; /*! * Simplify a variable-line-width polyline. @@ -133,240 +162,6 @@ class Simplify */ constexpr static coord_t min_resolution = 5; // 5 units, regardless of how big those are, to allow for rounding errors. - template - bool detectSmall(const Polygonal& polygon, const coord_t& min_size) const - { - if (polygon.size() < min_size) // For polygon, 2 or fewer vertices is degenerate. Delete it. For polyline, 1 vertex is degenerate. - { - return true; - } - if (polygon.size() == min_size) - { - const auto a = getPosition(polygon[0]); - const auto b = getPosition(polygon[1]); - const auto c = getPosition(polygon[polygon.size() - 1]); - if (std::max(std::max(vSize2(b - a), vSize2(c - a)), vSize2(c - b)) < min_resolution * min_resolution) - { - // ... unless they are degenetate. - return true; - } - } - return false; - } - - /*! - * The main simplification algorithm starts here. - * \tparam Polygonal A polygonal object, which is a list of vertices. - * \param polygon The polygonal chain to simplify. - * \param is_closed Whether this is a closed polygon or an open polyline. - * \return A simplified polygonal chain. - */ - template - Polygonal simplify(const Polygonal& polygon, const bool is_closed) const - { - const size_t min_size = is_closed ? 3 : 2; - if (detectSmall(polygon, min_size)) - { - return createEmpty(polygon); - } - if (polygon.size() == min_size) // For polygon, don't reduce below 3. For polyline, not below 2. - { - return polygon; - } - - std::vector to_delete(polygon.size(), false); - auto comparator = [](const std::pair& vertex_a, const std::pair& vertex_b) - { - return vertex_a.second > vertex_b.second || (vertex_a.second == vertex_b.second && vertex_a.first > vertex_b.first); - }; - std::priority_queue, std::vector>, decltype(comparator)> by_importance(comparator); - - Polygonal result = polygon; // Make a copy so that we can also shift vertices. - for (int64_t current_removed = -1; (polygon.size() - current_removed) > min_size && current_removed != 0;) - { - current_removed = 0; - - // Add the initial points. - for (size_t i = 0; i < result.size(); ++i) - { - if (to_delete[i]) - { - continue; - } - const coord_t vertex_importance = importance(result, to_delete, i, is_closed); - by_importance.emplace(i, vertex_importance); - } - - // Iteratively remove the least important point until a threshold. - coord_t vertex_importance = 0; - while (! by_importance.empty() && (polygon.size() - current_removed) > min_size) - { - std::pair vertex = by_importance.top(); - by_importance.pop(); - // The importance may have changed since this vertex was inserted. Re-compute it now. - // If it doesn't change, it's safe to process. - vertex_importance = importance(result, to_delete, vertex.first, is_closed); - if (vertex_importance != vertex.second) - { - by_importance.emplace(vertex.first, vertex_importance); // Re-insert with updated importance. - continue; - } - - if (vertex_importance <= max_deviation_ * max_deviation_) - { - current_removed += remove(result, to_delete, vertex.first, vertex_importance, is_closed) ? 1 : 0; - } - } - } - - // Now remove the marked vertices in one sweep. - Polygonal filtered = createEmpty(polygon); - for (size_t i = 0; i < result.size(); ++i) - { - if (! to_delete[i]) - { - appendVertex(filtered, result[i]); - } - } - - if (detectSmall(filtered, min_size)) - { - return createEmpty(filtered); - } - return filtered; - } - - /*! - * A measure of the importance of a vertex. - * \tparam Polygonal A polygonal object, which is a list of vertices. - * \param polygon The polygon or polyline the vertex is part of. - * \param to_delete For each vertex, whether it is set to be deleted. - * \param index The vertex index to compute the importance of. - * \param is_closed Whether the polygon is closed (a polygon) or open - * (a polyline). - * \return A measure of how important the vertex is. Higher importance means - * that the vertex should probably be retained in the output. - */ - template - coord_t importance(const Polygonal& polygon, const std::vector& to_delete, const size_t index, const bool is_closed) const - { - const size_t poly_size = polygon.size(); - if (! is_closed && (index == 0 || index == poly_size - 1)) - { - return std::numeric_limits::max(); // Endpoints of the polyline must always be retained. - } - // From here on out we can safely look at the vertex neighbors and assume it's a polygon. We won't go out of bounds of the polyline. - - const Point2LL& vertex = getPosition(polygon[index]); - const size_t before_index = previousNotDeleted(index, to_delete); - const size_t after_index = nextNotDeleted(index, to_delete); - - const coord_t area_deviation = getAreaDeviation(polygon[before_index], polygon[index], polygon[after_index]); - if (area_deviation > max_area_deviation_) // Removing this line causes the variable line width to get flattened out too much. - { - return std::numeric_limits::max(); - } - - const Point2LL& before = getPosition(polygon[before_index]); - const Point2LL& after = getPosition(polygon[after_index]); - const coord_t deviation2 = LinearAlg2D::getDist2FromLine(vertex, before, after); - if (deviation2 <= min_resolution * min_resolution) // Deviation so small that it's always desired to remove them. - { - return deviation2; - } - if (vSize2(before - vertex) > max_resolution_ * max_resolution_ && vSize2(after - vertex) > max_resolution_ * max_resolution_) - { - return std::numeric_limits::max(); // Long line segments, no need to remove this one. - } - return deviation2; - } - - /*! - * Mark a vertex for removal. - * - * This function looks in the vertex and the four edges surrounding it to - * determine the best way to remove the given vertex. It may choose instead - * to delete an edge, fusing two vertices together. - * \tparam Polygonal A polygonal object, which is a list of vertices. - * \param polygon The polygon to remove a vertex from. - * \param to_delete The vertices that have been marked for deletion so far. - * This will be edited in-place. - * \param vertex The index of the vertex to remove. - * \param deviation2 The previously found deviation for this vertex. - * \param is_closed Whether we're working on a closed polygon or an open - \return Whether something is actually removed - * polyline. - */ - template - bool remove(Polygonal& polygon, std::vector& to_delete, const size_t vertex, const coord_t deviation2, const bool is_closed) const - { - if (deviation2 <= min_resolution * min_resolution) - { - // At less than the minimum resolution we're always allowed to delete the vertex. - // Even if the adjacent line segments are very long. - to_delete[vertex] = true; - return true; - } - - const size_t before = previousNotDeleted(vertex, to_delete); - const size_t after = nextNotDeleted(vertex, to_delete); - const Point2LL& vertex_position = getPosition(polygon[vertex]); - const Point2LL& before_position = getPosition(polygon[before]); - const Point2LL& after_position = getPosition(polygon[after]); - const coord_t length2_before = vSize2(vertex_position - before_position); - const coord_t length2_after = vSize2(vertex_position - after_position); - - if (length2_before <= max_resolution_ * max_resolution_ && length2_after <= max_resolution_ * max_resolution_) // Both adjacent line segments are short. - { - // Removing this vertex does little harm. No long lines will be shifted. - to_delete[vertex] = true; - return true; - } - - // Otherwise, one edge next to this vertex is longer than max_resolution. The other is shorter. - // In this case we want to remove the short edge by replacing it with a vertex where the two surrounding edges intersect. - // Find the two line segments surrounding the short edge here ("before" and "after" edges). - Point2LL before_from, before_to, after_from, after_to; - if (length2_before <= length2_after) // Before is the shorter line. - { - if (! is_closed && before == 0) // No edge before the short edge. - { - return false; // Edge cannot be deleted without shifting a long edge. Don't remove anything. - } - const size_t before_before = previousNotDeleted(before, to_delete); - before_from = getPosition(polygon[before_before]); - before_to = getPosition(polygon[before]); - after_from = getPosition(polygon[vertex]); - after_to = getPosition(polygon[after]); - } - else - { - if (! is_closed && after == polygon.size() - 1) // No edge after the short edge. - { - return false; // Edge cannot be deleted without shifting a long edge. Don't remove anything. - } - const size_t after_after = nextNotDeleted(after, to_delete); - before_from = getPosition(polygon[before]); - before_to = getPosition(polygon[vertex]); - after_from = getPosition(polygon[after]); - after_to = getPosition(polygon[after_after]); - } - Point2LL intersection; - const bool did_intersect = LinearAlg2D::lineLineIntersection(before_from, before_to, after_from, after_to, intersection); - if (! did_intersect) // Lines are parallel. - { - return false; // Cannot remove edge without shifting a long edge. Don't remove anything. - } - const coord_t intersection_deviation = LinearAlg2D::getDist2FromLineSegment(before_to, intersection, after_from); - if (intersection_deviation <= max_deviation_ * max_deviation_) // Intersection point doesn't deviate too much. Use it! - { - to_delete[vertex] = true; - polygon[length2_before <= length2_after ? before : after] = createIntersection(polygon[before], intersection, polygon[after]); - return true; - } - return false; - } - /*! * Helper method to find the index of the next vertex that is not about to * get deleted. @@ -393,22 +188,6 @@ class Simplify */ size_t previousNotDeleted(size_t index, const std::vector& to_delete) const; - /*! - * Create an empty polygon with the same properties as an original polygon, - * but without the vertex data. - * \param original The polygon to copy the properties from. - * \return An empty polygon. - */ - Polygon createEmpty(const Polygon& original) const; - - /*! - * Create an empty extrusion line with the same properties as an original - * extrusion line, but without the vertex data. - * \param original The extrusion line to copy the properties from. - * \return An empty extrusion line. - */ - ExtrusionLine createEmpty(const ExtrusionLine& original) const; - /*! * Append a vertex to this polygon. * @@ -416,7 +195,7 @@ class Simplify * \param polygon The polygon to add to. * \param vertex The vertex to add. */ - void appendVertex(Polygon& polygon, const Point2LL& vertex) const; + void appendVertex(Polyline& polygon, const Point2LL& vertex) const; /*! * Append a vertex to this extrusion line. @@ -488,6 +267,62 @@ class Simplify * \return The area deviation that would be caused by removing the vertex. */ coord_t getAreaDeviation(const ExtrusionJunction& before, const ExtrusionJunction& vertex, const ExtrusionJunction& after) const; + +private: + /*! + * Create an empty polygonal with the same properties as an original polygon, + * but without the vertex data. + * \param original The polygonal to copy the properties from. + * \return An empty polygonal. + */ + template + static Polygonal createEmpty(const Polygonal& original); + + template + bool detectSmall(const Polygonal& polygon, const coord_t& min_size) const; + + /*! + * The main simplification algorithm starts here. + * \tparam Polygonal A polygonal object, which is a list of vertices. + * \param polygon The polygonal chain to simplify. + * \param is_closed Whether this is a closed polygon or an open polyline. + * \return A simplified polygonal chain. + */ + template + Polygonal simplify(const Polygonal& polygon, const bool is_closed) const; + + /*! + * A measure of the importance of a vertex. + * \tparam Polygonal A polygonal object, which is a list of vertices. + * \param polygon The polygon or polyline the vertex is part of. + * \param to_delete For each vertex, whether it is set to be deleted. + * \param index The vertex index to compute the importance of. + * \param is_closed Whether the polygon is closed (a polygon) or open + * (a polyline). + * \return A measure of how important the vertex is. Higher importance means + * that the vertex should probably be retained in the output. + */ + template + coord_t importance(const Polygonal& polygon, const std::vector& to_delete, const size_t index, const bool is_closed) const; + + /*! + * Mark a vertex for removal. + * + * This function looks in the vertex and the four edges surrounding it to + * determine the best way to remove the given vertex. It may choose instead + * to delete an edge, fusing two vertices together. + * \tparam Polygonal A polygonal object, which is a list of vertices. + * \param polygon The polygon to remove a vertex from. + * \param to_delete The vertices that have been marked for deletion so far. + * This will be edited in-place. + * \param vertex The index of the vertex to remove. + * \param deviation2 The previously found deviation for this vertex. + * \param is_closed Whether we're working on a closed polygon or an open + \return Whether something is actually removed + * polyline. + */ + template + bool remove(Polygonal& polygon, std::vector& to_delete, const size_t vertex, const coord_t deviation2, const bool is_closed) const; }; } // namespace cura diff --git a/include/utils/SparseGrid.h b/include/utils/SparseGrid.h index 11b0708942..4705d69a60 100644 --- a/include/utils/SparseGrid.h +++ b/include/utils/SparseGrid.h @@ -10,8 +10,8 @@ #include #include -#include "Point2LL.h" #include "SquareGrid.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/SparseLineGrid.h b/include/utils/SparseLineGrid.h index 5b22818797..197767d30b 100644 --- a/include/utils/SparseLineGrid.h +++ b/include/utils/SparseLineGrid.h @@ -10,9 +10,9 @@ #include #include -#include "Point2LL.h" #include "SVG.h" // debug #include "SparseGrid.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/SparsePointGrid.h b/include/utils/SparsePointGrid.h index 4565a5dc4a..cb90667095 100644 --- a/include/utils/SparsePointGrid.h +++ b/include/utils/SparsePointGrid.h @@ -9,8 +9,8 @@ #include #include -#include "Point2LL.h" #include "SparseGrid.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/SparsePointGridInclusive.h b/include/utils/SparsePointGridInclusive.h index 962c0d602e..f5f30dc65d 100644 --- a/include/utils/SparsePointGridInclusive.h +++ b/include/utils/SparsePointGridInclusive.h @@ -9,8 +9,8 @@ #include #include -#include "Point2LL.h" #include "SparsePointGrid.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/SquareGrid.h b/include/utils/SquareGrid.h index ebbab19ca2..90665d5e81 100644 --- a/include/utils/SquareGrid.h +++ b/include/utils/SquareGrid.h @@ -9,7 +9,7 @@ #include #include -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/ToolpathVisualizer.h b/include/utils/ToolpathVisualizer.h index d3ad54ac7a..57f808b702 100644 --- a/include/utils/ToolpathVisualizer.h +++ b/include/utils/ToolpathVisualizer.h @@ -4,7 +4,7 @@ #include "ExtrusionSegment.h" #include "SVG.h" -#include "polygon.h" +#include "geometry/Polygon.h" namespace cura { @@ -21,11 +21,11 @@ class ToolpathVisualizer { } - void outline(const Polygons& input); + void outline(const Shape& input); void toolpaths(const std::vector& all_segments, bool rounded_visualization = true); - void underfill(const Polygons& underfills); - void overfill(const Polygons& overfills, const Polygons& double_overfills = Polygons()); - void width_legend(const Polygons& input, coord_t nozzle_size, coord_t max_dev, coord_t min_w, bool rounded_visualization); + void underfill(const Shape& underfills); + void overfill(const Shape& overfills, const Shape& double_overfills = Shape()); + void width_legend(const Shape& input, coord_t nozzle_size, coord_t max_dev, coord_t min_w, bool rounded_visualization); void widths(const std::vector& all_segments, coord_t nozzle_size, coord_t max_dev, coord_t min_w, bool rounded_visualization, bool exaggerate_widths = false); private: diff --git a/include/utils/VoxelUtils.h b/include/utils/VoxelUtils.h index f6cd123677..9eaca6e447 100644 --- a/include/utils/VoxelUtils.h +++ b/include/utils/VoxelUtils.h @@ -7,8 +7,8 @@ #include #include -#include "utils/Point2LL.h" -#include "utils/polygon.h" +#include "geometry/Point2LL.h" +#include "geometry/Polygon.h" namespace cura { @@ -92,7 +92,7 @@ class VoxelUtils * \param process_cell_func Function to perform on each voxel cell * \return Whether executing was stopped short as indicated by the \p cell_processing_function */ - bool walkPolygons(const Polygons& polys, coord_t z, const std::function& process_cell_func) const; + bool walkPolygons(const Shape& polys, coord_t z, const std::function& process_cell_func) const; /*! * Process voxels near the line segments of a polygon. @@ -105,13 +105,13 @@ class VoxelUtils * \param process_cell_func Function to perform on each voxel cell * \return Whether executing was stopped short as indicated by the \p cell_processing_function */ - bool walkDilatedPolygons(const Polygons& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const; + bool walkDilatedPolygons(const Shape& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const; private: /*! * \warning the \p polys is assumed to be translated by half the cell_size in xy already */ - bool _walkAreas(const Polygons& polys, coord_t z, const std::function& process_cell_func) const; + bool _walkAreas(const Shape& polys, coord_t z, const std::function& process_cell_func) const; public: /*! @@ -124,7 +124,7 @@ class VoxelUtils * \param process_cell_func Function to perform on each voxel cell * \return Whether executing was stopped short as indicated by the \p cell_processing_function */ - bool walkAreas(const Polygons& polys, coord_t z, const std::function& process_cell_func) const; + bool walkAreas(const Shape& polys, coord_t z, const std::function& process_cell_func) const; /*! * Process all voxels inside the area of a polygons object. @@ -137,7 +137,7 @@ class VoxelUtils * \param process_cell_func Function to perform on each voxel cell * \return Whether executing was stopped short as indicated by the \p cell_processing_function */ - bool walkDilatedAreas(const Polygons& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const; + bool walkDilatedAreas(const Shape& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const; /*! * Dilate with a kernel. diff --git a/include/utils/actions/smooth.h b/include/utils/actions/smooth.h index a414d9850a..338c47e391 100644 --- a/include/utils/actions/smooth.h +++ b/include/utils/actions/smooth.h @@ -1,4 +1,4 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #ifndef UTILS_VIEWS_SMOOTH_H @@ -6,8 +6,6 @@ #include #include -#include -#include #include #include @@ -20,8 +18,10 @@ #include #include +#include "geometry/Point2LL.h" #include "settings/Settings.h" #include "settings/types/Angle.h" +#include "utils/Coord_t.h" #include "utils/types/arachne.h" #include "utils/types/generic.h" #include "utils/types/geometry.h" diff --git a/include/utils/linearAlg2D.h b/include/utils/linearAlg2D.h index 5c7867e4d6..b9f08fb45b 100644 --- a/include/utils/linearAlg2D.h +++ b/include/utils/linearAlg2D.h @@ -4,10 +4,13 @@ #ifndef UTILS_LINEAR_ALG_2D_H #define UTILS_LINEAR_ALG_2D_H -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { + +class Point3Matrix; + class LinearAlg2D { public: @@ -64,37 +67,7 @@ class LinearAlg2D return -1; } - static bool lineLineIntersection(const Point2LL& a, const Point2LL& b, const Point2LL& c, const Point2LL& d, Point2LL& output) - { - // Adapted from Apex: https://github.com/Ghostkeeper/Apex/blob/eb75f0d96e36c7193d1670112826842d176d5214/include/apex/line_segment.hpp#L91 - // Adjusted to work with lines instead of line segments. - const Point2LL l1_delta = b - a; - const Point2LL l2_delta = d - c; - const coord_t divisor = cross(l1_delta, l2_delta); // Pre-compute divisor needed for the intersection check. - if (divisor == 0) - { - // The lines are parallel if the cross product of their directions is zero. - return false; - } - - // Create a parametric representation of each line. - // We'll equate the parametric equations to each other to find the intersection then. - // Parametric equation is L = P + Vt (where P and V are a starting point and directional vector). - // We'll map the starting point of one line onto the parameter system of the other line. - // Then using the divisor we can see whether and where they cross. - const Point2LL starts_delta = a - c; - const coord_t l1_parametric = cross(l2_delta, starts_delta); - Point2LL result = a + Point2LL(round_divide_signed(l1_parametric * l1_delta.X, divisor), round_divide_signed(l1_parametric * l1_delta.Y, divisor)); - - if (std::abs(result.X) > std::numeric_limits::max() || std::abs(result.Y) > std::numeric_limits::max()) - { - // Intersection is so far away that it could lead to integer overflows. - // Even though the lines aren't 100% parallel, it's better to pretend they are. They are practically parallel. - return false; - } - output = result; - return true; - } + static bool lineLineIntersection(const Point2LL& a, const Point2LL& b, const Point2LL& c, const Point2LL& d, Point2LL& output); /*! * Find whether a point projected on a line segment would be projected to @@ -403,12 +376,7 @@ class LinearAlg2D /*! * Get the rotation matrix for rotating around a specific point in place. */ - static Point3Matrix rotateAround(const Point2LL& middle, double rotation) - { - PointMatrix rotation_matrix(rotation); - Point3Matrix rotation_matrix_homogeneous(rotation_matrix); - return Point3Matrix::translate(middle).compose(rotation_matrix_homogeneous).compose(Point3Matrix::translate(-middle)); - } + static Point3Matrix rotateAround(const Point2LL& middle, double rotation); /*! * Test whether a point is inside a corner. diff --git a/include/utils/math.h b/include/utils/math.h index 23561cc86f..bb911f67b7 100644 --- a/include/utils/math.h +++ b/include/utils/math.h @@ -6,42 +6,44 @@ #include #include -#include + +#include "utils/types/generic.h" namespace cura { -template -inline T square(const T& a) +template +[[nodiscard]] T square(const T& a) { return a * a; } -inline int64_t round_divide_signed(const int64_t dividend, const int64_t divisor) //!< Return dividend divided by divisor rounded to the nearest integer +[[nodiscard]] inline int64_t round_divide_signed(const int64_t dividend, const int64_t divisor) //!< Return dividend divided by divisor rounded to the nearest integer { if ((dividend < 0) ^ (divisor < 0)) // Either the numerator or the denominator is negative, so the result must be negative. { return (dividend - divisor / 2) / divisor; // Flip the .5 offset to do proper rounding in the negatives too. } - else - { - return (dividend + divisor / 2) / divisor; - } + return (dividend + divisor / 2) / divisor; } -inline uint64_t ceil_divide_signed(const int64_t dividend, const int64_t divisor) //!< Return dividend divided by divisor rounded up towards positive infinity. + +[[nodiscard]] inline uint64_t ceil_divide_signed(const int64_t dividend, const int64_t divisor) //!< Return dividend divided by divisor rounded up towards positive infinity. { return static_cast((dividend / divisor) + (dividend * divisor > 0 ? 1 : 0)); } -inline uint64_t floor_divide_signed(const int64_t dividend, const int64_t divisor) //!< Return dividend divided by divisor rounded down towards negative infinity. + +[[nodiscard]] inline uint64_t floor_divide_signed(const int64_t dividend, const int64_t divisor) //!< Return dividend divided by divisor rounded down towards negative infinity. { return static_cast((dividend / divisor) + (dividend * divisor > 0 ? 0 : -1)); } -inline uint64_t round_divide(const uint64_t dividend, const uint64_t divisor) //!< Return dividend divided by divisor rounded to the nearest integer + +[[nodiscard]] inline uint64_t round_divide(const uint64_t dividend, const uint64_t divisor) //!< Return dividend divided by divisor rounded to the nearest integer { return (dividend + divisor / 2) / divisor; } -inline uint64_t round_up_divide(const uint64_t dividend, const uint64_t divisor) //!< Return dividend divided by divisor rounded to the nearest integer + +[[nodiscard]] inline uint64_t round_up_divide(const uint64_t dividend, const uint64_t divisor) //!< Return dividend divided by divisor rounded to the nearest integer { return (dividend + divisor - 1) / divisor; } diff --git a/include/utils/orderOptimizer.h b/include/utils/orderOptimizer.h index cf3abaa2ba..e6e3eb0c39 100644 --- a/include/utils/orderOptimizer.h +++ b/include/utils/orderOptimizer.h @@ -9,7 +9,7 @@ #include // pair #include -#include "Point2LL.h" +#include "geometry/Point2LL.h" namespace cura { diff --git a/include/utils/polygon.h b/include/utils/polygon.h deleted file mode 100644 index e9eb6c2fcb..0000000000 --- a/include/utils/polygon.h +++ /dev/null @@ -1,1605 +0,0 @@ -// Copyright (c) 2024 UltiMaker -// CuraEngine is released under the terms of the AGPLv3 or higher - -#ifndef UTILS_POLYGON_H -#define UTILS_POLYGON_H - -#include -#include // std::reverse, fill_n array -#include -#include // fabs -#include -#include // int64_t.min -#include -#include -#include -#include - -#include "../settings/types/Angle.h" //For angles between vertices. -#include "../settings/types/Ratio.h" -#include "Point2LL.h" - -#define CHECK_POLY_ACCESS -#ifdef CHECK_POLY_ACCESS -#define POLY_ASSERT(e) assert(e) -#else -#define POLY_ASSERT(e) \ - do \ - { \ - } while (0) -#endif - -namespace cura -{ - -template -bool shorterThan(const T& shape, const coord_t check_length) -{ - const auto* p0 = &shape.back(); - int64_t length = 0; - for (const auto& p1 : shape) - { - length += vSize(*p0 - p1); - if (length >= check_length) - { - return false; - } - p0 = &p1; - } - return true; -} - -class PartsView; -class Polygons; -class Polygon; -class PolygonRef; - -class ListPolyIt; - -typedef std::list ListPolygon; //!< A polygon represented by a linked list instead of a vector -typedef std::vector ListPolygons; //!< Polygons represented by a vector of linked lists instead of a vector of vectors - -const static int clipper_init = (0); -#define NO_INDEX (std::numeric_limits::max()) - -class ConstPolygonPointer; - -/*! - * Outer polygons should be counter-clockwise, - * inner hole polygons should be clockwise. - * (When negative X is to the left and negative Y is downward.) - */ -class ConstPolygonRef -{ - friend class Polygons; - friend class Polygon; - friend class PolygonRef; - friend class ConstPolygonPointer; - -protected: - ClipperLib::Path* path; - -public: - ConstPolygonRef(const ClipperLib::Path& polygon) - : path(const_cast(&polygon)) - { - } - - ConstPolygonRef() = delete; // you cannot have a reference without an object! - - virtual ~ConstPolygonRef() - { - } - - bool operator==(ConstPolygonRef& other) const = delete; // polygon comparison is expensive and probably not what you want when you use the equality operator - - ConstPolygonRef& operator=(const ConstPolygonRef& other) = delete; // Cannot assign to a const object - - /*! - * Gets the number of vertices in this polygon. - * \return The number of vertices in this polygon. - */ - size_t size() const; - - /*! - * Returns whether there are any vertices in this polygon. - * \return ``true`` if the polygon has no vertices at all, or ``false`` if - * it does have vertices. - */ - bool empty() const; - - const Point2LL& operator[](size_t index) const - { - POLY_ASSERT(index < size()); - return (*path)[index]; - } - - const ClipperLib::Path& operator*() const - { - return *path; - } - - ClipperLib::Path::const_iterator begin() const - { - return path->begin(); - } - - ClipperLib::Path::const_iterator end() const - { - return path->end(); - } - - ClipperLib::Path::const_reverse_iterator rbegin() const - { - return path->rbegin(); - } - - ClipperLib::Path::const_reverse_iterator rend() const - { - return path->rend(); - } - - ClipperLib::Path::const_reference front() const - { - return path->front(); - } - - ClipperLib::Path::const_reference back() const - { - return path->back(); - } - - const void* data() const - { - return path->data(); - } - - - /*! - * On Y-axis positive upward displays, Orientation will return true if the polygon's orientation is counter-clockwise. - * - * from http://www.angusj.com/delphi/clipper/documentation/Docs/Units/ClipperLib/Functions/Orientation.htm - */ - bool orientation() const - { - return ClipperLib::Orientation(*path); - } - - Polygons offset(int distance, ClipperLib::JoinType joinType = ClipperLib::jtMiter, double miter_limit = 1.2) const; - - coord_t polygonLength() const - { - return polylineLength() + vSize(path->front() - path->back()); - } - - coord_t polylineLength() const - { - coord_t length = 0; - Point2LL p0 = path->front(); - for (size_t n = 1; n < path->size(); n++) - { - Point2LL p1 = (*path)[n]; - length += vSize(p0 - p1); - p0 = p1; - } - return length; - } - - /*! - * Split these poly line objects into several line segment objects consisting of only two verts - * and store them in the \p result - */ - void splitPolylineIntoSegments(Polygons& result) const; - Polygons splitPolylineIntoSegments() const; - - /*! - * Split these polygon objects into several line segment objects consisting of only two verts - * and store them in the \p result - */ - void splitPolygonIntoSegments(Polygons& result) const; - Polygons splitPolygonIntoSegments() const; - - bool shorterThan(const coord_t check_length) const; - - Point2LL min() const - { - Point2LL ret = Point2LL(POINT_MAX, POINT_MAX); - for (Point2LL p : *path) - { - ret.X = std::min(ret.X, p.X); - ret.Y = std::min(ret.Y, p.Y); - } - return ret; - } - - Point2LL max() const - { - Point2LL ret = Point2LL(POINT_MIN, POINT_MIN); - for (Point2LL p : *path) - { - ret.X = std::max(ret.X, p.X); - ret.Y = std::max(ret.Y, p.Y); - } - return ret; - } - - double area() const - { - return ClipperLib::Area(*path); - } - - Point2LL centerOfMass() const - { - if (path->size() > 0) - { - Point2LL p0 = (*path)[0]; - if (path->size() > 1) - { - double x = 0, y = 0; - for (size_t n = 1; n <= path->size(); n++) - { - Point2LL p1 = (*path)[n % path->size()]; - double second_factor = static_cast((p0.X * p1.Y) - (p1.X * p0.Y)); - - x += double(p0.X + p1.X) * second_factor; - y += double(p0.Y + p1.Y) * second_factor; - p0 = p1; - } - - double area = Area(*path); - - x = x / 6 / area; - y = y / 6 / area; - - return Point2LL(std::llrint(x), std::llrint(y)); - } - else - { - return p0; - } - } - else - { - return Point2LL(); - } - } - - Point2LL closestPointTo(Point2LL p) const - { - Point2LL ret = p; - double bestDist = std::numeric_limits::max(); - for (size_t n = 0; n < path->size(); n++) - { - double dist = vSize2f(p - (*path)[n]); - if (dist < bestDist) - { - ret = (*path)[n]; - bestDist = dist; - } - } - return ret; - } - - /*! - * Check if we are inside the polygon. We do this by tracing from the point towards the positive X direction, - * every line we cross increments the crossings counter. If we have an even number of crossings then we are not inside the polygon. - * Care needs to be taken, if p.Y exactly matches a vertex to the right of p, then we need to count 1 intersect if the - * outline passes vertically past; and 0 (or 2) intersections if that point on the outline is a 'top' or 'bottom' vertex. - * The easiest way to do this is to break out two cases for increasing and decreasing Y ( from p0 to p1 ). - * A segment is tested if pa.Y <= p.Y < pb.Y, where pa and pb are the points (from p0,p1) with smallest & largest Y. - * When both have the same Y, no intersections are counted but there is a special test to see if the point falls - * exactly on the line. - * - * Returns false if outside, true if inside; if the point lies exactly on the border, will return 'border_result'. - * - * \deprecated This function is no longer used, since the Clipper function is used by the function PolygonRef::inside(.) - * - * \param p The point for which to check if it is inside this polygon - * \param border_result What to return when the point is exactly on the border - * \return Whether the point \p p is inside this polygon (or \p border_result when it is on the border) - */ - bool _inside(Point2LL p, bool border_result = false) const; - - /*! - * Clipper function. - * Returns false if outside, true if inside; if the point lies exactly on the border, will return 'border_result'. - * - * http://www.angusj.com/delphi/clipper/documentation/Docs/Units/ClipperLib/Functions/PointInPolygon.htm - */ - bool inside(Point2LL p, bool border_result = false) const - { - int res = ClipperLib::PointInPolygon(p, *path); - if (res == -1) - { - return border_result; - } - return res == 1; - } - - bool inside(const auto& polygon) const - { - for (const auto& point : *path) - { - if (! ClipperLib::PointInPolygon(point, *polygon.path)) - { - return false; - } - } - return true; - } - - /*! - * Smooth out small perpendicular segments and store the result in \p result. - * Smoothing is performed by removing the inner most vertex of a line segment smaller than \p remove_length - * which has an angle with the next and previous line segment smaller than roughly 150* - * - * Note that in its current implementation this function doesn't remove line segments with an angle smaller than 30* - * Such would be the case for an N shape. - * - * \param remove_length The length of the largest segment removed - * \param result (output) The result polygon, assumed to be empty - */ - void smooth(int remove_length, PolygonRef result) const; - - /*! - * Smooth out sharp inner corners, by taking a shortcut which bypasses the corner - * - * \param angle The maximum angle of inner corners to be smoothed out - * \param shortcut_length The desired length of the shortcut line segment introduced (shorter shortcuts may be unavoidable) - * \param result The resulting polygon - */ - void smooth_outward(const AngleDegrees angle, int shortcut_length, PolygonRef result) const; - - /*! - * Smooth out the polygon and store the result in \p result. - * Smoothing is performed by removing vertices for which both connected line segments are smaller than \p remove_length - * - * \param remove_length The length of the largest segment removed - * \param result (output) The result polygon, assumed to be empty - */ - void smooth2(int remove_length, PolygonRef result) const; - - /*! - * Compute the morphological intersection between this polygon and another. - * - * Note that the result may consist of multiple polygons, if you have bad - * luck. - * - * \param other The polygon with which to intersect this polygon. - */ - Polygons intersection(const ConstPolygonRef& other) const; - - -private: - /*! - * Smooth out a simple corner consisting of two linesegments. - * - * Auxiliary function for \ref smooth_outward - * - * \param p0 The point before the corner - * \param p1 The corner - * \param p2 The point after the corner - * \param p0_it Iterator to the point before the corner - * \param p1_it Iterator to the corner - * \param p2_it Iterator to the point after the corner - * \param v10 Vector from \p p1 to \p p0 - * \param v12 Vector from \p p1 to \p p2 - * \param v02 Vector from \p p0 to \p p2 - * \param shortcut_length The desired length ofthe shortcutting line - * \param cos_angle The cosine on the angle in L 012 - */ - static void smooth_corner_simple( - const Point2LL p0, - const Point2LL p1, - const Point2LL p2, - const ListPolyIt p0_it, - const ListPolyIt p1_it, - const ListPolyIt p2_it, - const Point2LL v10, - const Point2LL v12, - const Point2LL v02, - const int64_t shortcut_length, - double cos_angle); - - /*! - * Smooth out a complex corner where the shortcut bypasses more than two line segments - * - * Auxiliary function for \ref smooth_outward - * - * \warning This function might try to remove the whole polygon - * Error code -1 means the whole polygon should be removed (which means it is a hole polygon) - * - * \param p1 The corner point - * \param[in,out] p0_it Iterator to the last point checked before \p p1 to consider cutting off - * \param[in,out] p2_it Iterator to the last point checked after \p p1 to consider cutting off - * \param shortcut_length The desired length ofthe shortcutting line - * \return Whether this whole polygon whould be removed by the smoothing - */ - static bool smooth_corner_complex(const Point2LL p1, ListPolyIt& p0_it, ListPolyIt& p2_it, const int64_t shortcut_length); - - /*! - * Try to take a step away from the corner point in order to take a bigger shortcut. - * - * Try to take the shortcut from a place as far away from the corner as the place we are taking the shortcut to. - * - * Auxiliary function for \ref smooth_outward - * - * \param[in] p1 The corner point - * \param[in] shortcut_length2 The square of the desired length ofthe shortcutting line - * \param[in,out] p0_it Iterator to the previously checked point somewhere beyond \p p1. Updated for the next iteration. - * \param[in,out] p2_it Iterator to the previously checked point somewhere before \p p1. Updated for the next iteration. - * \param[in,out] forward_is_blocked Whether trying another step forward is blocked by the smoothing outward condition. Updated for the next iteration. - * \param[in,out] backward_is_blocked Whether trying another step backward is blocked by the smoothing outward condition. Updated for the next iteration. - * \param[in,out] forward_is_too_far Whether trying another step forward is blocked by the shortcut length condition. Updated for the next iteration. - * \param[in,out] backward_is_too_far Whether trying another step backward is blocked by the shortcut length condition. Updated for the next iteration. - */ - static void smooth_outward_step( - const Point2LL p1, - const int64_t shortcut_length2, - ListPolyIt& p0_it, - ListPolyIt& p2_it, - bool& forward_is_blocked, - bool& backward_is_blocked, - bool& forward_is_too_far, - bool& backward_is_too_far); -}; - - -class PolygonPointer; - -class PolygonRef : public ConstPolygonRef -{ - friend class PolygonPointer; - friend class Polygons; - friend class PolygonsPart; - -public: - PolygonRef(ClipperLib::Path& polygon) - : ConstPolygonRef(polygon) - { - } - - PolygonRef(const PolygonRef& other) - : ConstPolygonRef(*other.path) - { - } - - PolygonRef() = delete; // you cannot have a reference without an object! - - virtual ~PolygonRef() - { - } - - /*! - * Reserve a number of polygons to prevent reallocation and breakage of pointers. - * \param min_size The minimum size the new underlying array should have. - */ - void reserve(size_t min_size) - { - path->reserve(min_size); - } - - template - ClipperLib::Path::iterator insert(ClipperLib::Path::const_iterator pos, iterator first, iterator last) - { - return path->insert(pos, first, last); - } - - PolygonRef& operator=(const ConstPolygonRef& other) = delete; // polygon assignment is expensive and probably not what you want when you use the assignment operator - - PolygonRef& operator=(ConstPolygonRef& other) = delete; // polygon assignment is expensive and probably not what you want when you use the assignment operator - // { path = other.path; return *this; } - - PolygonRef& operator=(PolygonRef&& other) - { - *path = std::move(*other.path); - return *this; - } - - Point2LL& operator[](size_t index) - { - POLY_ASSERT(index < size()); - return (*path)[index]; - } - - const Point2LL& operator[](size_t index) const - { - return ConstPolygonRef::operator[](index); - } - - ClipperLib::Path::iterator begin() - { - return path->begin(); - } - - ClipperLib::Path::iterator end() - { - return path->end(); - } - - ClipperLib::Path::reference front() - { - return path->front(); - } - - ClipperLib::Path::reference back() - { - return path->back(); - } - - void* data() - { - return path->data(); - } - - void add(const Point2LL p) - { - path->push_back(p); - } - - ClipperLib::Path& operator*() - { - return *path; - } - - template - void emplace_back(Args&&... args) - { - path->emplace_back(args...); - } - - void remove(size_t index) - { - POLY_ASSERT(index < size() && index <= static_cast(std::numeric_limits::max())); - path->erase(path->begin() + static_cast(index)); - } - - void insert(size_t index, Point2LL p) - { - POLY_ASSERT(index < size() && index <= static_cast(std::numeric_limits::max())); - path->insert(path->begin() + static_cast(index), p); - } - - void clear() - { - path->clear(); - } - - void reverse() - { - ClipperLib::ReversePath(*path); - } - - /*! - * Translate the whole polygon in some direction. - * - * \param translation The direction in which to move the polygon - */ - void translate(Point2LL translation) - { - for (Point2LL& p : *this) - { - p += translation; - } - } - - void removeColinearEdges(const AngleRadians max_deviation_angle); - - /*! - * Removes consecutive line segments with same orientation and changes this polygon. - * - * 1. Removes verts which are connected to line segments which are too small. - * 2. Removes verts which detour from a direct line from the previous and next vert by a too small amount. - * 3. Moves a vert when a small line segment is connected to a much longer one. in order to maintain the outline of the object. - * 4. Don't remove a vert when the impact on the outline of the object is too great. - * - * Note that the simplify is a best effort algorithm. It does not guarantee that no lines below the provided smallest_line_segment_squared are left. - * - * The following example (Two very long line segments (" & , respectively) that are connected by a very small line segment (i) is unsimplifable by this - * function, even though the actual area change of removing line segment i is very small. The reason for this is that in the case of long lines, even a small - * deviation from it's original direction is very noticeable in the final result, especially if the polygons above make a slightly different choice. - * - * """"""""""""""""""""""""""""""""i,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,, - - * - * \param smallest_line_segment_squared maximal squared length of removed line segments - * \param allowed_error_distance_squared The square of the distance of the middle point to the line segment of the consecutive and previous point for which the middle point is - removed - */ - void simplify(const coord_t smallest_line_segment_squared = MM2INT(0.01) * MM2INT(0.01), const coord_t allowed_error_distance_squared = 25); - - /*! - * See simplify(.) - */ - void simplifyPolyline(const coord_t smallest_line_segment_squared = 100, const coord_t allowed_error_distance_squared = 25); - -protected: - /*! - * Private implementation for both simplify and simplifyPolygons. - * - * Made private to avoid accidental use of the wrong function. - */ - void _simplify(const coord_t smallest_line_segment_squared = 100, const coord_t allowed_error_distance_squared = 25, bool processing_polylines = false); - -public: - void pop_back() - { - path->pop_back(); - } - - /*! - * Apply a matrix to each vertex in this polygon - */ - void applyMatrix(const PointMatrix& matrix); - void applyMatrix(const Point3Matrix& matrix); -}; - -class ConstPolygonPointer -{ -protected: - const ClipperLib::Path* path; - -public: - ConstPolygonPointer() - : path(nullptr) - { - } - ConstPolygonPointer(const ConstPolygonRef* ref) - : path(ref->path) - { - } - ConstPolygonPointer(const ConstPolygonRef& ref) - : path(ref.path) - { - } - - ConstPolygonRef operator*() const - { - assert(path); - return ConstPolygonRef(*path); - } - const ClipperLib::Path* operator->() const - { - assert(path); - return path; - } - - operator bool() const - { - return path; - } - - bool operator==(const ConstPolygonPointer& rhs) const - { - return path == rhs.path; - } -}; - -class PolygonPointer : public ConstPolygonPointer -{ -public: - PolygonPointer() - : ConstPolygonPointer(nullptr) - { - } - PolygonPointer(PolygonRef* ref) - : ConstPolygonPointer(ref) - { - } - - PolygonPointer(PolygonRef& ref) - : ConstPolygonPointer(ref) - { - } - - PolygonRef operator*() - { - assert(path); - return PolygonRef(*const_cast(path)); - } - - ConstPolygonRef operator*() const - { - assert(path); - return ConstPolygonRef(*path); - } - - ClipperLib::Path* operator->() - { - assert(path); - return const_cast(path); - } - - const ClipperLib::Path* operator->() const - { - assert(path); - return path; - } - - operator bool() const - { - return path; - } -}; - -} // namespace cura - - -namespace std -{ -template<> -struct hash -{ - size_t operator()(const cura::ConstPolygonRef& poly) const - { - return std::hash()(&*poly); - } -}; -template<> -struct hash -{ - size_t operator()(const cura::ConstPolygonPointer& poly) const - { - return std::hash()(&**poly); - } -}; -template<> -struct hash -{ - size_t operator()(const cura::PolygonPointer& poly) const - { - const cura::ConstPolygonRef ref = *static_cast(poly); - return std::hash()(&*ref); - } -}; -} // namespace std - -namespace cura -{ - -class Polygon : public PolygonRef -{ -public: - ClipperLib::Path poly; - - Polygon() - : PolygonRef(poly) - { - } - - Polygon(const ConstPolygonRef& other) - : PolygonRef(poly) - , poly(*other.path) - { - } - - Polygon(const Polygon& other) - : PolygonRef(poly) - , poly(*other.path) - { - } - - Polygon(Polygon&& moved) - : PolygonRef(poly) - , poly(std::move(moved.poly)) - { - } - - virtual ~Polygon() - { - } - - Polygon& operator=(const ConstPolygonRef& other) = delete; // copying a single polygon is generally not what you want - // { - // path = other.path; - // poly = *other.path; - // return *this; - // } - - Polygon& operator=(Polygon&& other) //!< move assignment - { - poly = std::move(other.poly); - return *this; - } -}; - -class PolygonsPart; - -class Polygons -{ - friend class Polygon; - friend class PolygonRef; - friend class ConstPolygonRef; - friend class PolygonUtils; - -public: - ClipperLib::Paths paths; - - size_t size() const - { - return paths.size(); - } - - void reserve(size_t new_cap) - { - paths.reserve(new_cap); - } - - /*! - * Convenience function to check if the polygon has no points. - * - * \return `true` if the polygon has no points, or `false` if it does. - */ - bool empty() const; - - size_t pointCount() const; //!< Return the amount of points in all polygons - - PolygonRef operator[](size_t index) - { - POLY_ASSERT(index < size()); - return paths[index]; - } - ConstPolygonRef operator[](size_t index) const - { - POLY_ASSERT(index < size()); - return paths[index]; - } - ClipperLib::Paths::iterator begin() - { - return paths.begin(); - } - ClipperLib::Paths::const_iterator begin() const - { - return paths.begin(); - } - ClipperLib::Paths::iterator end() - { - return paths.end(); - } - ClipperLib::Paths::const_iterator end() const - { - return paths.end(); - } - /*! - * Remove a polygon from the list and move the last polygon to its place - * - * \warning changes the order of the polygons! - */ - void remove(size_t index) - { - POLY_ASSERT(index < size()); - if (index < paths.size() - 1) - { - paths[index] = std::move(paths.back()); - } - paths.resize(paths.size() - 1); - } - - void pop_back() - { - paths.pop_back(); - } - - /*! - * Remove a range of polygons - */ - void erase(ClipperLib::Paths::iterator start, ClipperLib::Paths::iterator end) - { - paths.erase(start, end); - } - void clear() - { - paths.clear(); - } - void add(ConstPolygonRef& poly) - { - paths.push_back(*poly.path); - } - void add(const ConstPolygonRef& poly) - { - paths.push_back(*poly.path); - } - void add(Polygon&& other_poly) - { - paths.emplace_back(std::move(*other_poly)); - } - void add(const Polygons& other) - { - std::copy(other.paths.begin(), other.paths.end(), std::back_inserter(paths)); - } - void addIfNotEmpty(ConstPolygonRef& poly) - { - if (! poly.empty()) - { - paths.push_back(*poly.path); - } - } - void addIfNotEmpty(const ConstPolygonRef& poly) - { - if (! poly.empty()) - { - paths.push_back(*poly.path); - } - } - void addIfNotEmpty(Polygon&& other_poly) - { - if (! other_poly.empty()) - { - paths.emplace_back(std::move(*other_poly)); - } - } - /*! - * Add a 'polygon' consisting of two points - */ - void addLine(const Point2LL from, const Point2LL to) - { - paths.emplace_back(ClipperLib::Path{ from, to }); - } - - void emplace_back(const Polygon& poly) - { - paths.emplace_back(*poly.path); - } - - void emplace_back(const ConstPolygonRef& poly) - { - paths.emplace_back(*poly.path); - } - - void emplace_back(const PolygonRef& poly) - { - paths.emplace_back(*poly.path); - } - - template - void emplace_back(Args... args) - { - paths.emplace_back(args...); - } - - PolygonRef newPoly() - { - paths.emplace_back(); - return PolygonRef(paths.back()); - } - PolygonRef front() - { - return PolygonRef(paths.front()); - } - ConstPolygonRef front() const - { - return ConstPolygonRef(paths.front()); - } - PolygonRef back() - { - return PolygonRef(paths.back()); - } - ConstPolygonRef back() const - { - return ConstPolygonRef(paths.back()); - } - - Polygons() - { - } - - Polygons(const Polygons& other) - { - paths = other.paths; - } - Polygons(Polygons&& other) - { - paths = std::move(other.paths); - } - Polygons& operator=(const Polygons& other) - { - paths = other.paths; - return *this; - } - Polygons& operator=(Polygons&& other) - { - if (this != &other) - { - paths = std::move(other.paths); - } - return *this; - } - - bool operator==(const Polygons& other) const = delete; - - /*! - * Convert ClipperLib::PolyTree to a Polygons object, - * which uses ClipperLib::Paths instead of ClipperLib::PolyTree - */ - static Polygons toPolygons(ClipperLib::PolyTree& poly_tree); - - Polygons difference(const Polygons& other) const - { - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - clipper.AddPaths(other.paths, ClipperLib::ptClip, true); - clipper.Execute(ClipperLib::ctDifference, ret.paths); - return ret; - } - Polygons unionPolygons(const Polygons& other, ClipperLib::PolyFillType fill_type = ClipperLib::pftNonZero) const - { - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - clipper.AddPaths(other.paths, ClipperLib::ptSubject, true); - clipper.Execute(ClipperLib::ctUnion, ret.paths, fill_type, fill_type); - return ret; - } - /*! - * Union all polygons with each other (When polygons.add(polygon) has been called for overlapping polygons) - */ - Polygons unionPolygons() const - { - return unionPolygons(Polygons()); - } - Polygons intersection(const Polygons& other) const - { - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - clipper.AddPaths(other.paths, ClipperLib::ptClip, true); - clipper.Execute(ClipperLib::ctIntersection, ret.paths); - return ret; - } - - - /*! - * Intersect polylines with this area Polygons object. - * - * \note Due to a clipper bug with polylines with nearly collinear segments, the polylines are cut up into separate polylines, and restitched back together at the end. - * - * \param polylines The (non-closed!) polylines to limit to the area of this Polygons object - * \param restitch Whether to stitch the resulting segments into longer polylines, or leave every segment as a single segment - * \param max_stitch_distance The maximum distance for two polylines to be stitched together with a segment - * \return The resulting polylines limited to the area of this Polygons object - */ - Polygons intersectionPolyLines(const Polygons& polylines, bool restitch = true, const coord_t max_stitch_distance = 10_mu) const; - - /*! - * Add the front to each polygon so that the polygon is represented as a polyline - */ - void toPolylines(); - - /*! - * Split this poly line object into several line segment objects - * and store them in the \p result - */ - void splitPolylinesIntoSegments(Polygons& result) const; - Polygons splitPolylinesIntoSegments() const; - - /*! - * Split this polygon object into several line segment objects - * and store them in the \p result - */ - void splitPolygonsIntoSegments(Polygons& result) const; - Polygons splitPolygonsIntoSegments() const; - - Polygons xorPolygons(const Polygons& other, ClipperLib::PolyFillType pft = ClipperLib::pftEvenOdd) const - { - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - clipper.AddPaths(other.paths, ClipperLib::ptClip, true); - clipper.Execute(ClipperLib::ctXor, ret.paths, pft); - return ret; - } - - Polygons execute(ClipperLib::PolyFillType pft = ClipperLib::pftEvenOdd) const - { - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - clipper.Execute(ClipperLib::ctXor, ret.paths, pft); - return ret; - } - - Polygons offset(coord_t distance, ClipperLib::JoinType joinType = ClipperLib::jtMiter, double miter_limit = 1.2) const; - - Polygons offsetPolyLine(int distance, ClipperLib::JoinType joinType = ClipperLib::jtMiter, bool inputPolyIsClosed = false) const - { - Polygons ret; - double miterLimit = 1.2; - ClipperLib::EndType end_type; - if (inputPolyIsClosed) - { - end_type = ClipperLib::etClosedLine; - } - else if (joinType == ClipperLib::jtMiter) - { - end_type = ClipperLib::etOpenSquare; - } - else - { - end_type = ClipperLib::etOpenRound; - } - ClipperLib::ClipperOffset clipper(miterLimit, 10.0); - clipper.AddPaths(paths, joinType, end_type); - clipper.MiterLimit = miterLimit; - clipper.Execute(ret.paths, distance); - return ret; - } - - /*! - * Check if we are inside the polygon. - * - * We do this by counting the number of polygons inside which this point lies. - * An odd number is inside, while an even number is outside. - * - * Returns false if outside, true if inside; if the point lies exactly on the border, will return \p border_result. - * - * \param p The point for which to check if it is inside this polygon - * \param border_result What to return when the point is exactly on the border - * \return Whether the point \p p is inside this polygon (or \p border_result when it is on the border) - */ - bool inside(Point2LL p, bool border_result = false) const; - - /*! - * Check if we are inside the polygon. We do this by tracing from the point towards the positive X direction, - * every line we cross increments the crossings counter. If we have an even number of crossings then we are not inside the polygon. - * Care needs to be taken, if p.Y exactly matches a vertex to the right of p, then we need to count 1 intersect if the - * outline passes vertically past; and 0 (or 2) intersections if that point on the outline is a 'top' or 'bottom' vertex. - * The easiest way to do this is to break out two cases for increasing and decreasing Y ( from p0 to p1 ). - * A segment is tested if pa.Y <= p.Y < pb.Y, where pa and pb are the points (from p0,p1) with smallest & largest Y. - * When both have the same Y, no intersections are counted but there is a special test to see if the point falls - * exactly on the line. - * - * Returns false if outside, true if inside; if the point lies exactly on the border, will return \p border_result. - * - * \deprecated This function is old and no longer used. instead use \ref Polygons::inside - * - * \param p The point for which to check if it is inside this polygon - * \param border_result What to return when the point is exactly on the border - * \return Whether the point \p p is inside this polygon (or \p border_result when it is on the border) - */ - bool insideOld(Point2LL p, bool border_result = false) const; - - /*! - * Find the polygon inside which point \p p resides. - * - * We do this by tracing from the point towards the positive X direction, - * every line we cross increments the crossings counter. If we have an even number of crossings then we are not inside the polygon. - * We then find the polygon with an uneven number of crossings which is closest to \p p. - * - * If \p border_result, we return the first polygon which is exactly on \p p. - * - * \param p The point for which to check in which polygon it is. - * \param border_result Whether a point exactly on a polygon counts as inside - * \return The index of the polygon inside which the point \p p resides - */ - size_t findInside(Point2LL p, bool border_result = false); - - /*! - * Approximates the convex hull of the polygons. - * \p extra_outset Extra offset outward - * \return the convex hull (approximately) - * - */ - Polygons approxConvexHull(int extra_outset = 0); - - /*! - * Make each of the polygons convex - */ - void makeConvex(); - - /*! - * Compute the area enclosed within the polygons (minus holes) - * - * \return The area in square micron - */ - double area() const; - - /*! - * Smooth out small perpendicular segments - * Smoothing is performed by removing the inner most vertex of a line segment smaller than \p remove_length - * which has an angle with the next and previous line segment smaller than roughly 150* - * - * Note that in its current implementation this function doesn't remove line segments with an angle smaller than 30* - * Such would be the case for an N shape. - * - * \param remove_length The length of the largest segment removed - * \return The smoothed polygon - */ - Polygons smooth(int remove_length) const; - - /*! - * Smooth out sharp inner corners, by taking a shortcut which bypasses the corner - * - * \param angle The maximum angle of inner corners to be smoothed out - * \param shortcut_length The desired length of the shortcut line segment introduced (shorter shortcuts may be unavoidable) - * \return The resulting polygons - */ - Polygons smooth_outward(const AngleDegrees angle, int shortcut_length); - - Polygons smooth2(int remove_length, int min_area) const; //!< removes points connected to small lines - - void removeColinearEdges(const AngleRadians max_deviation_angle = AngleRadians(0.0005)) - { - Polygons& thiss = *this; - for (size_t p = 0; p < size(); p++) - { - thiss[p].removeColinearEdges(max_deviation_angle); - if (thiss[p].size() < 3) - { - remove(p); - p--; - } - } - } - -public: - void scale(const Ratio& ratio) - { - if (ratio == 1.) - { - return; - } - - for (auto& points : *this) - { - for (auto& pt : points) - { - pt = pt * static_cast(ratio); - } - } - } - - void translate(const Point2LL vec) - { - if (vec.X == 0 && vec.Y == 0) - { - return; - } - - for (PolygonRef poly : *this) - { - poly.translate(vec); - } - } - - /*! - * Remove all but the polygons on the very outside. - * Exclude holes and parts within holes. - * \return the resulting polygons. - */ - Polygons getOutsidePolygons() const; - - /*! - * Split up the polygons into groups according to the even-odd rule. - * Each PolygonsPart in the result has an outline as first polygon, whereas the rest are holes. - */ - std::vector splitIntoParts(bool unionAll = false) const; - - /*! - * Sort the polygons into bins where each bin has polygons which are contained within one of the polygons in the previous bin. - * - * \warning When polygons are crossing each other the result is undefined. - */ - std::vector sortByNesting() const; - - /*! - * Utility method for creating the tube (or 'donut') of a shape. - * \param inner_offset Offset relative to the original shape-outline towards the inside of the shape. Sort-of like a negative normal offset, except it's the offset part that's - * kept, not the shape. \param outer_offset Offset relative to the original shape-outline towards the outside of the shape. Comparable to normal offset. \return The resulting - * polygons. - */ - Polygons tubeShape(const coord_t inner_offset, const coord_t outer_offset) const; - -private: - /*! - * recursive part of \ref Polygons::removeEmptyHoles and \ref Polygons::getEmptyHoles - * \param node The node of the polygons part to process - * \param remove_holes Whether to remove empty holes or everything but the empty holes - * \param ret Where to store polygons which are not empty holes - */ - void removeEmptyHoles_processPolyTreeNode(const ClipperLib::PolyNode& node, const bool remove_holes, Polygons& ret) const; - void splitIntoParts_processPolyTreeNode(ClipperLib::PolyNode* node, std::vector& ret) const; - void sortByNesting_processPolyTreeNode(ClipperLib::PolyNode* node, const size_t nesting_idx, std::vector& ret) const; - -public: - /*! - * Split up the polygons into groups according to the even-odd rule. - * Each vector in the result has the index to an outline as first index, whereas the rest are indices to holes. - * - * \warning Note that this function reorders the polygons! - */ - PartsView splitIntoPartsView(bool unionAll = false); - -private: - void splitIntoPartsView_processPolyTreeNode(PartsView& partsView, Polygons& reordered, ClipperLib::PolyNode* node) const; - -public: - /*! - * Removes polygons with area smaller than \p min_area_size (note that min_area_size is in mm^2, not in micron^2). - * Unless \p remove_holes is true, holes are not removed even if their area is below \p min_area_size. - * However, holes that are contained within outlines whose area is below the threshold are removed though. - */ - void removeSmallAreas(const double min_area_size, const bool remove_holes = false); - - /*! - * Removes polygons with circumference smaller than \p min_circumference_size (in micron). - * Unless \p remove_holes is true, holes are not removed even if their circumference is below \p min_circumference_size. - * However, holes that are contained within outlines whose circumference is below the threshold are removed though. - */ - [[maybe_unused]] void removeSmallCircumference(const coord_t min_circumference_size, const bool remove_holes = false); - - /*! - * Removes polygons with circumference smaller than \p min_circumference_size (in micron) _and_ - * an area smaller then \p min_area_size (note that min_area_size is in mm^2, not in micron^2). - * Unless \p remove_holes is true, holes are not removed even if their circumference is - * below \p min_circumference_size and their area smaller then \p min_area_size. - * However, holes that are contained within outlines whose circumference is below the threshold are removed though. - */ - [[maybe_unused]] void removeSmallAreaCircumference(const double min_area_size, const coord_t min_circumference_size, const bool remove_holes = false); - - /*! - * Removes overlapping consecutive line segments which don't delimit a - * positive area. - * - * This function is meant to work on polygons, not polylines. When misused - * on polylines, it may cause too many vertices to be removed. - * See \ref removeDegenerateVertsPolyline for a version that works on - * polylines. - */ - void removeDegenerateVerts(); - - /*! - * Removes overlapping consecutive line segments which don't delimit a - * positive area. - * - * This version is meant to work on polylines, not polygons. It leaves the - * endpoints of the polyline untouched. When misused on polygons, it may - * leave some degenerate vertices in. - * See \ref removeDegenerateVerts for a version that works on polygons. - */ - void removeDegenerateVertsPolyline(); - - /*! - * Removes overlapping consecutive line segments which don't delimit a - * positive area. - * \param for_polyline Indicate that we're removing degenerate vertices from - * a polyline, causing the endpoints of the polyline to be left untouched. - * When removing vertices from a polygon, the start and end can be - * considered for removal too, but when processing a polyline, removing - * those would cause the polyline to become shorter. - */ - void _removeDegenerateVerts(const bool for_polyline = false); - - /*! - * Removes the same polygons from this set (and also empty polygons). - * Polygons are considered the same if all points lie within [same_distance] of their counterparts. - */ - Polygons remove(const Polygons& to_be_removed, int same_distance = 0) const - { - Polygons result; - for (size_t poly_keep_idx = 0; poly_keep_idx < size(); poly_keep_idx++) - { - ConstPolygonRef poly_keep = (*this)[poly_keep_idx]; - bool should_be_removed = false; - if (poly_keep.size() > 0) - // for (int hole_poly_idx = 0; hole_poly_idx < to_be_removed.size(); hole_poly_idx++) - for (ConstPolygonRef poly_rem : to_be_removed) - { - // PolygonRef poly_rem = to_be_removed[hole_poly_idx]; - if (poly_rem.size() != poly_keep.size() || poly_rem.size() == 0) - continue; - - // find closest point, supposing this point aligns the two shapes in the best way - size_t closest_point_idx = 0; - coord_t smallestDist2 = -1; - for (size_t point_rem_idx = 0; point_rem_idx < poly_rem.size(); point_rem_idx++) - { - coord_t dist2 = vSize2(poly_rem[point_rem_idx] - poly_keep[0]); - if (dist2 < smallestDist2 || smallestDist2 < 0) - { - smallestDist2 = dist2; - closest_point_idx = point_rem_idx; - } - } - bool poly_rem_is_poly_keep = true; - // compare the two polygons on all points - if (smallestDist2 > same_distance * same_distance) - continue; - for (size_t point_idx = 0; point_idx < poly_rem.size(); point_idx++) - { - coord_t dist2 = vSize2(poly_rem[(closest_point_idx + point_idx) % poly_rem.size()] - poly_keep[point_idx]); - if (dist2 > same_distance * same_distance) - { - poly_rem_is_poly_keep = false; - break; - } - } - if (poly_rem_is_poly_keep) - { - should_be_removed = true; - break; - } - } - if (! should_be_removed) - result.add(poly_keep); - } - return result; - } - - Polygons processEvenOdd(ClipperLib::PolyFillType poly_fill_type = ClipperLib::PolyFillType::pftEvenOdd) const - { - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - clipper.Execute(ClipperLib::ctUnion, ret.paths, poly_fill_type); - return ret; - } - - /*! - * Ensure the polygon is manifold, by removing small areas where the polygon touches itself. - * ____ ____ - * | | | | - * | |____ ==> | / ____ - * """"| | """ / | - * |____| |____| - * - */ - void ensureManifold(); - - coord_t polygonLength() const - { - coord_t length = 0; - for (ConstPolygonRef poly : *this) - { - length += poly.polygonLength(); - } - return length; - } - - coord_t polyLineLength() const; - - Point2LL min() const - { - Point2LL ret = Point2LL(POINT_MAX, POINT_MAX); - for (const ClipperLib::Path& polygon : paths) - { - for (Point2LL p : polygon) - { - ret.X = std::min(ret.X, p.X); - ret.Y = std::min(ret.Y, p.Y); - } - } - return ret; - } - - Point2LL max() const - { - Point2LL ret = Point2LL(POINT_MIN, POINT_MIN); - for (const ClipperLib::Path& polygon : paths) - { - for (Point2LL p : polygon) - { - ret.X = std::max(ret.X, p.X); - ret.Y = std::max(ret.Y, p.Y); - } - } - return ret; - } - - void applyMatrix(const PointMatrix& matrix) - { - for (size_t i = 0; i < paths.size(); i++) - { - for (size_t j = 0; j < paths[i].size(); j++) - { - paths[i][j] = matrix.apply(paths[i][j]); - } - } - } - - void applyMatrix(const Point3Matrix& matrix) - { - for (size_t i = 0; i < paths.size(); i++) - { - for (size_t j = 0; j < paths[i].size(); j++) - { - paths[i][j] = matrix.apply(paths[i][j]); - } - } - } - - Polygons offset(const std::vector& offset_dists) const; - - /*! - * @brief Export the polygon to a WKT string - * - * @param stream The stream to write to - */ - [[maybe_unused]] void writeWkt(std::ostream& stream) const; - - /*! - * @brief Import the polygon from a WKT string - * - * @param wkt The WKT string to read from - * @return Polygons The polygons read from the stream - */ - [[maybe_unused]] static Polygons fromWkt(const std::string& wkt); - - /*! - * @brief Remove self-intersections from the polygons - * _note_: this function uses wagyu to remove the self intersections. - * since wagyu uses a different internal representation of the polygons - * we need to convert back and forward between data structures which - * might impact performance, use wisely! - * - * @return Polygons - the cleaned polygons - */ - Polygons removeNearSelfIntersections() const; -}; - -/*! - * A single area with holes. The first polygon is the outline, while the rest are holes within this outline. - * - * This class has little more functionality than Polygons, but serves to show that a specific instance is ordered such that the first Polygon is the outline and the rest are holes. - */ -class PolygonsPart : public Polygons -{ -public: - PolygonRef outerPolygon() - { - return paths[0]; - } - ConstPolygonRef outerPolygon() const - { - return paths[0]; - } - - /*! - * Tests whether the given point is inside this polygon part. - * \param p The point to test whether it is inside. - * \param border_result If the point is exactly on the border, this will be - * returned instead. - */ - bool inside(Point2LL p, bool border_result = false) const; -}; - -/*! - * Extension of vector> which is similar to a vector of PolygonParts, except the base of the container is indices to polygons into the original Polygons, - * instead of the polygons themselves - */ -class PartsView : public std::vector> -{ -public: - Polygons& polygons_; - PartsView(Polygons& polygons) - : polygons_(polygons) - { - } - /*! - * Get the index of the PolygonsPart of which the polygon with index \p poly_idx is part. - * - * \param poly_idx The index of the polygon in \p polygons - * \param boundary_poly_idx Optional output parameter: The index of the boundary polygon of the part in \p polygons - * \return The PolygonsPart containing the polygon with index \p poly_idx - */ - size_t getPartContaining(size_t poly_idx, size_t* boundary_poly_idx = nullptr) const; - /*! - * Assemble the PolygonsPart of which the polygon with index \p poly_idx is part. - * - * \param poly_idx The index of the polygon in \p polygons - * \param boundary_poly_idx Optional output parameter: The index of the boundary polygon of the part in \p polygons - * \return The PolygonsPart containing the polygon with index \p poly_idx - */ - PolygonsPart assemblePartContaining(size_t poly_idx, size_t* boundary_poly_idx = nullptr) const; - /*! - * Assemble the PolygonsPart of which the polygon with index \p poly_idx is part. - * - * \param part_idx The index of the part - * \return The PolygonsPart with index \p poly_idx - */ - PolygonsPart assemblePart(size_t part_idx) const; -}; - -} // namespace cura - -#endif // UTILS_POLYGON_H diff --git a/include/utils/polygonUtils.h b/include/utils/polygonUtils.h index b3895a5ff7..825bc62e79 100644 --- a/include/utils/polygonUtils.h +++ b/include/utils/polygonUtils.h @@ -1,4 +1,4 @@ -// Copyright (c) 2021 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #ifndef UTILS_POLYGON_UTILS_H @@ -7,12 +7,13 @@ #include // function #include #include // unique_ptr +#include #include #include "PolygonsPointIndex.h" #include "SparseLineGrid.h" #include "SparsePointGridInclusive.h" -#include "polygon.h" +#include "geometry/Polygon.h" namespace cura { @@ -20,14 +21,15 @@ namespace cura /*! * Result of finding the closest point to a given within a set of polygons, with extra information on where the point is. */ -struct ClosestPolygonPoint +template +struct ClosestPoint { Point2LL location_; //!< Result location - ConstPolygonPointer poly_; //!< Polygon in which the result was found (or nullptr if no result was found) - size_t poly_idx_; //!< The index of the polygon in some Polygons where ClosestPolygonPoint::poly can be found + const LineType* poly_{ nullptr }; //!< Line in which the result was found (or nullptr if no result was found) + size_t poly_idx_; //!< The index of the polygon in some Polygons where ClosestPoint::poly can be found size_t point_idx_; //!< Index to the first point in the polygon of the line segment on which the result was found - ClosestPolygonPoint(Point2LL p, size_t pos, ConstPolygonRef poly) + ClosestPoint(Point2LL p, size_t pos, const LineType* poly) : location_(p) , poly_(poly) , poly_idx_(NO_INDEX) @@ -35,7 +37,7 @@ struct ClosestPolygonPoint { } - ClosestPolygonPoint(Point2LL p, size_t pos, ConstPolygonRef poly, size_t poly_idx) + ClosestPoint(Point2LL p, size_t pos, const LineType* poly, size_t poly_idx) : location_(p) , poly_(poly) , poly_idx_(poly_idx) @@ -43,14 +45,14 @@ struct ClosestPolygonPoint { } - ClosestPolygonPoint(ConstPolygonRef poly) + ClosestPoint(const LineType* poly) : poly_(poly) , poly_idx_(NO_INDEX) , point_idx_(NO_INDEX) { } - ClosestPolygonPoint() + ClosestPoint() : poly_idx_(NO_INDEX) , point_idx_(NO_INDEX) { @@ -66,7 +68,7 @@ struct ClosestPolygonPoint return point_idx_ != NO_INDEX; } - bool operator==(const ClosestPolygonPoint& rhs) const + bool operator==(const ClosestPoint& rhs) const { // no need to compare on poy_idx // it's sometimes unused while poly is always initialized @@ -74,14 +76,16 @@ struct ClosestPolygonPoint } }; +using ClosestPointPolygon = ClosestPoint; + } // namespace cura namespace std { template<> -struct hash +struct hash { - size_t operator()(const cura::ClosestPolygonPoint& cpp) const + size_t operator()(const cura::ClosestPointPolygon& cpp) const { return std::hash()(cpp.p()); } @@ -147,14 +151,14 @@ class PolygonUtils * \param n_dots number of dots to spread out * \param result Where to store the generated points */ - static void spreadDots(PolygonsPointIndex start, PolygonsPointIndex end, unsigned int n_dots, std::vector& result); + static void spreadDots(PolygonsPointIndex start, PolygonsPointIndex end, unsigned int n_dots, std::vector& result); /*! * Generate a grid of dots inside of the area of the \p polygons. */ - static std::vector spreadDotsArea(const Polygons& polygons, coord_t grid_size); + static std::vector spreadDotsArea(const Shape& polygons, coord_t grid_size); - static std::vector spreadDotsArea(const Polygons& polygons, Point2LL grid_size); + static std::vector spreadDotsArea(const Shape& polygons, Point2LL grid_size); /*! * Whether a polygon intersects with a line-segment. If true, the closest collision point to 'b' is stored in the result. @@ -162,7 +166,7 @@ class PolygonUtils static bool lineSegmentPolygonsIntersection( const Point2LL& a, const Point2LL& b, - const Polygons& current_outlines, + const Shape& current_outlines, const LocToLineGrid& outline_locator, Point2LL& result, const coord_t within_max_dist); @@ -175,7 +179,7 @@ class PolygonUtils * \param poly The polygon. * \param point_idx The index of the point in the polygon. */ - static Point2LL getVertexInwardNormal(ConstPolygonRef poly, unsigned int point_idx); + static Point2LL getVertexInwardNormal(const Polyline& poly, unsigned int point_idx); /*! * Get a point from the \p poly with a given \p offset. @@ -185,7 +189,7 @@ class PolygonUtils * \param offset The distance the point has to be moved outward from the polygon. * \return A point at the given distance inward from the point on the boundary polygon. */ - static Point2LL getBoundaryPointWithOffset(ConstPolygonRef poly, unsigned int point_idx, int64_t offset); + static Point2LL getBoundaryPointWithOffset(const Polyline& poly, unsigned int point_idx, int64_t offset); /*! * Move a point away from the boundary by looking at the boundary normal of the nearest vert. @@ -193,7 +197,7 @@ class PolygonUtils * \param point_on_boundary The object holding the point on the boundary along with the information of which line segment the point is on. * \param offset The distance the point has to be moved inward from the polygon. */ - static Point2LL moveInsideDiagonally(ClosestPolygonPoint point_on_boundary, int64_t inset); + static Point2LL moveInsideDiagonally(ClosestPointPolygon point_on_boundary, int64_t inset); /*! * Moves the point \p from onto the nearest polygon or leaves the point as-is, when the comb boundary is not within the root of \p max_dist2 distance. @@ -207,7 +211,7 @@ class PolygonUtils * \param max_dist2 The squared maximal allowed distance from the point to the nearest polygon. * \return The index to the polygon onto which we have moved the point. */ - static size_t moveInside(const Polygons& polygons, Point2LL& from, int distance = 0, int64_t max_dist2 = std::numeric_limits::max()); + static size_t moveInside(const Shape& polygons, Point2LL& from, int distance = 0, int64_t max_dist2 = std::numeric_limits::max()); /** * \brief Moves the point \p from onto the nearest polygon or leaves the @@ -225,7 +229,7 @@ class PolygonUtils * the polygon. * \return Always returns 0. */ - static unsigned int moveInside(const ConstPolygonRef polygon, Point2LL& from, int distance = 0, int64_t max_dist2 = std::numeric_limits::max()); + static unsigned int moveInside(const ClosedPolyline& polygon, Point2LL& from, int distance = 0, int64_t max_dist2 = std::numeric_limits::max()); /*! * Moves the point \p from onto the nearest polygon or leaves the point as-is, when the comb boundary is not within the root of \p max_dist2 distance. @@ -235,7 +239,7 @@ class PolygonUtils * * \warning If \p loc_to_line_grid is used, it's best to have all and only \p polygons in there. * If \p from is not closest to \p polygons this function may - * return a ClosestPolygonPoint on a polygon in \p loc_to_line_grid which is not in \p polygons. + * return a ClosestPointPolygon on a polygon in \p loc_to_line_grid which is not in \p polygons. * * \param polygons The polygons onto which to move the point * \param from[in,out] The point to move. @@ -246,12 +250,12 @@ class PolygonUtils * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. * \return The point on the polygon closest to \p from */ - static ClosestPolygonPoint moveInside2( - const Polygons& polygons, + static ClosestPointPolygon moveInside2( + const Shape& polygons, Point2LL& from, const int distance = 0, const int64_t max_dist2 = std::numeric_limits::max(), - const Polygons* loc_to_line_polygons = nullptr, + const Shape* loc_to_line_polygons = nullptr, const LocToLineGrid* loc_to_line_grid = nullptr, const std::function& penalty_function = no_penalty_function); @@ -274,9 +278,9 @@ class PolygonUtils * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. * \return The point on the polygon closest to \p from */ - static ClosestPolygonPoint moveInside2( - const Polygons& loc_to_line_polygons, - ConstPolygonRef polygon, + static ClosestPointPolygon moveInside2( + const Shape& loc_to_line_polygons, + const Polygon& polygon, Point2LL& from, const int distance = 0, const int64_t max_dist2 = std::numeric_limits::max(), @@ -298,7 +302,7 @@ class PolygonUtils * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. * \return The index to the polygon onto which we have moved the point. */ - static unsigned int moveOutside(const Polygons& polygons, Point2LL& from, int distance = 0, int64_t max_dist2 = std::numeric_limits::max()); + static unsigned int moveOutside(const Shape& polygons, Point2LL& from, int distance = 0, int64_t max_dist2 = std::numeric_limits::max()); /*! * Compute a point at a distance from a point on the boundary in orthogonal direction to the boundary. @@ -308,7 +312,7 @@ class PolygonUtils * \param distance The distance by which to move the point. * \return A point at a \p distance from the point in \p cpp orthogonal to the boundary there. */ - static Point2LL moveInside(const ClosestPolygonPoint& cpp, const int distance); + static Point2LL moveInside(const ClosestPointPolygon& cpp, const int distance); /*! * The opposite of moveInside. @@ -320,7 +324,7 @@ class PolygonUtils * \param distance The distance by which to move the point. * \return A point at a \p distance from the point in \p cpp orthogonal to the boundary there. */ - static Point2LL moveOutside(const ClosestPolygonPoint& cpp, const int distance); + static Point2LL moveOutside(const ClosestPointPolygon& cpp, const int distance); /*! * Moves the point \p from onto the nearest polygon or leaves the point as-is, when the comb boundary is not within \p distance. @@ -345,12 +349,12 @@ class PolygonUtils * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. * \return The point on the polygon closest to \p from */ - static ClosestPolygonPoint ensureInsideOrOutside( - const Polygons& polygons, + static ClosestPointPolygon ensureInsideOrOutside( + const Shape& polygons, Point2LL& from, int preferred_dist_inside, int64_t max_dist2 = std::numeric_limits::max(), - const Polygons* loc_to_line_polygons = nullptr, + const Shape* loc_to_line_polygons = nullptr, const LocToLineGrid* loc_to_line_grid = nullptr, const std::function& penalty_function = no_penalty_function); @@ -377,12 +381,12 @@ class PolygonUtils * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. * \return The point on the polygon closest to \p from */ - static ClosestPolygonPoint ensureInsideOrOutside( - const Polygons& polygons, + static ClosestPointPolygon ensureInsideOrOutside( + const Shape& polygons, Point2LL& from, - const ClosestPolygonPoint& closest_polygon_point, + const ClosestPointPolygon& closest_polygon_point, int preferred_dist_inside, - const Polygons* loc_to_line_polygons = nullptr, + const Shape* loc_to_line_polygons = nullptr, const LocToLineGrid* loc_to_line_grid = nullptr, const std::function& penalty_function = no_penalty_function); @@ -390,7 +394,7 @@ class PolygonUtils * * \warning Assumes \p poly1_result and \p poly2_result have their pos and poly fields initialized! */ - static void walkToNearestSmallestConnection(ClosestPolygonPoint& poly1_result, ClosestPolygonPoint& poly2_result); + static void walkToNearestSmallestConnection(ClosestPointPolygon& poly1_result, ClosestPointPolygon& poly2_result); /*! * Find the nearest closest point on a polygon from a given index. @@ -400,7 +404,7 @@ class PolygonUtils * \param start_idx The index of the point in the polygon from which to start looking. * \return The nearest point from \p start_idx going along the \p polygon (in both directions) with a locally minimal distance to \p from. */ - static ClosestPolygonPoint findNearestClosest(Point2LL from, ConstPolygonRef polygon, int start_idx); + static ClosestPointPolygon findNearestClosest(Point2LL from, const Polygon& polygon, int start_idx); /*! * Find the nearest closest point on a polygon from a given index walking in one direction along the polygon. @@ -411,7 +415,7 @@ class PolygonUtils * \param direction The direction to walk: 1 for walking along the \p polygon, -1 for walking in opposite direction * \return The nearest point from \p start_idx going along the \p polygon with a locally minimal distance to \p from. */ - static ClosestPolygonPoint findNearestClosest(const Point2LL from, ConstPolygonRef polygon, int start_idx, int direction); + static ClosestPointPolygon findNearestClosest(const Point2LL from, const Polygon& polygon, int start_idx, int direction); /*! * Find the point closest to \p from in all polygons in \p polygons. @@ -420,7 +424,7 @@ class PolygonUtils * * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. */ - static ClosestPolygonPoint findClosest(Point2LL from, const Polygons& polygons, const std::function& penalty_function = no_penalty_function); + static ClosestPointPolygon findClosest(Point2LL from, const Shape& polygons, const std::function& penalty_function = no_penalty_function); /*! * Find the point closest to \p from in the polygon \p polygon. @@ -429,7 +433,7 @@ class PolygonUtils * * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. */ - static ClosestPolygonPoint findClosest(Point2LL from, ConstPolygonRef polygon, const std::function& penalty_function = no_penalty_function); + static ClosestPointPolygon findClosest(Point2LL from, const Polygon& polygon, const std::function& penalty_function = no_penalty_function); /*! * Find the nearest vertex to \p from in \p polys @@ -437,7 +441,7 @@ class PolygonUtils * \param polys The polygons in which to search * \return The nearest vertex on the polygons */ - static PolygonsPointIndex findNearestVert(const Point2LL from, const Polygons& polys); + static PolygonsPointIndex findNearestVert(const Point2LL from, const Shape& polys); /*! * Find the nearest vertex to \p from in \p poly @@ -445,7 +449,7 @@ class PolygonUtils * \param poly The polygon in which to search * \return The index to the nearest vertex on the polygon */ - static unsigned int findNearestVert(const Point2LL from, ConstPolygonRef poly); + static unsigned int findNearestVert(const Point2LL from, const Polygon& poly); /*! * Create a SparsePointGridInclusive mapping from locations to line segments occurring in the \p polygons @@ -456,7 +460,7 @@ class PolygonUtils * \param square_size The cell size used to bundle line segments (also used to chop up lines so that multiple cells contain the same long line) * \return A bucket grid mapping spatial locations to poly-point indices into \p polygons */ - static std::unique_ptr createLocToLineGrid(const Polygons& polygons, int square_size); + static std::unique_ptr createLocToLineGrid(const Shape& polygons, int square_size); /*! * Find the line segment closest to a given point \p from within a cell-block of a size defined in the SparsePointGridInclusive \p loc_to_line @@ -470,8 +474,8 @@ class PolygonUtils * \param penalty_function A function returning a penalty term on the squared distance score of a candidate point. * \return The nearest point on the polygon if the polygon was within a distance equal to the cell_size of the SparsePointGridInclusive */ - static std::optional - findClose(Point2LL from, const Polygons& polygons, const LocToLineGrid& loc_to_line, const std::function& penalty_function = no_penalty_function); + static std::optional + findClose(Point2LL from, const Shape& polygons, const LocToLineGrid& loc_to_line, const std::function& penalty_function = no_penalty_function); /*! * Find the line segment closest to any point on \p from within cell-blocks of a size defined in the SparsePointGridInclusive \p destination_loc_to_line @@ -486,9 +490,9 @@ class PolygonUtils * \return A collection of near crossing from the \p from polygon to the \p destination polygon. Each element in the sollection is a pair with as first a cpp in the \p from * polygon and as second a cpp in the \p destination polygon. */ - static std::vector> findClose( - ConstPolygonRef from, - const Polygons& destination, + static std::vector> findClose( + const Polygon& from, + const Shape& destination, const LocToLineGrid& destination_loc_to_line, const std::function& penalty_function = no_penalty_function); @@ -517,12 +521,13 @@ class PolygonUtils * \param start_idx the index of the prev poly point on the poly. * \param poly_start_idx The index of the point in the polygon which is to be handled as the start of the polygon. No point further than this point will be the result. */ - static bool getNextPointWithDistance(Point2LL from, int64_t dist, ConstPolygonRef poly, int start_idx, int poly_start_idx, GivenDistPoint& result); + static bool getNextPointWithDistance(Point2LL from, int64_t dist, const OpenPolyline& poly, int start_idx, int poly_start_idx, GivenDistPoint& result); /*! * Walk a given \p distance along the polygon from a given point \p from on the polygon */ - static ClosestPolygonPoint walk(const ClosestPolygonPoint& from, coord_t distance); + template + static ClosestPoint walk(const ClosestPoint& from, coord_t distance); /*! * Get the point on a polygon which intersects a line parallel to a line going through the starting point and through another point. @@ -536,7 +541,7 @@ class PolygonUtils * \param forward Whether to look forward from \p start in the direction of the polygon, or go in the other direction. * \return The earliest point on the polygon in the given direction which crosses a line parallel to the given one at the distance \p dist - if any */ - static std::optional getNextParallelIntersection(const ClosestPolygonPoint& start, const Point2LL& line_to, const coord_t dist, const bool forward); + static std::optional getNextParallelIntersection(const ClosestPointPolygon& start, const Point2LL& line_to, const coord_t dist, const bool forward); /*! * Checks whether a given line segment collides with a given polygon(s). @@ -559,7 +564,7 @@ class PolygonUtils * polygon(s) */ static bool - polygonCollidesWithLineSegment(ConstPolygonRef poly, const Point2LL& transformed_startPoint, const Point2LL& transformed_endPoint, PointMatrix transformation_matrix); + polygonCollidesWithLineSegment(const Polygon& poly, const Point2LL& transformed_startPoint, const Point2LL& transformed_endPoint, PointMatrix transformation_matrix); /*! * Checks whether a given line segment collides with a given polygon(s). @@ -575,7 +580,7 @@ class PolygonUtils * \return whether the line segment collides with the boundary of the * polygon(s) */ - static bool polygonCollidesWithLineSegment(ConstPolygonRef poly, const Point2LL& startPoint, const Point2LL& endPoint); + static bool polygonCollidesWithLineSegment(const Polygon& poly, const Point2LL& startPoint, const Point2LL& endPoint); /*! * Checks whether a given line segment collides with a given polygon(s). @@ -597,8 +602,7 @@ class PolygonUtils * \return whether the line segment collides with the boundary of the * polygon(s) */ - static bool - polygonCollidesWithLineSegment(const Polygons& polys, const Point2LL& transformed_startPoint, const Point2LL& transformed_endPoint, PointMatrix transformation_matrix); + static bool polygonCollidesWithLineSegment(const Shape& polys, const Point2LL& transformed_startPoint, const Point2LL& transformed_endPoint, PointMatrix transformation_matrix); /*! * Checks whether a given line segment collides with a given polygon(s). @@ -614,7 +618,7 @@ class PolygonUtils * \return whether the line segment collides with the boundary of the * polygon(s) */ - static bool polygonCollidesWithLineSegment(const Polygons& polys, const Point2LL& startPoint, const Point2LL& endPoint); + static bool polygonCollidesWithLineSegment(const Shape& polys, const Point2LL& startPoint, const Point2LL& endPoint); /*! * Checks whether two polygon groups intersect - does a BB hit check first and if that succeeds, the full intersection @@ -623,7 +627,7 @@ class PolygonUtils * \param poly_b Another polygon group * \return true if \p poly_a and \p poly_b intersect, false otherwise */ - static bool polygonsIntersect(const ConstPolygonRef& poly_a, const ConstPolygonRef& poly_b); + static bool polygonsIntersect(const Polygon& poly_a, const Polygon& poly_b); /*! * Checks whether two polygons are adjacent (closer than \p max_gap) @@ -633,7 +637,7 @@ class PolygonUtils * \param[in] max_gap Polygons must be closer together than this distance to be considered adjacent. * \return true if a vertex in \p inner_poly is sufficiently close to a line in \p outer_poly, false otherwise */ - static bool polygonOutlinesAdjacent(const ConstPolygonRef inner_poly, const ConstPolygonRef outer_poly, const coord_t max_gap); + static bool polygonOutlinesAdjacent(const Polygon& inner_poly, const Polygon& outer_poly, const coord_t max_gap); /*! * Searches \p possible_adjacent_polys for polygons that are closer to \p poly than \p max_gap. The indices of adjacent polygons are stored in \p adjacent_poly_indices. @@ -643,11 +647,8 @@ class PolygonUtils * \param[in] possible_adjacent_polys The vector of polygons we are testing. * \param[in] max_gap Polygons must be closer together than this distance to be considered adjacent. */ - static void findAdjacentPolygons( - std::vector& adjacent_poly_indices, - const ConstPolygonRef& poly, - const std::vector& possible_adjacent_polys, - const coord_t max_gap); + static void + findAdjacentPolygons(std::vector& adjacent_poly_indices, const Polygon& poly, const std::vector& possible_adjacent_polys, const coord_t max_gap); /*! * Calculate the Hamming Distance between two polygons relative to their own @@ -662,7 +663,7 @@ class PolygonUtils * two polygons. This will be between 0.0 (the polygons are exactly equal) * and 1.0 (the polygons are completely disjunct). */ - static double relativeHammingDistance(const Polygons& poly_a, const Polygons& poly_b); + static double relativeHammingDistance(const Shape& poly_a, const Shape& poly_b); /*! * Create an approximation of a circle. @@ -691,11 +692,11 @@ class PolygonUtils /*! * Connect all polygons to their holes using zero widths hole channels, so that the polygons and their outlines are connected together */ - static Polygons connect(const Polygons& input); + static Shape connect(const Shape& input); - static void fixSelfIntersections(const coord_t epsilon, Polygons& thiss); + static void fixSelfIntersections(const coord_t epsilon, Shape& polygon); - static Polygons unionManySmall(const Polygons& p); + static Shape unionManySmall(const Shape& polygon); /*! @@ -704,7 +705,7 @@ class PolygonUtils * \param aabb The AABB with which the polygon that has to be intersected with * \return A new Polygon that is said intersection */ - static Polygons clipPolygonWithAABB(const Polygons& src, const AABB& aabb); + static Shape clipPolygonWithAABB(const Shape& src, const AABB& aabb); /*! * Generate a few outset polygons around the given base, according to the given line width @@ -714,7 +715,7 @@ class PolygonUtils * \param line_width The actual line width to distance the polygons from each other (and from the base) * \return The generated outset polygons */ - static Polygons generateOutset(const Polygons& inner_poly, size_t count, coord_t line_width); + static Shape generateOutset(const Shape& inner_poly, size_t count, coord_t line_width); /*! * Generate inset polygons inside the given base, until there is no space left, according to the given line width @@ -724,20 +725,20 @@ class PolygonUtils * \param initial_inset The inset distance to be added to the first generated polygon * \return The generated inset polygons */ - static Polygons generateInset(const Polygons& outer_poly, coord_t line_width, coord_t initial_inset = 0); + static Shape generateInset(const Shape& outer_poly, coord_t line_width, coord_t initial_inset = 0); private: /*! * Helper function for PolygonUtils::moveInside2: moves a point \p from which was moved onto \p closest_polygon_point towards inside/outside when it's not already * inside/outside by enough distance. * - * \param closest_polygon_point The ClosestPolygonPoint we have to move inside + * \param closest_polygon_point The ClosestPointPolygon we have to move inside * \param distance The distance by which to move the point. * \param from[in,out] The point to move. * \param max_dist2 The squared maximal allowed distance from the point to the nearest polygon. * \return The point on the polygon closest to \p from */ - static ClosestPolygonPoint _moveInside2(const ClosestPolygonPoint& closest_polygon_point, const int distance, Point2LL& from, const int64_t max_dist2); + static ClosestPointPolygon _moveInside2(const ClosestPointPolygon& closest_polygon_point, const int distance, Point2LL& from, const int64_t max_dist2); }; diff --git a/include/utils/types/generic.h b/include/utils/types/generic.h index 7d53dcfe86..4b4d6fc474 100644 --- a/include/utils/types/generic.h +++ b/include/utils/types/generic.h @@ -1,15 +1,15 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #ifndef CURAENGINE_GENERIC_H #define CURAENGINE_GENERIC_H -#include - #include #include #include +#include + namespace cura::utils { // clang-format off @@ -63,6 +63,12 @@ concept floating_point = std::floating_point; template concept numeric = std::is_arithmetic_v>; + +template +concept multipliable = requires(T a, T b) +{ + { a * b }; +}; } // namespace cura::utils #endif // CURAENGINE_GENERIC_H diff --git a/src/Application.cpp b/src/Application.cpp index 897b4c89eb..cb0f17b297 100644 --- a/src/Application.cpp +++ b/src/Application.cpp @@ -18,10 +18,8 @@ #include #include -#include "FffProcessor.h" #include "communication/ArcusCommunication.h" //To connect via Arcus to the front-end. #include "communication/CommandLine.h" //To use the command line to slice stuff. -#include "plugins/slots.h" #include "progress/Progress.h" #include "utils/ThreadPool.h" #include "utils/string.h" //For stringcasecompare. @@ -133,6 +131,7 @@ void Application::printHelp() const fmt::print(" -p\n\tLog progress information.\n"); fmt::print(" -d Add definition search paths seperated by a `:` (Unix) or `;` (Windows)\n"); fmt::print(" -j\n\tLoad settings.def.json file to register all settings and their defaults.\n"); + fmt::print(" -r\n\tLoad a json file containing resolved setting values.\n"); fmt::print(" -s =\n\tSet a setting to a value for the last supplied object, \n\textruder train, or general settings.\n"); fmt::print(" -l \n\tLoad an STL model. \n"); fmt::print(" -g\n\tSwitch setting focus to the current mesh group only.\n\tUsed for one-at-a-time printing.\n"); diff --git a/src/BeadingStrategy/BeadingStrategyFactory.cpp b/src/BeadingStrategy/BeadingStrategyFactory.cpp index c929bd5d9f..8b2d289449 100644 --- a/src/BeadingStrategy/BeadingStrategyFactory.cpp +++ b/src/BeadingStrategy/BeadingStrategyFactory.cpp @@ -41,22 +41,22 @@ BeadingStrategyPtr BeadingStrategyFactory::makeStrategy( wall_add_middle_threshold, inward_distributed_center_wall_count); spdlog::debug("Applying the Redistribute meta-strategy with outer-wall width = {}, inner-wall width = {}", preferred_bead_width_outer, preferred_bead_width_inner); - ret = make_unique(preferred_bead_width_outer, minimum_variable_line_ratio, move(ret)); + ret = make_unique(preferred_bead_width_outer, minimum_variable_line_ratio, std::move(ret)); if (print_thin_walls) { spdlog::debug("Applying the Widening Beading meta-strategy with minimum input width {} and minimum output width {}.", min_feature_size, min_bead_width); - ret = make_unique(move(ret), min_feature_size, min_bead_width); + ret = make_unique(std::move(ret), min_feature_size, min_bead_width); } if (outer_wall_offset > 0) { spdlog::debug("Applying the OuterWallOffset meta-strategy with offset = {}", outer_wall_offset); - ret = make_unique(outer_wall_offset, move(ret)); + ret = make_unique(outer_wall_offset, std::move(ret)); } // Apply the LimitedBeadingStrategy last, since that adds a 0-width marker wall which other beading strategies shouldn't touch. spdlog::debug("Applying the Limited Beading meta-strategy with maximum bead count = {}", max_bead_count); - ret = make_unique(max_bead_count, move(ret)); + ret = make_unique(max_bead_count, std::move(ret)); return ret; } } // namespace cura diff --git a/src/ConicalOverhang.cpp b/src/ConicalOverhang.cpp index 9918d8a782..699d9697d7 100644 --- a/src/ConicalOverhang.cpp +++ b/src/ConicalOverhang.cpp @@ -4,6 +4,8 @@ #include "ConicalOverhang.h" +#include "geometry/Polygon.h" +#include "geometry/SingleShape.h" #include "mesh.h" #include "settings/types/Angle.h" //To process the overhang angle. #include "settings/types/LayerIndex.h" @@ -16,7 +18,7 @@ namespace cura void ConicalOverhang::apply(Slicer* slicer, const Mesh& mesh) { const AngleRadians angle = mesh.settings_.get("conical_overhang_angle"); - const double maxHoleArea = mesh.settings_.get("conical_overhang_hole_size"); + const double max_hole_area = mesh.settings_.get("conical_overhang_hole_size"); const double tan_angle = tan(angle); // the XY-component of the angle const coord_t layer_thickness = mesh.settings_.get("layer_height"); coord_t max_dist_from_lower_layer = std::llround(tan_angle * static_cast(layer_thickness)); // max dist which can be bridged @@ -29,41 +31,41 @@ void ConicalOverhang::apply(Slicer* slicer, const Mesh& mesh) { // magically nothing happens when max_dist_from_lower_layer == 0 // below magic code solves that constexpr coord_t safe_dist = 20; - Polygons diff = layer_above.polygons.difference(layer.polygons.offset(-safe_dist)); - layer.polygons = layer.polygons.unionPolygons(diff); - layer.polygons = layer.polygons.smooth(safe_dist); - layer.polygons = Simplify(safe_dist, safe_dist / 2, 0).polygon(layer.polygons); + Shape diff = layer_above.polygons_.difference(layer.polygons_.offset(-safe_dist)); + layer.polygons_ = layer.polygons_.unionPolygons(diff); + layer.polygons_ = layer.polygons_.smooth(safe_dist); + layer.polygons_ = Simplify(safe_dist, safe_dist / 2, 0).polygon(layer.polygons_); // somehow layer.polygons get really jagged lines with a lot of vertices // without the above steps slicing goes really slow } else { // Get the current layer and split it into parts - std::vector layerParts = layer.polygons.splitIntoParts(); + std::vector layer_parts = layer.polygons_.splitIntoParts(); // Get a copy of the layer above to prune away before we shrink it - Polygons above = layer_above.polygons; + Shape above = layer_above.polygons_; // Now go through all the holes in the current layer and check if they intersect anything in the layer above // If not, then they're the top of a hole and should be cut from the layer above before the union - for (unsigned int part = 0; part < layerParts.size(); part++) + for (unsigned int part = 0; part < layer_parts.size(); part++) { - if (layerParts[part].size() > 1) // first poly is the outer contour, 1..n are the holes + if (layer_parts[part].size() > 1) // first poly is the outer contour, 1..n are the holes { - for (unsigned int hole_nr = 1; hole_nr < layerParts[part].size(); ++hole_nr) + for (unsigned int hole_nr = 1; hole_nr < layer_parts[part].size(); ++hole_nr) { - Polygons holePoly; - holePoly.add(layerParts[part][hole_nr]); - if (maxHoleArea > 0.0 && INT2MM2(std::abs(holePoly.area())) < maxHoleArea) + Shape hole_poly; + hole_poly.push_back(layer_parts[part][hole_nr]); + if (max_hole_area > 0.0 && INT2MM2(std::abs(hole_poly.area())) < max_hole_area) { - Polygons holeWithAbove = holePoly.intersection(above); - if (! holeWithAbove.empty()) + Shape hole_with_above = hole_poly.intersection(above); + if (! hole_with_above.empty()) { // The hole had some intersection with the above layer, check if it's a complete overlap - Polygons holeDifference = holePoly.xorPolygons(holeWithAbove); - if (holeDifference.empty()) + Shape hole_difference = hole_poly.xorPolygons(hole_with_above); + if (hole_difference.empty()) { // The hole was returned unchanged, so the layer above must completely cover it. Remove the hole from the layer above. - above = above.difference(holePoly); + above = above.difference(hole_poly); } } } @@ -71,7 +73,7 @@ void ConicalOverhang::apply(Slicer* slicer, const Mesh& mesh) } } // And now union with offset of the resulting above layer - layer.polygons = layer.polygons.unionPolygons(above.offset(-max_dist_from_lower_layer)); + layer.polygons_ = layer.polygons_.unionPolygons(above.offset(-max_dist_from_lower_layer)); } } } diff --git a/src/FffGcodeWriter.cpp b/src/FffGcodeWriter.cpp index 040b7d8376..5154971d39 100644 --- a/src/FffGcodeWriter.cpp +++ b/src/FffGcodeWriter.cpp @@ -7,14 +7,12 @@ #include // numeric_limits #include #include +#include #include #include #include #include -#include -#include -#include #include #include "Application.h" @@ -27,6 +25,9 @@ #include "WallToolPaths.h" #include "bridge.h" #include "communication/Communication.h" //To send layer view data. +#include "geometry/LinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/PointMatrix.h" #include "infill.h" #include "progress/Progress.h" #include "raft.h" @@ -35,6 +36,7 @@ #include "utils/linearAlg2D.h" #include "utils/math.h" #include "utils/orderOptimizer.h" +#include "utils/polygonUtils.h" namespace cura { @@ -224,10 +226,10 @@ unsigned int FffGcodeWriter::findSpiralizedLayerSeamVertexIndex(const SliceDataS // note that the code below doesn't assume that last_layer_nr is one less than layer_nr but the print is going // to come out pretty weird if that isn't true as it implies that there are empty layers - ConstPolygonRef last_wall = (*storage.spiralize_wall_outlines[last_layer_nr])[0]; + const Polygon& last_wall = (*storage.spiralize_wall_outlines[last_layer_nr])[0]; // Even though this is just one (contiguous) part, the spiralize wall may still be multiple parts if the part is somewhere thinner than 1 line width. // This case is so rare that we don't bother with finding the best polygon to start with. Just start with the first polygon (`spiral_wall[0]`). - ConstPolygonRef wall = layer.parts[0].spiral_wall[0]; + const Polygon& wall = layer.parts[0].spiral_wall[0]; const size_t n_points = wall.size(); const Point2LL last_wall_seam_vertex = last_wall[storage.spiralize_seam_vertex_indices[last_layer_nr]]; @@ -587,7 +589,7 @@ void FffGcodeWriter::processRaft(const SliceDataStorage& storage) constexpr bool retract_before_outer_wall = false; constexpr coord_t wipe_dist = 0; - Polygons raft_polygons; + Shape raft_polygons; std::optional last_planned_position = std::optional(); unsigned int current_extruder_nr = base_extruder_nr; @@ -616,7 +618,7 @@ void FffGcodeWriter::processRaft(const SliceDataStorage& storage) Application::getInstance().communication_->sendLayerComplete(layer_nr, z, layer_height); - Polygons raft_lines; + OpenLinesSet raft_lines; AngleDegrees fill_angle = (num_surface_layers + num_interface_layers) % 2 ? 45 : 135; // 90 degrees rotated from the interface layer. constexpr bool zig_zaggify_infill = false; constexpr bool connect_polygons = true; // causes less jerks, so better adhesion @@ -638,14 +640,14 @@ void FffGcodeWriter::processRaft(const SliceDataStorage& storage) struct ParameterizedRaftPath { coord_t line_spacing; - Polygons outline; + Shape outline; }; std::vector raft_outline_paths; - raft_outline_paths.emplace_back(ParameterizedRaftPath{ line_spacing, storage.raftBaseOutline }); + raft_outline_paths.emplace_back(ParameterizedRaftPath{ line_spacing, storage.raft_base_outline }); if (storage.primeTower.enabled_) { - const Polygons& raft_outline_prime_tower = storage.primeTower.getOuterPoly(layer_nr); + const Shape& raft_outline_prime_tower = storage.primeTower.getOuterPoly(layer_nr); if (line_spacing_prime_tower == line_spacing) { // Base layer is shared with prime tower base @@ -797,13 +799,13 @@ void FffGcodeWriter::processRaft(const SliceDataStorage& storage) Application::getInstance().communication_->sendLayerComplete(layer_nr, z, interface_layer_height); - Polygons raft_outline_path; + Shape raft_outline_path; const coord_t small_offset = gcode_layer.configs_storage_.raft_interface_config.getLineWidth() / 2; // Do this manually because of micron-movement created in corners when insetting a polygon that was offset with round joint type. - raft_outline_path = storage.raftInterfaceOutline.offset(-small_offset); + raft_outline_path = storage.raft_interface_outline.offset(-small_offset); raft_outline_path = Simplify(interface_settings).polygon(raft_outline_path); // Remove those micron-movements. const coord_t infill_outline_width = gcode_layer.configs_storage_.raft_interface_config.getLineWidth(); - Polygons raft_lines; + OpenLinesSet raft_lines; AngleDegrees fill_angle = (num_surface_layers + num_interface_layers - raft_interface_layer) % 2 ? 45 : 135; // 90 degrees rotated from the first top layer. constexpr bool zig_zaggify_infill = true; constexpr bool connect_polygons = true; // why not? @@ -968,13 +970,13 @@ void FffGcodeWriter::processRaft(const SliceDataStorage& storage) } Application::getInstance().communication_->sendLayerComplete(layer_nr, z, surface_layer_height); - Polygons raft_outline_path; + Shape raft_outline_path; const coord_t small_offset = gcode_layer.configs_storage_.raft_interface_config.getLineWidth() / 2; // Do this manually because of micron-movement created in corners when insetting a polygon that was offset with round joint type. - raft_outline_path = storage.raftSurfaceOutline.offset(-small_offset); + raft_outline_path = storage.raft_surface_outline.offset(-small_offset); raft_outline_path = Simplify(interface_settings).polygon(raft_outline_path); // Remove those micron-movements. const coord_t infill_outline_width = gcode_layer.configs_storage_.raft_surface_config.getLineWidth(); - Polygons raft_lines; + OpenLinesSet raft_lines; AngleDegrees fill_angle = (num_surface_layers - raft_surface_layer) % 2 ? 45 : 135; // Alternate between -45 and +45 degrees, ending up 90 degrees rotated from the default skin angle. constexpr bool zig_zaggify_infill = true; @@ -998,7 +1000,7 @@ void FffGcodeWriter::processRaft(const SliceDataStorage& storage) raft_outline_path = raft_outline_path.difference(storage.primeTower.getOuterPoly(layer_nr)); } - for (const Polygons raft_island : raft_outline_path.splitIntoParts()) + for (const Shape& raft_island : raft_outline_path.splitIntoParts()) { Infill infill_comp( EFillMethod::ZIG_ZAG, @@ -1338,14 +1340,13 @@ void FffGcodeWriter::processSkirtBrim(const SliceDataStorage& storage, LayerPlan struct BrimLineReference { const size_t inset_idx; - ConstPolygonPointer poly; + const Polyline* poly; }; size_t total_line_count = 0; - for (const SkirtBrimLine& line : storage.skirt_brim[extruder_nr]) + for (const MixedLinesSet& lines : storage.skirt_brim[extruder_nr]) { - total_line_count += line.closed_polygons.size(); - total_line_count += line.open_polylines.size(); + total_line_count += lines.size(); // For layer_nr != 0 add only the innermost brim line (which is only the case if skirt_height > 1) if (layer_nr != 0) @@ -1353,9 +1354,8 @@ void FffGcodeWriter::processSkirtBrim(const SliceDataStorage& storage, LayerPlan break; } } - Polygons all_brim_lines; - + MixedLinesSet all_brim_lines; all_brim_lines.reserve(total_line_count); const coord_t line_w = train.settings_.get("skirt_brim_line_width") * train.settings_.get("initial_layer_line_width_factor"); @@ -1365,27 +1365,15 @@ void FffGcodeWriter::processSkirtBrim(const SliceDataStorage& storage, LayerPlan for (size_t inset_idx = 0; inset_idx < storage.skirt_brim[extruder_nr].size(); inset_idx++) { - const auto& offset = storage.skirt_brim[extruder_nr][inset_idx]; - const auto closed_polygons_open_polylines = { offset.closed_polygons, offset.open_polylines }; - const auto closed_open = { true, false }; - for (const auto [polygon, closed] : ranges::views::zip(closed_polygons_open_polylines, closed_open)) + const MixedLinesSet& offset = storage.skirt_brim[extruder_nr][inset_idx]; + for (const PolylinePtr& line : offset) { - for (ConstPolygonRef line : polygon) + if (line->segmentsCount() > 0) { - if (line.size() <= 1) - { - continue; - } - all_brim_lines.emplace_back(line); - if (closed) - { - // add closing segment - all_brim_lines.back().add(line.front()); - } - ConstPolygonPointer pp(all_brim_lines.back()); - for (Point2LL p : line) + all_brim_lines.push_back(line); + for (const Point2LL& p : *line) { - grid.insert(p, BrimLineReference{ inset_idx, pp }); + grid.insert(p, BrimLineReference{ inset_idx, line.get() }); } } } @@ -1398,7 +1386,7 @@ void FffGcodeWriter::processSkirtBrim(const SliceDataStorage& storage, LayerPlan } const auto smart_brim_ordering = train.settings_.get("brim_smart_ordering") && train.settings_.get("adhesion_type") == EPlatformAdhesion::BRIM; - std::unordered_multimap order_requirements; + std::unordered_multimap order_requirements; for (const std::pair>& p : grid) { const BrimLineReference& here = p.second.val; @@ -1464,7 +1452,7 @@ void FffGcodeWriter::processSkirtBrim(const SliceDataStorage& storage, LayerPlan start_close_to, fan_speed, reverse_print_direction, - layer_nr == 0 ? order_requirements : PathOrderOptimizer::no_order_requirements_); + layer_nr == 0 ? order_requirements : PathOrderOptimizer::no_order_requirements_); } @@ -1475,10 +1463,8 @@ void FffGcodeWriter::processSkirtBrim(const SliceDataStorage& storage, LayerPlan if ((layer_nr == 0) && (extruder_nr == mesh_group_settings.get("support_extruder_nr_layer_0").extruder_nr_)) { total_line_count += storage.support_brim.size(); - Polygons support_brim_lines = storage.support_brim; - support_brim_lines.toPolylines(); gcode_layer.addLinesByOptimizer( - support_brim_lines, + storage.support_brim, gcode_layer.configs_storage_.skirt_brim_config_per_extruder[extruder_nr], SpaceFillType::PolyLines, enable_travel_optimization, @@ -1498,9 +1484,9 @@ void FffGcodeWriter::processOozeShield(const SliceDataStorage& storage, LayerPla { return; // ooze shield already generated by brim } - if (storage.oozeShield.size() > 0 && layer_nr < storage.oozeShield.size()) + if (storage.ooze_shield.size() > 0 && layer_nr < storage.ooze_shield.size()) { - gcode_layer.addPolygonsByOptimizer(storage.oozeShield[layer_nr], gcode_layer.configs_storage_.skirt_brim_config_per_extruder[0]); + gcode_layer.addPolygonsByOptimizer(storage.ooze_shield[layer_nr], gcode_layer.configs_storage_.skirt_brim_config_per_extruder[0]); } } @@ -1551,12 +1537,13 @@ void FffGcodeWriter::calculateExtruderOrderPerLayer(const SliceDataStorage& stor } size_t extruder_count = Application::getInstance().current_slice_->scene.extruders.size(); + const std::vector extruders_used = storage.getExtrudersUsed(); const Settings& mesh_group_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; PrimeTowerMethod prime_tower_mode = mesh_group_settings.get("prime_tower_mode"); for (LayerIndex layer_nr = -Raft::getTotalExtraLayers(); layer_nr < static_cast(storage.print_layer_count); layer_nr++) { std::vector>& extruder_order_per_layer_here = (layer_nr < 0) ? extruder_order_per_layer_negative_layers : extruder_order_per_layer; - std::vector extruder_order = getUsedExtrudersOnLayer(storage, last_extruder, layer_nr); + std::vector extruder_order = getUsedExtrudersOnLayer(storage, last_extruder, layer_nr, extruders_used); extruder_order_per_layer_here.push_back(extruder_order); if (! extruder_order.empty()) @@ -1581,10 +1568,14 @@ void FffGcodeWriter::calculatePrimeLayerPerExtruder(const SliceDataStorage& stor } } -std::vector FffGcodeWriter::getUsedExtrudersOnLayer(const SliceDataStorage& storage, const size_t start_extruder, const LayerIndex& layer_nr) const +std::vector FffGcodeWriter::getUsedExtrudersOnLayer( + const SliceDataStorage& storage, + const size_t start_extruder, + const LayerIndex& layer_nr, + const std::vector& global_extruders_used) const { const Settings& mesh_group_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; - size_t extruder_count = Application::getInstance().current_slice_->scene.extruders.size(); + size_t extruder_count = global_extruders_used.size(); assert(static_cast(extruder_count) > 0); std::vector ret; std::vector extruder_is_used_on_this_layer = storage.getExtrudersUsed(layer_nr); @@ -1609,7 +1600,7 @@ std::vector FffGcodeWriter::getUsedExtrudersOnLayer(const SliceData ordered_extruders.push_back(start_extruder); for (size_t extruder_nr = 0; extruder_nr < extruder_count; extruder_nr++) { - if (extruder_nr != start_extruder) + if (extruder_nr != start_extruder && global_extruders_used[extruder_nr]) { ordered_extruders.push_back(extruder_nr); } @@ -1706,12 +1697,12 @@ void FffGcodeWriter::addMeshLayerToGCode_meshSurfaceMode(const SliceMeshStorage& const SliceLayer* layer = &mesh.layers[gcode_layer.getLayerNr()]; - Polygons polygons; + Shape polygons; for (const SliceLayerPart& part : layer->parts) { if (! part.outline.empty()) { - polygons.add(part.outline); + polygons.push_back(part.outline); } } @@ -1732,7 +1723,7 @@ void FffGcodeWriter::addMeshOpenPolyLinesToGCode(const SliceMeshStorage& mesh, c { const SliceLayer* layer = &mesh.layers[gcode_layer.getLayerNr()]; - gcode_layer.addLinesByOptimizer(layer->openPolyLines, mesh_config.inset0_config, SpaceFillType::PolyLines); + gcode_layer.addLinesByOptimizer(layer->open_polylines, mesh_config.inset0_config, SpaceFillType::PolyLines); } void FffGcodeWriter::addMeshLayerToGCode( @@ -1895,8 +1886,8 @@ bool FffGcodeWriter::processMultiLayerInfill( const bool zig_zaggify_infill = mesh.settings.get("zig_zaggify_infill") || infill_pattern == EFillMethod::ZIG_ZAG; const bool connect_polygons = mesh.settings.get("connect_infill_polygons"); const size_t infill_multiplier = mesh.settings.get("infill_multiplier"); - Polygons infill_polygons; - Polygons infill_lines; + Shape infill_polygons; + OpenLinesSet infill_lines; std::vector infill_paths = part.infill_wall_toolpaths; for (size_t density_idx = part.infill_area_per_combine_per_density.size() - 1; (int)density_idx >= 0; density_idx--) { // combine different density infill areas (for gradual infill) @@ -2015,9 +2006,9 @@ bool FffGcodeWriter::processSingleLayerInfill( const coord_t infill_line_width = mesh_config.infill_config[0].getLineWidth(); // Combine the 1 layer thick infill with the top/bottom skin and print that as one thing. - Polygons infill_polygons; + Shape infill_polygons; std::vector> wall_tool_paths; // All wall toolpaths binned by inset_idx (inner) and by density_idx (outer) - Polygons infill_lines; + OpenLinesSet infill_lines; const auto pattern = mesh.settings.get("infill_pattern"); const bool zig_zaggify_infill = mesh.settings.get("zig_zaggify_infill") || pattern == EFillMethod::ZIG_ZAG; @@ -2047,12 +2038,12 @@ bool FffGcodeWriter::processSingleLayerInfill( return -static_cast(line_count) * line_width; }; - Polygons sparse_in_outline = part.infill_area_per_combine_per_density[last_idx][0]; + Shape sparse_in_outline = part.infill_area_per_combine_per_density[last_idx][0]; // if infill walls are required below the boundaries of skin regions above, partition the infill along the // boundary edge - Polygons infill_below_skin; - Polygons infill_not_below_skin; + Shape infill_below_skin; + Shape infill_not_below_skin; const bool hasSkinEdgeSupport = partitionInfillBySkinAbove(infill_below_skin, infill_not_below_skin, gcode_layer, mesh, part, infill_line_width); const auto pocket_size = mesh.settings.get("cross_infill_pocket_size"); @@ -2070,8 +2061,8 @@ bool FffGcodeWriter::processSingleLayerInfill( continue; } - Polygons infill_lines_here; - Polygons infill_polygons_here; + OpenLinesSet infill_lines_here; + Shape infill_polygons_here; // the highest density infill combines with the next to create a grid with density_factor 1 int infill_line_distance_here = infill_line_distance << (density_idx + 1); @@ -2112,7 +2103,7 @@ bool FffGcodeWriter::processSingleLayerInfill( infill_line_distance_here /= 2; } - Polygons in_outline = part.infill_area_per_combine_per_density[density_idx][0]; + Shape in_outline = part.infill_area_per_combine_per_density[density_idx][0]; std::shared_ptr lightning_layer; if (mesh.lightning_generator) @@ -2166,10 +2157,10 @@ bool FffGcodeWriter::processSingleLayerInfill( if (density_idx < last_idx) { const coord_t cut_offset = get_cut_offset(zig_zaggify_infill, infill_line_width, min_skin_below_wall_count); - Polygons tool = infill_below_skin.offset(static_cast(cut_offset)); - infill_lines_here = tool.intersectionPolyLines(infill_lines_here); + Shape tool = infill_below_skin.offset(static_cast(cut_offset)); + infill_lines_here = tool.intersection(infill_lines_here); } - infill_lines.add(infill_lines_here); + infill_lines.push_back(infill_lines_here); // normal processing for the infill that isn't below skin in_outline = infill_not_below_skin; if (density_idx == last_idx) @@ -2178,7 +2169,7 @@ bool FffGcodeWriter::processSingleLayerInfill( } } - const coord_t circumference = in_outline.polygonLength(); + const coord_t circumference = in_outline.length(); // Originally an area of 0.4*0.4*2 (2 line width squares) was found to be a good threshold for removal. // However we found that this doesn't scale well with polygons with larger circumference (https://github.com/Ultimaker/Cura/issues/3992). // Given that the original test worked for approximately 2x2cm models, this scaling by circumference should make it work for any size. @@ -2231,11 +2222,11 @@ bool FffGcodeWriter::processSingleLayerInfill( if (density_idx < last_idx) { const coord_t cut_offset = get_cut_offset(zig_zaggify_infill, infill_line_width, wall_line_count); - Polygons tool = sparse_in_outline.offset(static_cast(cut_offset)); - infill_lines_here = tool.intersectionPolyLines(infill_lines_here); + Shape tool = sparse_in_outline.offset(static_cast(cut_offset)); + infill_lines_here = tool.intersection(infill_lines_here); } - infill_lines.add(infill_lines_here); - infill_polygons.add(infill_polygons_here); + infill_lines.push_back(infill_lines_here); + infill_polygons.push_back(infill_polygons_here); } wall_tool_paths.emplace_back(part.infill_wall_toolpaths); // The extra infill walls were generated separately. Add these too. @@ -2268,7 +2259,7 @@ bool FffGcodeWriter::processSingleLayerInfill( } else if (! infill_polygons.empty()) { - PolygonRef start_poly = infill_polygons[rand() % infill_polygons.size()]; + const Polygon& start_poly = infill_polygons[rand() % infill_polygons.size()]; near_start_location = start_poly[rand() % start_poly.size()]; } else // So walls_generated must be true. @@ -2351,8 +2342,8 @@ bool FffGcodeWriter::processSingleLayerInfill( } bool FffGcodeWriter::partitionInfillBySkinAbove( - Polygons& infill_below_skin, - Polygons& infill_not_below_skin, + Shape& infill_below_skin, + Shape& infill_not_below_skin, const LayerPlan& gcode_layer, const SliceMeshStorage& mesh, const SliceLayerPart& part, @@ -2360,7 +2351,7 @@ bool FffGcodeWriter::partitionInfillBySkinAbove( { constexpr coord_t tiny_infill_offset = 20; const auto skin_edge_support_layers = mesh.settings.get("skin_edge_support_layers"); - Polygons skin_above_combined; // skin regions on the layers above combined with small gaps between + Shape skin_above_combined; // skin regions on the layers above combined with small gaps between // working from the highest layer downwards, combine the regions of skin on all the layers // but don't let the regions merge together @@ -2375,12 +2366,12 @@ bool FffGcodeWriter::partitionInfillBySkinAbove( for (const SkinPart& skin_part : part_i.skin_parts) { // Limit considered areas to the ones that should have infill underneath at the current layer. - const Polygons relevant_outline = skin_part.outline.intersection(part.getOwnInfillArea()); + const Shape relevant_outline = skin_part.outline.intersection(part.getOwnInfillArea()); if (! skin_above_combined.empty()) { // does this skin part overlap with any of the skin parts on the layers above? - const Polygons overlap = skin_above_combined.intersection(relevant_outline); + const Shape overlap = skin_above_combined.intersection(relevant_outline); if (! overlap.empty()) { // yes, it overlaps, need to leave a gap between this skin part and the others @@ -2399,13 +2390,13 @@ bool FffGcodeWriter::partitionInfillBySkinAbove( // ------- -------------------------- ---------- // expand the overlap region slightly to make a small gap - const Polygons overlap_expanded = overlap.offset(tiny_infill_offset); + const Shape overlap_expanded = overlap.offset(tiny_infill_offset); // subtract the expanded overlap region from the regions accumulated from higher layers skin_above_combined = skin_above_combined.difference(overlap_expanded); // subtract the expanded overlap region from this skin part and add the remainder to the overlap region - skin_above_combined.add(relevant_outline.difference(overlap_expanded)); + skin_above_combined.push_back(relevant_outline.difference(overlap_expanded)); // and add the overlap area as well - skin_above_combined.add(overlap); + skin_above_combined.push_back(overlap); } else // this layer is the 1st layer above the layer whose infill we're printing { @@ -2425,17 +2416,17 @@ bool FffGcodeWriter::partitionInfillBySkinAbove( // ------- ------------------------------------- skin_above_combined = skin_above_combined.difference(relevant_outline.offset(tiny_infill_offset)); - skin_above_combined.add(relevant_outline); + skin_above_combined.push_back(relevant_outline); } } else // no overlap { - skin_above_combined.add(relevant_outline); + skin_above_combined.push_back(relevant_outline); } } else // this is the first skin region we have looked at { - skin_above_combined.add(relevant_outline); + skin_above_combined.push_back(relevant_outline); } } } @@ -2457,7 +2448,7 @@ bool FffGcodeWriter::partitionInfillBySkinAbove( // need to take skin/infill overlap that was added in SkinInfillAreaComputation::generateInfill() into account const coord_t infill_skin_overlap = mesh.settings.get((part.wall_toolpaths.size() > 1) ? "wall_line_width_x" : "wall_line_width_0") / 2; - const Polygons infill_below_skin_overlap = infill_below_skin.offset(-(infill_skin_overlap + tiny_infill_offset)); + const Shape infill_below_skin_overlap = infill_below_skin.offset(-(infill_skin_overlap + tiny_infill_offset)); return ! infill_below_skin_overlap.empty() && ! infill_not_below_skin.empty(); } @@ -2521,7 +2512,7 @@ void FffGcodeWriter::processSpiralizedWall( // wall doesn't have usable outline return; } - const ClipperLib::Path* last_wall_outline = &*part.spiral_wall[0]; // default to current wall outline + const Polygon* last_wall_outline = &(part.spiral_wall[0]); // default to current wall outline int last_seam_vertex_idx = -1; // last layer seam vertex index int layer_nr = gcode_layer.getLayerNr(); if (layer_nr > 0) @@ -2529,7 +2520,7 @@ void FffGcodeWriter::processSpiralizedWall( if (storage.spiralize_wall_outlines[layer_nr - 1] != nullptr) { // use the wall outline from the previous layer - last_wall_outline = &*(*storage.spiralize_wall_outlines[layer_nr - 1])[0]; + last_wall_outline = &(storage.spiralize_wall_outlines[layer_nr - 1]->front()); // and the seam vertex index pre-computed for that layer last_seam_vertex_idx = storage.spiralize_seam_vertex_indices[layer_nr - 1]; } @@ -2538,10 +2529,9 @@ void FffGcodeWriter::processSpiralizedWall( const bool is_top_layer = ((size_t)layer_nr == (storage.spiralize_wall_outlines.size() - 1) || storage.spiralize_wall_outlines[layer_nr + 1] == nullptr); const int seam_vertex_idx = storage.spiralize_seam_vertex_indices[layer_nr]; // use pre-computed seam vertex index for current layer // output a wall slice that is interpolated between the last and current walls - for (const ConstPolygonRef& wall_outline : part.spiral_wall) + for (const Polygon& wall_outline : part.spiral_wall) { - gcode_layer - .spiralizeWallSlice(mesh_config.inset0_config, wall_outline, ConstPolygonRef(*last_wall_outline), seam_vertex_idx, last_seam_vertex_idx, is_top_layer, is_bottom_layer); + gcode_layer.spiralizeWallSlice(mesh_config.inset0_config, wall_outline, *last_wall_outline, seam_vertex_idx, last_seam_vertex_idx, is_top_layer, is_bottom_layer); } } @@ -2585,7 +2575,7 @@ bool FffGcodeWriter::processInsets( added_something = true; gcode_layer.setIsInside(true); // going to print stuff inside print object // start this first wall at the same vertex the spiral starts - const ConstPolygonRef spiral_inset = part.spiral_wall[0]; + const Polygon& spiral_inset = part.spiral_wall[0]; const size_t spiral_start_vertex = storage.spiralize_seam_vertex_indices[initial_bottom_layers]; if (spiral_start_vertex < spiral_inset.size()) { @@ -2600,7 +2590,7 @@ bool FffGcodeWriter::processInsets( { // accumulate the outlines of all of the parts that are on the layer below - Polygons outlines_below; + Shape outlines_below; AABB boundaryBox(part.outline); for (const std::shared_ptr& mesh_ptr : storage.meshes) { @@ -2611,7 +2601,7 @@ bool FffGcodeWriter::processInsets( { if (boundaryBox.hit(prevLayerPart.boundaryBox)) { - outlines_below.add(prevLayerPart.outline); + outlines_below.push_back(prevLayerPart.outline); } } } @@ -2637,7 +2627,7 @@ bool FffGcodeWriter::processInsets( AABB support_roof_bb(support_layer.support_roof); if (boundaryBox.hit(support_roof_bb)) { - outlines_below.add(support_layer.support_roof); + outlines_below.push_back(support_layer.support_roof); } } else @@ -2647,7 +2637,7 @@ bool FffGcodeWriter::processInsets( AABB support_part_bb(support_part.getInfillArea()); if (boundaryBox.hit(support_part_bb)) { - outlines_below.add(support_part.getInfillArea()); + outlines_below.push_back(support_part.getInfillArea()); } } } @@ -2670,7 +2660,7 @@ bool FffGcodeWriter::processInsets( // subtract the outlines of the parts below this part to give the shapes of the unsupported regions and then // shrink those shapes so that any that are narrower than two times max_air_gap will be removed - Polygons compressed_air(part.outline.difference(outlines_below).offset(-max_air_gap)); + Shape compressed_air(part.outline.difference(outlines_below).offset(-max_air_gap)); // now expand the air regions by the same amount as they were shrunk plus half the outer wall line width // which is required because when the walls are being generated, the vertices do not fall on the part's outline @@ -2681,14 +2671,14 @@ bool FffGcodeWriter::processInsets( else { // clear to disable use of bridging settings - gcode_layer.setBridgeWallMask(Polygons()); + gcode_layer.setBridgeWallMask(Shape()); } const AngleDegrees overhang_angle = mesh.settings.get("wall_overhang_angle"); if (overhang_angle >= 90) { // clear to disable overhang detection - gcode_layer.setOverhangMask(Polygons()); + gcode_layer.setOverhangMask(Shape()); } else { @@ -2697,11 +2687,11 @@ bool FffGcodeWriter::processInsets( // expanded to take into account the overhang angle, the greater the overhang angle, the larger the supported area is // considered to be const coord_t overhang_width = layer_height * std::tan(overhang_angle / (180 / std::numbers::pi)); - Polygons overhang_region = part.outline.offset(-half_outer_wall_width).difference(outlines_below.offset(10 + overhang_width - half_outer_wall_width)).offset(10); + Shape overhang_region = part.outline.offset(-half_outer_wall_width).difference(outlines_below.offset(10 + overhang_width - half_outer_wall_width)).offset(10); gcode_layer.setOverhangMask(overhang_region); } - const auto roofing_mask = [&]() -> Polygons + const auto roofing_mask_fn = [&]() -> Shape { const size_t roofing_layer_count = std::min(mesh.settings.get("roofing_layer_count"), mesh.settings.get("top_layers")); @@ -2723,16 +2713,16 @@ bool FffGcodeWriter::processInsets( return roofing_mask; }(); - gcode_layer.setRoofingMask(roofing_mask); + gcode_layer.setRoofingMask(roofing_mask_fn); } else { // clear to disable use of bridging settings - gcode_layer.setBridgeWallMask(Polygons()); + gcode_layer.setBridgeWallMask(Shape()); // clear to disable overhang detection - gcode_layer.setOverhangMask(Polygons()); + gcode_layer.setOverhangMask(Shape()); // clear to disable use of roofing settings - gcode_layer.setRoofingMask(Polygons()); + gcode_layer.setRoofingMask(Shape()); } if (spiralize && extruder_nr == mesh.settings.get("wall_0_extruder_nr").extruder_nr_ && ! part.spiral_wall.empty()) @@ -2786,7 +2776,7 @@ bool FffGcodeWriter::processInsets( return added_something; } -std::optional FffGcodeWriter::getSeamAvoidingLocation(const Polygons& filling_part, int filling_angle, Point2LL last_position) const +std::optional FffGcodeWriter::getSeamAvoidingLocation(const Shape& filling_part, int filling_angle, Point2LL last_position) const { if (filling_part.empty()) { @@ -2974,7 +2964,7 @@ void FffGcodeWriter::processTopBottom( support_layer = &storage.support.supportLayers[support_layer_nr - (bridge_layer - 1)]; } - Polygons supported_skin_part_regions; + Shape supported_skin_part_regions; const double angle = bridgeAngle(mesh.settings, skin_part.skin_fill, storage, layer_nr, bridge_layer, support_layer, supported_skin_part_regions); @@ -3099,7 +3089,7 @@ void FffGcodeWriter::processSkinPrintFeature( LayerPlan& gcode_layer, const SliceMeshStorage& mesh, const size_t extruder_nr, - const Polygons& area, + const Shape& area, const GCodePathConfig& config, EFillMethod pattern, const AngleDegrees skin_angle, @@ -3109,8 +3099,8 @@ void FffGcodeWriter::processSkinPrintFeature( bool& added_something, double fan_speed) const { - Polygons skin_polygons; - Polygons skin_lines; + Shape skin_polygons; + OpenLinesSet skin_lines; std::vector skin_paths; constexpr int infill_multiplier = 1; @@ -3167,7 +3157,7 @@ void FffGcodeWriter::processSkinPrintFeature( nullptr, nullptr, nullptr, - small_areas_on_surface ? Polygons() : exposed_to_air); + small_areas_on_surface ? Shape() : exposed_to_air); // add paths if (! skin_polygons.empty() || ! skin_lines.empty() || ! skin_paths.empty()) @@ -3451,7 +3441,26 @@ bool FffGcodeWriter::processSupportInfill(const SliceDataStorage& storage, Layer const GCodePathConfig& config = configs[0]; constexpr bool retract_before_outer_wall = false; constexpr coord_t wipe_dist = 0; - const ZSeamConfig z_seam_config(EZSeamType::SHORTEST, gcode_layer.getLastPlannedPositionOrStartingPosition(), EZSeamCornerPrefType::Z_SEAM_CORNER_PREF_NONE, false); + ZSeamConfig z_seam_config + = ZSeamConfig(EZSeamType::SHORTEST, gcode_layer.getLastPlannedPositionOrStartingPosition(), EZSeamCornerPrefType::Z_SEAM_CORNER_PREF_NONE, false); + Shape disallowed_area_for_seams{}; + if (infill_extruder.settings_.get("support_z_seam_away_from_model")) + { + for (std::shared_ptr mesh_ptr : storage.meshes) + { + auto& mesh = *mesh_ptr; + for (auto& part : mesh.layers[gcode_layer.getLayerNr()].parts) + { + disallowed_area_for_seams.push_back(part.print_outline); + } + } + if (! disallowed_area_for_seams.empty()) + { + coord_t min_distance = infill_extruder.settings_.get("support_z_seam_min_distance"); + disallowed_area_for_seams = disallowed_area_for_seams.offset(min_distance, ClipperLib::jtRound); + } + } + InsetOrderOptimizer wall_orderer( *this, storage, @@ -3470,7 +3479,8 @@ bool FffGcodeWriter::processSupportInfill(const SliceDataStorage& storage, Layer extruder_nr, extruder_nr, z_seam_config, - wall_toolpaths); + wall_toolpaths, + disallowed_area_for_seams); added_something |= wall_orderer.addToLayer(); } @@ -3483,9 +3493,9 @@ bool FffGcodeWriter::processSupportInfill(const SliceDataStorage& storage, Layer { const coord_t support_line_width = default_support_line_width * (combine_idx + 1); - Polygons support_polygons; + Shape support_polygons; std::vector wall_toolpaths_here; - Polygons support_lines; + OpenLinesSet support_lines; const size_t max_density_idx = part.infill_area_per_combine_per_density_.size() - 1; for (size_t density_idx = max_density_idx; (density_idx + 1) > 0; --density_idx) { @@ -3504,7 +3514,7 @@ bool FffGcodeWriter::processSupportInfill(const SliceDataStorage& storage, Layer { support_line_distance_here /= 2; } - const Polygons& area = Simplify(infill_extruder.settings_).polygon(part.infill_area_per_combine_per_density_[density_idx][combine_idx]); + const Shape& area = Simplify(infill_extruder.settings_).polygon(part.infill_area_per_combine_per_density_[density_idx][combine_idx]); constexpr size_t wall_count = 0; // Walls are generated somewhere else, so their layers aren't vertically combined. const coord_t small_area_width = 0; @@ -3555,7 +3565,7 @@ bool FffGcodeWriter::processSupportInfill(const SliceDataStorage& storage, Layer // to the start of the support does not go through the model we have to tell the slicer what the current location of the nozzle is // by adding a travel move to the end vertex of the last spiral. Of course, if the slicer could track the final location on the previous // layer then this wouldn't be necessary but that's not done due to the multi-threading. - const Polygons* last_wall_outline = storage.spiralize_wall_outlines[layer_nr - 1]; + const Shape* last_wall_outline = storage.spiralize_wall_outlines[layer_nr - 1]; if (last_wall_outline != nullptr) { gcode_layer.addTravel((*last_wall_outline)[0][storage.spiralize_seam_vertex_indices[layer_nr - 1]]); @@ -3658,11 +3668,8 @@ bool FffGcodeWriter::processSupportInfill(const SliceDataStorage& storage, Layer } -bool FffGcodeWriter::addSupportRoofsToGCode( - const SliceDataStorage& storage, - const Polygons& support_roof_outlines, - const GCodePathConfig& current_roof_config, - LayerPlan& gcode_layer) const +bool FffGcodeWriter::addSupportRoofsToGCode(const SliceDataStorage& storage, const Shape& support_roof_outlines, const GCodePathConfig& current_roof_config, LayerPlan& gcode_layer) + const { const SupportLayer& support_layer = storage.support.supportLayers[std::max(LayerIndex{ 0 }, gcode_layer.getLayerNr())]; @@ -3708,8 +3715,8 @@ bool FffGcodeWriter::addSupportRoofsToGCode( support_roof_line_distance *= roof_extruder.settings_.get("initial_layer_line_width_factor"); } - Polygons infill_outline = support_roof_outlines; - Polygons wall; + Shape infill_outline = support_roof_outlines; + Shape wall; // make sure there is a wall if this is on the first layer if (gcode_layer.getLayerNr() == 0) { @@ -3742,9 +3749,9 @@ bool FffGcodeWriter::addSupportRoofsToGCode( skip_some_zags, zag_skip_count, pocket_size); - Polygons roof_polygons; + Shape roof_polygons; std::vector roof_paths; - Polygons roof_lines; + OpenLinesSet roof_lines; roof_computation.generate(roof_paths, roof_polygons, roof_lines, roof_extruder.settings_, gcode_layer.getLayerNr(), SectionType::SUPPORT); if ((gcode_layer.getLayerNr() == 0 && wall.empty()) || (gcode_layer.getLayerNr() > 0 && roof_paths.empty() && roof_polygons.empty() && roof_lines.empty())) { @@ -3859,9 +3866,9 @@ bool FffGcodeWriter::addSupportBottomsToGCode(const SliceDataStorage& storage, L skip_some_zags, zag_skip_count, pocket_size); - Polygons bottom_polygons; + Shape bottom_polygons; std::vector bottom_paths; - Polygons bottom_lines; + OpenLinesSet bottom_lines; bottom_computation.generate(bottom_paths, bottom_polygons, bottom_lines, bottom_extruder.settings_, gcode_layer.getLayerNr(), SectionType::SUPPORT); if (bottom_paths.empty() && bottom_polygons.empty() && bottom_lines.empty()) { diff --git a/src/FffPolygonGenerator.cpp b/src/FffPolygonGenerator.cpp index b0d5ff5ff2..e7baf8e0a2 100644 --- a/src/FffPolygonGenerator.cpp +++ b/src/FffPolygonGenerator.cpp @@ -49,6 +49,7 @@ #include "utils/ThreadPool.h" #include "utils/gettime.h" #include "utils/math.h" +#include "geometry/OpenPolyline.h" #include "utils/Simplify.h" // clang-format on @@ -404,6 +405,8 @@ void FffPolygonGenerator::slices2polygons(SliceDataStorage& storage, TimeKeeper& Progress::messageProgressStage(Progress::Stage::SUPPORT, &time_keeper); + storage.primeTower.initializeExtruders(storage.getExtrudersUsed()); + AreaSupport::generateOverhangAreas(storage); AreaSupport::generateSupportAreas(storage); TreeSupport tree_support_generator(storage); @@ -566,17 +569,16 @@ void FffPolygonGenerator::processInfillMesh(SliceDataStorage& storage, const siz // they have to be polylines, because they might break up further when doing the cutting for (SliceLayerPart& part : layer.parts) { - for (const PolygonRef& poly : part.outline) + for (const Polygon& poly : part.outline) { - layer.openPolyLines.add(poly); - layer.openPolyLines.back().add(layer.openPolyLines.back()[0]); // add the segment which closes the polygon + layer.open_polylines.push_back(poly.toPseudoOpenPolyline()); } } layer.parts.clear(); } - std::vector new_parts; - Polygons new_polylines; + std::vector new_parts; + OpenLinesSet new_polylines; for (const size_t other_mesh_idx : mesh_order) { // limit the infill mesh's outline to within the infill of all meshes with lower order @@ -602,17 +604,17 @@ void FffPolygonGenerator::processInfillMesh(SliceDataStorage& storage, const siz { // early out continue; } - Polygons new_outline = part.outline.intersection(other_part.getOwnInfillArea()); + Shape new_outline = part.outline.intersection(other_part.getOwnInfillArea()); if (new_outline.size() == 1) { // we don't have to call splitIntoParts, because a single polygon can only be a single part - PolygonsPart outline_part_here; - outline_part_here.add(new_outline[0]); + SingleShape outline_part_here; + outline_part_here.push_back(new_outline[0]); new_parts.push_back(outline_part_here); } else if (new_outline.size() > 1) { // we don't know whether it's a multitude of parts because of newly introduced holes, or because the polygon has been split up - std::vector new_parts_here = new_outline.splitIntoParts(); - for (PolygonsPart& new_part_here : new_parts_here) + std::vector new_parts_here = new_outline.splitIntoParts(); + for (SingleShape& new_part_here : new_parts_here) { new_parts.push_back(new_part_here); } @@ -625,20 +627,20 @@ void FffPolygonGenerator::processInfillMesh(SliceDataStorage& storage, const siz } if (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL) { - const Polygons& own_infill_area = other_part.getOwnInfillArea(); - Polygons cut_lines = own_infill_area.intersectionPolyLines(layer.openPolyLines); - new_polylines.add(cut_lines); + const Shape& own_infill_area = other_part.getOwnInfillArea(); + OpenLinesSet cut_lines = own_infill_area.intersection(layer.open_polylines); + new_polylines.push_back(cut_lines); // NOTE: closed polygons will be represented as polylines, which will be closed automatically in the PathOrderOptimizer if (! own_infill_area.empty()) { - other_part.infill_area_own = own_infill_area.difference(layer.openPolyLines.offsetPolyLine(surface_line_width / 2)); + other_part.infill_area_own = own_infill_area.difference(layer.open_polylines.offset(surface_line_width / 2)); } } } } layer.parts.clear(); - for (const PolygonsPart& part : new_parts) + for (SingleShape& part : new_parts) { if (part.empty()) { @@ -651,10 +653,10 @@ void FffPolygonGenerator::processInfillMesh(SliceDataStorage& storage, const siz if (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL) { - layer.openPolyLines = new_polylines; + layer.open_polylines = new_polylines; } - if (layer.parts.size() > 0 || (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL && layer.openPolyLines.size() > 0)) + if (layer.parts.size() > 0 || (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL && layer.open_polylines.size() > 0)) { mesh.layer_nr_max_filled_layer = layer_idx; // last set by the highest non-empty layer } @@ -752,7 +754,7 @@ bool FffPolygonGenerator::isEmptyLayer(SliceDataStorage& storage, const LayerInd continue; } SliceLayer& layer = mesh.layers[layer_idx]; - if (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL && layer.openPolyLines.size() > 0) + if (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL && layer.open_polylines.size() > 0) { return false; } @@ -958,7 +960,7 @@ void FffPolygonGenerator::processOozeShield(SliceDataStorage& storage) { constexpr bool around_support = true; constexpr bool around_prime_tower = false; - storage.oozeShield.push_back(storage.getLayerOutlines(layer_nr, around_support, around_prime_tower).offset(ooze_shield_dist, ClipperLib::jtRound).getOutsidePolygons()); + storage.ooze_shield.push_back(storage.getLayerOutlines(layer_nr, around_support, around_prime_tower).offset(ooze_shield_dist, ClipperLib::jtRound).getOutsidePolygons()); } const AngleDegrees angle = mesh_group_settings.get("ooze_shield_angle"); @@ -968,18 +970,18 @@ void FffPolygonGenerator::processOozeShield(SliceDataStorage& storage) = tan(mesh_group_settings.get("ooze_shield_angle")) * mesh_group_settings.get("layer_height"); // Allow for a 60deg angle in the oozeShield. for (LayerIndex layer_nr = 1; layer_nr <= storage.max_print_height_second_to_last_extruder; layer_nr++) { - storage.oozeShield[layer_nr] = storage.oozeShield[layer_nr].unionPolygons(storage.oozeShield[layer_nr - 1].offset(-allowed_angle_offset)); + storage.ooze_shield[layer_nr] = storage.ooze_shield[layer_nr].unionPolygons(storage.ooze_shield[layer_nr - 1].offset(-allowed_angle_offset)); } for (LayerIndex layer_nr = storage.max_print_height_second_to_last_extruder; layer_nr > 0; layer_nr--) { - storage.oozeShield[layer_nr - 1] = storage.oozeShield[layer_nr - 1].unionPolygons(storage.oozeShield[layer_nr].offset(-allowed_angle_offset)); + storage.ooze_shield[layer_nr - 1] = storage.ooze_shield[layer_nr - 1].unionPolygons(storage.ooze_shield[layer_nr].offset(-allowed_angle_offset)); } } const double largest_printed_area = 1.0; // TODO: make var a parameter, and perhaps even a setting? for (LayerIndex layer_nr = 0; layer_nr <= storage.max_print_height_second_to_last_extruder; layer_nr++) { - storage.oozeShield[layer_nr].removeSmallAreas(largest_printed_area); + storage.ooze_shield[layer_nr].removeSmallAreas(largest_printed_area); } if (mesh_group_settings.get("prime_tower_enable")) { @@ -996,7 +998,7 @@ void FffPolygonGenerator::processOozeShield(SliceDataStorage& storage) } for (LayerIndex layer_nr = 0; layer_nr <= storage.max_print_height_second_to_last_extruder; layer_nr++) { - storage.oozeShield[layer_nr] = storage.oozeShield[layer_nr].difference(storage.primeTower.getOuterPoly(layer_nr).offset(max_line_width / 2)); + storage.ooze_shield[layer_nr] = storage.ooze_shield[layer_nr].difference(storage.primeTower.getOuterPoly(layer_nr).offset(max_line_width / 2)); } } } @@ -1013,7 +1015,7 @@ void FffPolygonGenerator::processDraftShield(SliceDataStorage& storage) const LayerIndex layer_skip{ 500 / layer_height + 1 }; - Polygons& draft_shield = storage.draft_protection_shield; + Shape& draft_shield = storage.draft_protection_shield; for (LayerIndex layer_nr = 0; layer_nr < storage.print_layer_count && layer_nr < draft_shield_layers; layer_nr += layer_skip) { constexpr bool around_support = true; @@ -1071,16 +1073,6 @@ void FffPolygonGenerator::processPlatformAdhesion(SliceDataStorage& storage) { skirt_brim.generateSupportBrim(); } - - for (const auto& extruder : Application::getInstance().current_slice_->scene.extruders) - { - Simplify simplifier(extruder.settings_); - for (auto skirt_brim_line : storage.skirt_brim[extruder.extruder_nr_]) - { - skirt_brim_line.closed_polygons = simplifier.polygon(skirt_brim_line.closed_polygons); - skirt_brim_line.open_polylines = simplifier.polyline(skirt_brim_line.open_polylines); - } - } } @@ -1100,7 +1092,7 @@ void FffPolygonGenerator::processFuzzyWalls(SliceMeshStorage& mesh) unsigned int start_layer_nr = (mesh.settings.get("adhesion_type") == EPlatformAdhesion::BRIM) ? 1 : 0; // don't make fuzzy skin on first layer if there's a brim - auto hole_area = Polygons(); + auto hole_area = Shape(); std::function accumulate_is_in_hole = []([[maybe_unused]] const bool& prev_result, [[maybe_unused]] const ExtrusionJunction& junction) { diff --git a/src/GCodePathConfig.cpp b/src/GCodePathConfig.cpp index 6cdfbd5d66..35e81edbb3 100644 --- a/src/GCodePathConfig.cpp +++ b/src/GCodePathConfig.cpp @@ -3,7 +3,7 @@ #include "GCodePathConfig.h" -#include "utils/Point2LL.h" // INT2MM +#include "geometry/Point2LL.h" // INT2MM namespace cura { diff --git a/src/InsetOrderOptimizer.cpp b/src/InsetOrderOptimizer.cpp index 82d4199d01..7be4d8f6ba 100644 --- a/src/InsetOrderOptimizer.cpp +++ b/src/InsetOrderOptimizer.cpp @@ -50,7 +50,8 @@ InsetOrderOptimizer::InsetOrderOptimizer( const size_t wall_0_extruder_nr, const size_t wall_x_extruder_nr, const ZSeamConfig& z_seam_config, - const std::vector& paths) + const std::vector& paths, + const Shape& disallowed_areas_for_seams) : gcode_writer_(gcode_writer) , storage_(storage) , gcode_layer_(gcode_layer) @@ -70,6 +71,7 @@ InsetOrderOptimizer::InsetOrderOptimizer( , z_seam_config_(z_seam_config) , paths_(paths) , layer_nr_(gcode_layer.getLayerNr()) + , disallowed_areas_for_seams_{ disallowed_areas_for_seams } { } @@ -94,12 +96,19 @@ bool InsetOrderOptimizer::addToLayer() bool added_something = false; constexpr bool detect_loops = false; - constexpr Polygons* combing_boundary = nullptr; + constexpr Shape* combing_boundary = nullptr; const auto group_outer_walls = settings_.get("group_outer_walls"); // When we alternate walls, also alternate the direction at which the first wall starts in. // On even layers we start with normal direction, on odd layers with inverted direction. - PathOrderOptimizer - order_optimizer(gcode_layer_.getLastPlannedPositionOrStartingPosition(), z_seam_config_, detect_loops, combing_boundary, reverse, order, group_outer_walls); + PathOrderOptimizer order_optimizer( + gcode_layer_.getLastPlannedPositionOrStartingPosition(), + z_seam_config_, + detect_loops, + combing_boundary, + reverse, + order, + group_outer_walls, + disallowed_areas_for_seams_); for (const auto& line : walls_to_be_added) { @@ -170,7 +179,7 @@ InsetOrderOptimizer::value_type InsetOrderOptimizer::getRegionOrder(const std::v | ranges::views::transform( [](const ExtrusionLine* line) { - const auto poly = line->toPolygon(); + const Polygon poly = line->toPolygon(); AABB aabb; aabb.include(poly); return std::make_pair(line, aabb.area()); @@ -203,10 +212,10 @@ InsetOrderOptimizer::value_type InsetOrderOptimizer::getRegionOrder(const std::v { // Create a polygon representing the inner area of the extrusion line; any // point inside this polygon is considered to the child of the extrusion line. - Polygons hole_polygons; + Shape hole_polygons; if (extrusion_line->is_closed_) { - hole_polygons.add(extrusion_line->toPolygon()); + hole_polygons.push_back(extrusion_line->toPolygon()); } if (hole_polygons.empty()) diff --git a/src/InterlockingGenerator.cpp b/src/InterlockingGenerator.cpp index ea1849d96f..abfab67ae7 100644 --- a/src/InterlockingGenerator.cpp +++ b/src/InterlockingGenerator.cpp @@ -12,6 +12,7 @@ #include "Application.h" #include "Slice.h" +#include "geometry/PointMatrix.h" #include "settings/types/LayerIndex.h" #include "slicer.h" #include "utils/VoxelUtils.h" @@ -68,16 +69,16 @@ void InterlockingGenerator::generateInterlockingStructure(std::vector& } } -std::pair InterlockingGenerator::growBorderAreasPerpendicular(const Polygons& a, const Polygons& b, const coord_t& detect) const +std::pair InterlockingGenerator::growBorderAreasPerpendicular(const Shape& a, const Shape& b, const coord_t& detect) const { const coord_t min_line = std::min(mesh_a_.mesh->settings_.get("min_wall_line_width"), mesh_b_.mesh->settings_.get("min_wall_line_width")); - const Polygons total_shrunk = a.offset(min_line).unionPolygons(b.offset(min_line)).offset(2 * -min_line); + const Shape total_shrunk = a.offset(min_line).unionPolygons(b.offset(min_line)).offset(2 * -min_line); - Polygons from_border_a = a.difference(total_shrunk); - Polygons from_border_b = b.difference(total_shrunk); + Shape from_border_a = a.difference(total_shrunk); + Shape from_border_b = b.difference(total_shrunk); - Polygons temp_a, temp_b; + Shape temp_a, temp_b; for (coord_t i = 0; i < (detect / min_line) + 2; ++i) { temp_a = from_border_a.offset(min_line); @@ -104,14 +105,14 @@ void InterlockingGenerator::handleThinAreas(const std::unordered_set const coord_t close_gaps = std::min(mesh_a_.mesh->settings_.get("line_width"), mesh_b_.mesh->settings_.get("line_width")) / 4; // Make an inclusionary polygon, to only actually handle thin areas near actual microstructures (so not in skin for example). - std::vector near_interlock_per_layer; - near_interlock_per_layer.assign(std::min(mesh_a_.layers.size(), mesh_b_.layers.size()), Polygons()); + std::vector near_interlock_per_layer; + near_interlock_per_layer.assign(std::min(mesh_a_.layers.size(), mesh_b_.layers.size()), Shape()); for (const auto& cell : has_all_meshes) { const Point3LL bottom_corner = vu_.toLowerCorner(cell); for (coord_t layer_nr = bottom_corner.z_; layer_nr < bottom_corner.z_ + cell_size_.z_ && layer_nr < static_cast(near_interlock_per_layer.size()); ++layer_nr) { - near_interlock_per_layer[static_cast(layer_nr)].add(vu_.toPolygon(cell)); + near_interlock_per_layer[static_cast(layer_nr)].push_back(vu_.toPolygon(cell)); } } for (auto& near_interlock : near_interlock_per_layer) @@ -123,20 +124,20 @@ void InterlockingGenerator::handleThinAreas(const std::unordered_set // Only alter layers when they are present in both meshes, zip should take care if that. for (auto [layer_nr, layer] : ranges::views::zip(mesh_a_.layers, mesh_b_.layers) | ranges::views::enumerate) { - Polygons& polys_a = std::get<0>(layer).polygons; - Polygons& polys_b = std::get<1>(layer).polygons; + Shape& polys_a = std::get<0>(layer).polygons_; + Shape& polys_b = std::get<1>(layer).polygons_; const auto [from_border_a, from_border_b] = growBorderAreasPerpendicular(polys_a, polys_b, detect); // Get the areas of each mesh that are _not_ thin (large), by performing a morphological open. - const Polygons large_a{ polys_a.offset(-detect).offset(detect) }; - const Polygons large_b{ polys_b.offset(-detect).offset(detect) }; + const Shape large_a{ polys_a.offset(-detect).offset(detect) }; + const Shape large_b{ polys_b.offset(-detect).offset(detect) }; // Derive the area that the thin areas need to expand into (so the added areas to the thin strips) from the information we already have. - const Polygons thin_expansion_a{ + const Shape thin_expansion_a{ large_b.intersection(polys_a.difference(large_a).offset(expand)).intersection(near_interlock_per_layer[layer_nr]).intersection(from_border_a).offset(rounding_errors) }; - const Polygons thin_expansion_b{ + const Shape thin_expansion_b{ large_a.intersection(polys_b.difference(large_b).offset(expand)).intersection(near_interlock_per_layer[layer_nr]).intersection(from_border_b).offset(rounding_errors) }; @@ -155,7 +156,7 @@ void InterlockingGenerator::generateInterlockingStructure() const std::unordered_set& has_all_meshes = voxels_per_mesh[1]; has_any_mesh.merge(has_all_meshes); // perform union and intersection simultaneously. Cannibalizes voxels_per_mesh - const std::vector layer_regions = computeUnionedVolumeRegions(); + const std::vector layer_regions = computeUnionedVolumeRegions(); if (air_filtering_) { @@ -183,11 +184,11 @@ std::vector> InterlockingGenerator::getShellVoxel Slicer* mesh = (mesh_idx == 0) ? &mesh_a_ : &mesh_b_; std::unordered_set& mesh_voxels = voxels_per_mesh[mesh_idx]; - std::vector rotated_polygons_per_layer(mesh->layers.size()); + std::vector rotated_polygons_per_layer(mesh->layers.size()); for (size_t layer_nr = 0; layer_nr < mesh->layers.size(); layer_nr++) { SlicerLayer& layer = mesh->layers[layer_nr]; - rotated_polygons_per_layer[layer_nr] = layer.polygons; + rotated_polygons_per_layer[layer_nr] = layer.polygons_; rotated_polygons_per_layer[layer_nr].applyMatrix(rotation_); } @@ -197,7 +198,7 @@ std::vector> InterlockingGenerator::getShellVoxel return voxels_per_mesh; } -void InterlockingGenerator::addBoundaryCells(const std::vector& layers, const DilationKernel& kernel, std::unordered_set& cells) const +void InterlockingGenerator::addBoundaryCells(const std::vector& layers, const DilationKernel& kernel, std::unordered_set& cells) const { auto voxel_emplacer = [&cells](GridPoint3 p) { @@ -209,7 +210,7 @@ void InterlockingGenerator::addBoundaryCells(const std::vector& layers { const coord_t z = static_cast(layer_nr); vu_.walkDilatedPolygons(layers[layer_nr], z, kernel, voxel_emplacer); - Polygons skin = layers[layer_nr]; + Shape skin = layers[layer_nr]; if (layer_nr > 0) { skin = skin.xorPolygons(layers[layer_nr - 1]); @@ -219,14 +220,14 @@ void InterlockingGenerator::addBoundaryCells(const std::vector& layers } } -std::vector InterlockingGenerator::computeUnionedVolumeRegions() const +std::vector InterlockingGenerator::computeUnionedVolumeRegions() const { const size_t max_layer_count = std::max(mesh_a_.layers.size(), mesh_b_.layers.size()) + 1; // introduce ghost layer on top for correct skin computation of topmost layer. - std::vector layer_regions(max_layer_count); + std::vector layer_regions(max_layer_count); for (LayerIndex layer_nr = 0; layer_nr < max_layer_count; layer_nr++) { - Polygons& layer_region = layer_regions[static_cast(layer_nr)]; + Shape& layer_region = layer_regions[static_cast(layer_nr)]; for (Slicer* mesh : { &mesh_a_, &mesh_b_ }) { if (layer_nr >= mesh->layers.size()) @@ -234,7 +235,7 @@ std::vector InterlockingGenerator::computeUnionedVolumeRegions() const break; } const SlicerLayer& layer = mesh->layers[static_cast(layer_nr)]; - layer_region.add(layer.polygons); + layer_region.push_back(layer.polygons_); } layer_region = layer_region.offset(ignored_gap_).offset(-ignored_gap_); // Morphological close to merge meshes into single volume layer_region.applyMatrix(rotation_); @@ -242,9 +243,9 @@ std::vector InterlockingGenerator::computeUnionedVolumeRegions() const return layer_regions; } -std::vector> InterlockingGenerator::generateMicrostructure() const +std::vector> InterlockingGenerator::generateMicrostructure() const { - std::vector> cell_area_per_mesh_per_layer; + std::vector> cell_area_per_mesh_per_layer; cell_area_per_mesh_per_layer.resize(2); cell_area_per_mesh_per_layer[0].resize(2); const coord_t beam_w_sum = beam_width_a_ + beam_width_b_; @@ -255,16 +256,16 @@ std::vector> InterlockingGenerator::generateMicrostructure Point2LL offset(mesh_idx ? middle : 0, 0); Point2LL area_size(width[mesh_idx], cell_size_.y_); - PolygonRef poly = cell_area_per_mesh_per_layer[0][mesh_idx].newPoly(); + Polygon& poly = cell_area_per_mesh_per_layer[0][mesh_idx].newLine(); poly.emplace_back(offset); poly.emplace_back(offset + Point2LL(area_size.X, 0)); poly.emplace_back(offset + area_size); poly.emplace_back(offset + Point2LL(0, area_size.Y)); } cell_area_per_mesh_per_layer[1] = cell_area_per_mesh_per_layer[0]; - for (Polygons& polys : cell_area_per_mesh_per_layer[1]) + for (Shape& polys : cell_area_per_mesh_per_layer[1]) { - for (PolygonRef poly : polys) + for (Polygon& poly : polys) { for (Point2LL& p : poly) { @@ -275,14 +276,14 @@ std::vector> InterlockingGenerator::generateMicrostructure return cell_area_per_mesh_per_layer; } -void InterlockingGenerator::applyMicrostructureToOutlines(const std::unordered_set& cells, const std::vector& layer_regions) const +void InterlockingGenerator::applyMicrostructureToOutlines(const std::unordered_set& cells, const std::vector& layer_regions) const { - std::vector> cell_area_per_mesh_per_layer = generateMicrostructure(); + std::vector> cell_area_per_mesh_per_layer = generateMicrostructure(); const PointMatrix unapply_rotation = rotation_.inverse(); const size_t max_layer_count = std::max(mesh_a_.layers.size(), mesh_b_.layers.size()); - std::vector structure_per_layer[2]; // for each mesh the structure on each layer + std::vector structure_per_layer[2]; // for each mesh the structure on each layer // Every `beam_layer_count` number of layers are combined to an interlocking beam layer // to store these we need ceil(max_layer_count / beam_layer_count) of these layers @@ -299,9 +300,9 @@ void InterlockingGenerator::applyMicrostructureToOutlines(const std::unordered_s { for (LayerIndex layer_nr = bottom_corner.z_; layer_nr < bottom_corner.z_ + cell_size_.z_ && layer_nr < max_layer_count; layer_nr += beam_layer_count_) { - Polygons areas_here = cell_area_per_mesh_per_layer[static_cast(layer_nr / beam_layer_count_) % cell_area_per_mesh_per_layer.size()][mesh_idx]; + Shape areas_here = cell_area_per_mesh_per_layer[static_cast(layer_nr / beam_layer_count_) % cell_area_per_mesh_per_layer.size()][mesh_idx]; areas_here.translate(Point2LL(bottom_corner.x_, bottom_corner.y_)); - structure_per_layer[mesh_idx][static_cast(layer_nr / beam_layer_count_)].add(areas_here); + structure_per_layer[mesh_idx][static_cast(layer_nr / beam_layer_count_)].push_back(areas_here); } } } @@ -310,7 +311,7 @@ void InterlockingGenerator::applyMicrostructureToOutlines(const std::unordered_s { for (size_t layer_nr = 0; layer_nr < structure_per_layer[mesh_idx].size(); layer_nr++) { - Polygons& layer_structure = structure_per_layer[mesh_idx][layer_nr]; + Shape& layer_structure = structure_per_layer[mesh_idx][layer_nr]; layer_structure = layer_structure.unionPolygons(); layer_structure.applyMatrix(unapply_rotation); } @@ -326,16 +327,16 @@ void InterlockingGenerator::applyMicrostructureToOutlines(const std::unordered_s break; } - Polygons layer_outlines = layer_regions[layer_nr]; + Shape layer_outlines = layer_regions[layer_nr]; layer_outlines.applyMatrix(unapply_rotation); - const Polygons areas_here = structure_per_layer[mesh_idx][layer_nr / static_cast(beam_layer_count_)].intersection(layer_outlines); - const Polygons& areas_other = structure_per_layer[! mesh_idx][layer_nr / static_cast(beam_layer_count_)]; + const Shape areas_here = structure_per_layer[mesh_idx][layer_nr / static_cast(beam_layer_count_)].intersection(layer_outlines); + const Shape& areas_other = structure_per_layer[! mesh_idx][layer_nr / static_cast(beam_layer_count_)]; SlicerLayer& layer = mesh->layers[layer_nr]; - layer.polygons = layer.polygons - .difference(areas_other) // reduce layer areas inward with beams from other mesh - .unionPolygons(areas_here); // extend layer areas outward with newly added beams + layer.polygons_ = layer.polygons_ + .difference(areas_other) // reduce layer areas inward with beams from other mesh + .unionPolygons(areas_here); // extend layer areas outward with newly added beams } } } diff --git a/src/LayerPlan.cpp b/src/LayerPlan.cpp index 3122cc46ac..b7fabe4edb 100644 --- a/src/LayerPlan.cpp +++ b/src/LayerPlan.cpp @@ -18,6 +18,7 @@ #include "Slice.h" #include "WipeScriptConfig.h" #include "communication/Communication.h" +#include "geometry/OpenPolyline.h" #include "pathPlanning/Comb.h" #include "pathPlanning/CombPaths.h" #include "plugins/slots.h" @@ -26,6 +27,7 @@ #include "sliceDataStorage.h" #include "utils/Simplify.h" #include "utils/linearAlg2D.h" +#include "utils/math.h" #include "utils/polygonUtils.h" #include "utils/section_type.h" @@ -66,7 +68,7 @@ GCodePath* LayerPlan::getLatestPathWithConfig( return ret; } -const Polygons* LayerPlan::getCombBoundaryInside() const +const Shape* LayerPlan::getCombBoundaryInside() const { return &comb_boundary_preferred_; } @@ -153,24 +155,24 @@ ExtruderTrain* LayerPlan::getLastPlannedExtruderTrain() return last_planned_extruder_; } -Polygons LayerPlan::computeCombBoundary(const CombBoundary boundary_type) +Shape LayerPlan::computeCombBoundary(const CombBoundary boundary_type) { - Polygons comb_boundary; + Shape comb_boundary; const CombingMode mesh_combing_mode = Application::getInstance().current_slice_->scene.current_mesh_group->settings.get("retraction_combing"); if (mesh_combing_mode != CombingMode::OFF && (layer_nr_ >= 0 || mesh_combing_mode != CombingMode::NO_SKIN)) { switch (layer_type_) { case Raft::LayerType::RaftBase: - comb_boundary = storage_.raftBaseOutline.offset(MM2INT(0.1)); + comb_boundary = storage_.raft_base_outline.offset(MM2INT(0.1)); break; case Raft::LayerType::RaftInterface: - comb_boundary = storage_.raftInterfaceOutline.offset(MM2INT(0.1)); + comb_boundary = storage_.raft_interface_outline.offset(MM2INT(0.1)); break; case Raft::LayerType::RaftSurface: - comb_boundary = storage_.raftSurfaceOutline.offset(MM2INT(0.1)); + comb_boundary = storage_.raft_surface_outline.offset(MM2INT(0.1)); break; case Raft::LayerType::Airgap: @@ -207,28 +209,28 @@ Polygons LayerPlan::computeCombBoundary(const CombBoundary boundary_type) { if (combing_mode == CombingMode::ALL) // Add the increased outline offset (skin, infill and part of the inner walls) { - comb_boundary.add(part.outline.offset(offset)); + comb_boundary.push_back(part.outline.offset(offset)); } else if (combing_mode == CombingMode::NO_SKIN) // Add the increased outline offset, subtract skin (infill and part of the inner walls) { - comb_boundary.add(part.outline.offset(offset).difference(part.inner_area.difference(part.infill_area))); + comb_boundary.push_back(part.outline.offset(offset).difference(part.inner_area.difference(part.infill_area))); } else if (combing_mode == CombingMode::NO_OUTER_SURFACES) { - Polygons top_and_bottom_most_fill; + Shape top_and_bottom_most_fill; for (const SliceLayerPart& outer_surface_part : layer.parts) { for (const SkinPart& skin_part : outer_surface_part.skin_parts) { - top_and_bottom_most_fill.add(skin_part.top_most_surface_fill); - top_and_bottom_most_fill.add(skin_part.bottom_most_surface_fill); + top_and_bottom_most_fill.push_back(skin_part.top_most_surface_fill); + top_and_bottom_most_fill.push_back(skin_part.bottom_most_surface_fill); } } - comb_boundary.add(part.outline.offset(offset).difference(top_and_bottom_most_fill)); + comb_boundary.push_back(part.outline.offset(offset).difference(top_and_bottom_most_fill)); } else if (combing_mode == CombingMode::INFILL) // Add the infill (infill only) { - comb_boundary.add(part.infill_area); + comb_boundary.push_back(part.infill_area); } } } @@ -569,7 +571,7 @@ void LayerPlan::addExtrusionMove( } void LayerPlan::addPolygon( - ConstPolygonRef polygon, + const Polygon& polygon, int start_idx, const bool backwards, const GCodePathConfig& config, @@ -624,7 +626,7 @@ void LayerPlan::addPolygon( } void LayerPlan::addPolygonsByOptimizer( - const Polygons& polygons, + const Shape& polygons, const GCodePathConfig& config, const ZSeamConfig& z_seam_config, coord_t wall_0_wipe_dist, @@ -638,16 +640,16 @@ void LayerPlan::addPolygonsByOptimizer( { return; } - PathOrderOptimizer orderOptimizer(start_near_location ? start_near_location.value() : getLastPlannedPositionOrStartingPosition(), z_seam_config); + PathOrderOptimizer orderOptimizer(start_near_location ? start_near_location.value() : getLastPlannedPositionOrStartingPosition(), z_seam_config); for (size_t poly_idx = 0; poly_idx < polygons.size(); poly_idx++) { - orderOptimizer.addPolygon(polygons[poly_idx]); + orderOptimizer.addPolygon(&polygons[poly_idx]); } orderOptimizer.optimize(); if (! reverse_order) { - for (const PathOrdering& path : orderOptimizer.paths_) + for (const PathOrdering& path : orderOptimizer.paths_) { addPolygon(*path.vertices_, path.start_vertex_, path.backwards_, config, wall_0_wipe_dist, spiralize, flow_ratio, always_retract); } @@ -656,8 +658,8 @@ void LayerPlan::addPolygonsByOptimizer( { for (int index = orderOptimizer.paths_.size() - 1; index >= 0; --index) { - const PathOrdering& path = orderOptimizer.paths_[index]; - addPolygon(**path.vertices_, path.start_vertex_, path.backwards_, config, wall_0_wipe_dist, spiralize, flow_ratio, always_retract); + const PathOrdering& path = orderOptimizer.paths_[index]; + addPolygon(*path.vertices_, path.start_vertex_, path.backwards_, config, wall_0_wipe_dist, spiralize, flow_ratio, always_retract); } } } @@ -804,10 +806,10 @@ void LayerPlan::addWallLine( // the default_config. Since the original line segment was straight we can simply print // to the first and last point of the intersected line segments alternating between // roofing and default_config's. - Polygons line_polys; - line_polys.addLine(p0, p1); + OpenLinesSet line_polys; + line_polys.addSegment(p0, p1); constexpr bool restitch = false; // only a single line doesn't need stitching - auto roofing_line_segments = roofing_mask_.intersectionPolyLines(line_polys, restitch); + auto roofing_line_segments = roofing_mask_.intersection(line_polys, restitch); if (roofing_line_segments.empty()) { @@ -877,19 +879,19 @@ void LayerPlan::addWallLine( // determine which segments of the line are bridges - Polygons line_polys; - line_polys.addLine(p0, p1); + OpenLinesSet line_polys; + line_polys.addSegment(p0, p1); constexpr bool restitch = false; // only a single line doesn't need stitching - line_polys = bridge_wall_mask_.intersectionPolyLines(line_polys, restitch); + line_polys = bridge_wall_mask_.intersection(line_polys, restitch); // line_polys now contains the wall lines that need to be printed using bridge_config while (line_polys.size() > 0) { // find the bridge line segment that's nearest to the current point - int nearest = 0; + size_t nearest = 0; double smallest_dist2 = vSize2f(cur_point - line_polys[0][0]); - for (unsigned i = 1; i < line_polys.size(); ++i) + for (size_t i = 1; i < line_polys.size(); ++i) { double dist2 = vSize2f(cur_point - line_polys[i][0]); if (dist2 < smallest_dist2) @@ -898,7 +900,7 @@ void LayerPlan::addWallLine( smallest_dist2 = dist2; } } - ConstPolygonRef bridge = line_polys[nearest]; + const OpenPolyline& bridge = line_polys[nearest]; // set b0 to the nearest vertex and b1 the furthest Point2LL b0 = bridge[0]; @@ -938,7 +940,7 @@ void LayerPlan::addWallLine( } // finished with this segment - line_polys.remove(nearest); + line_polys.removeAt(nearest); } // if we haven't yet reached p1, fill the gap with default_config line @@ -959,7 +961,7 @@ void LayerPlan::addWallLine( } void LayerPlan::addWall( - ConstPolygonRef wall, + const Polygon& wall, int start_idx, const Settings& settings, const GCodePathConfig& default_config, @@ -1049,15 +1051,15 @@ void LayerPlan::addWall( // determine which segments of the line are bridges - Polygons line_polys; - line_polys.addLine(p0.p_, p1.p_); + OpenLinesSet line_polys; + line_polys.addSegment(p0.p_, p1.p_); constexpr bool restitch = false; // only a single line doesn't need stitching - line_polys = bridge_wall_mask_.intersectionPolyLines(line_polys, restitch); + line_polys = bridge_wall_mask_.intersection(line_polys, restitch); while (line_polys.size() > 0) { // find the bridge line segment that's nearest to p0 - int nearest = 0; + size_t nearest = 0; double smallest_dist2 = vSize2f(p0.p_ - line_polys[0][0]); for (unsigned i = 1; i < line_polys.size(); ++i) { @@ -1068,7 +1070,7 @@ void LayerPlan::addWall( smallest_dist2 = dist2; } } - ConstPolygonRef bridge = line_polys[nearest]; + const OpenPolyline& bridge = line_polys[nearest]; // set b0 to the nearest vertex and b1 the furthest Point2LL b0 = bridge[0]; @@ -1094,7 +1096,7 @@ void LayerPlan::addWall( distance_to_bridge_start += bridge_line_len; // finished with this segment - line_polys.remove(nearest); + line_polys.removeAt(nearest); } } else if (! bridge_wall_mask_.inside(p0.p_, true)) @@ -1112,7 +1114,7 @@ void LayerPlan::addWall( bool first_line = true; const coord_t small_feature_max_length = settings.get("small_feature_max_length"); - const bool is_small_feature = (small_feature_max_length > 0) && (layer_nr_ == 0 || wall.inset_idx_ == 0) && cura::shorterThan(wall, small_feature_max_length); + const bool is_small_feature = (small_feature_max_length > 0) && (layer_nr_ == 0 || wall.inset_idx_ == 0) && wall.shorterThan(small_feature_max_length); Ratio small_feature_speed_factor = settings.get((layer_nr_ == 0) ? "small_feature_speed_factor_0" : "small_feature_speed_factor"); const Velocity min_speed = fan_speed_layer_time_settings_per_extruder_[getLastPlannedExtruderTrain()->extruder_nr_].cool_min_speed; small_feature_speed_factor = std::max((double)small_feature_speed_factor, (double)(min_speed / default_config.getSpeed())); @@ -1264,7 +1266,7 @@ void LayerPlan::addInfillWall(const ExtrusionLine& wall, const GCodePathConfig& } void LayerPlan::addWalls( - const Polygons& walls, + const Shape& walls, const Settings& settings, const GCodePathConfig& default_config, const GCodePathConfig& roofing_config, @@ -1275,21 +1277,21 @@ void LayerPlan::addWalls( bool always_retract) { // TODO: Deprecated in favor of ExtrusionJunction version below. - PathOrderOptimizer orderOptimizer(getLastPlannedPositionOrStartingPosition(), z_seam_config); - for (size_t poly_idx = 0; poly_idx < walls.size(); poly_idx++) + PathOrderOptimizer orderOptimizer(getLastPlannedPositionOrStartingPosition(), z_seam_config); + for (const Polygon& polygon : walls) { - orderOptimizer.addPolygon(walls[poly_idx]); + orderOptimizer.addPolygon(&polygon); } orderOptimizer.optimize(); - for (const PathOrdering& path : orderOptimizer.paths_) + for (const PathOrdering& path : orderOptimizer.paths_) { - addWall(**path.vertices_, path.start_vertex_, settings, default_config, roofing_config, bridge_config, wall_0_wipe_dist, flow_ratio, always_retract); + addWall(*path.vertices_, path.start_vertex_, settings, default_config, roofing_config, bridge_config, wall_0_wipe_dist, flow_ratio, always_retract); } } - +template void LayerPlan::addLinesByOptimizer( - const Polygons& polygons, + const LinesSet& lines, const GCodePathConfig& config, const SpaceFillType space_fill_type, const bool enable_travel_optimization, @@ -1298,9 +1300,9 @@ void LayerPlan::addLinesByOptimizer( const std::optional near_start_location, const double fan_speed, const bool reverse_print_direction, - const std::unordered_multimap& order_requirements) + const std::unordered_multimap& order_requirements) { - Polygons boundary; + Shape boundary; if (enable_travel_optimization && ! comb_boundary_minimum_.empty()) { // use the combing boundary inflated so that all infill lines are inside the boundary @@ -1318,30 +1320,98 @@ void LayerPlan::addLinesByOptimizer( } dist += 100; // ensure boundary is slightly outside all skin/infill lines } - boundary.add(comb_boundary_minimum_.offset(dist)); + boundary.push_back(comb_boundary_minimum_.offset(dist)); // simplify boundary to cut down processing time boundary = Simplify(MM2INT(0.1), MM2INT(0.1), 0).polygon(boundary); } constexpr bool detect_loops = true; - PathOrderOptimizer order_optimizer( + PathOrderOptimizer order_optimizer( near_start_location.value_or(getLastPlannedPositionOrStartingPosition()), ZSeamConfig(), detect_loops, &boundary, reverse_print_direction, order_requirements); - for (size_t line_idx = 0; line_idx < polygons.size(); line_idx++) + if constexpr (std::is_same::value) + { + for (const OpenPolyline& polyline : lines) + { + order_optimizer.addPolyline(&polyline); + } + } + if constexpr (std::is_same::value) { - order_optimizer.addPolyline(polygons[line_idx]); + for (const ClosedPolyline& polyline : lines) + { + order_optimizer.addPolygon(&polyline); + } } order_optimizer.optimize(); addLinesInGivenOrder(order_optimizer.paths_, config, space_fill_type, wipe_dist, flow_ratio, fan_speed); } +void LayerPlan::addLinesByOptimizer( + const MixedLinesSet& lines, + const GCodePathConfig& config, + const SpaceFillType space_fill_type, + const bool enable_travel_optimization, + const coord_t wipe_dist, + const Ratio flow_ratio, + const std::optional near_start_location, + const double fan_speed, + const bool reverse_print_direction, + const std::unordered_multimap& order_requirements) +{ + Shape boundary; + if (enable_travel_optimization && ! comb_boundary_minimum_.empty()) + { + // use the combing boundary inflated so that all infill lines are inside the boundary + int dist = 0; + if (layer_nr_ >= 0) + { + // determine how much the skin/infill lines overlap the combing boundary + for (const std::shared_ptr& mesh : storage_.meshes) + { + const coord_t overlap = std::max(mesh->settings.get("skin_overlap_mm"), mesh->settings.get("infill_overlap_mm")); + if (overlap > dist) + { + dist = overlap; + } + } + dist += 100; // ensure boundary is slightly outside all skin/infill lines + } + boundary.push_back(comb_boundary_minimum_.offset(dist)); + // simplify boundary to cut down processing time + boundary = Simplify(MM2INT(0.1), MM2INT(0.1), 0).polygon(boundary); + } + constexpr bool detect_loops = false; // We already know which lines are closed + PathOrderOptimizer order_optimizer( + near_start_location.value_or(getLastPlannedPositionOrStartingPosition()), + ZSeamConfig(), + detect_loops, + &boundary, + reverse_print_direction, + order_requirements); + for (const std::shared_ptr& line : lines) + { + if (const std::shared_ptr open_line = dynamic_pointer_cast(line)) + { + order_optimizer.addPolyline(open_line.get()); + } + else if (const std::shared_ptr closed_line = dynamic_pointer_cast(line)) + { + order_optimizer.addPolygon(closed_line.get()); + } + } + + order_optimizer.optimize(); + + addLinesInGivenOrder(order_optimizer.paths_, config, space_fill_type, wipe_dist, flow_ratio, fan_speed); +} void LayerPlan::addLinesInGivenOrder( - const std::vector>& paths, + const std::vector>& lines, const GCodePathConfig& config, const SpaceFillType space_fill_type, const coord_t wipe_dist, @@ -1350,10 +1420,14 @@ void LayerPlan::addLinesInGivenOrder( { coord_t half_line_width = config.getLineWidth() / 2; coord_t line_width_2 = half_line_width * half_line_width; - for (size_t order_idx = 0; order_idx < paths.size(); order_idx++) + for (size_t order_idx = 0; order_idx < lines.size(); order_idx++) { - const PathOrdering& path = paths[order_idx]; - ConstPolygonRef polyline = *path.vertices_; + const PathOrdering& path = lines[order_idx]; + const Polyline& polyline = *path.vertices_; + if (! polyline.isValid()) + { + continue; + } const size_t start_idx = path.start_vertex_; assert(start_idx == 0 || start_idx == polyline.size() - 1 || path.is_closed_); const Point2LL start = polyline[start_idx]; @@ -1413,16 +1487,16 @@ void LayerPlan::addLinesInGivenOrder( int line_width = config.getLineWidth(); // Don't wipe if current extrusion is too small - if (polyline.polylineLength() <= line_width * 2) + if (polyline.length() <= line_width * 2) { wipe = false; } // Don't wipe if next starting point is very near - if (wipe && (order_idx < paths.size() - 1)) + if (wipe && (order_idx < lines.size() - 1)) { - const PathOrdering& next_path = paths[order_idx + 1]; - ConstPolygonRef next_polygon = *next_path.vertices_; + const PathOrdering& next_path = lines[order_idx + 1]; + const Polyline& next_polygon = *next_path.vertices_; const size_t next_start = next_path.start_vertex_; const Point2LL& next_p0 = next_polygon[next_start]; if (vSize2(next_p0 - p1) <= line_width * line_width * 4) @@ -1444,8 +1518,8 @@ void LayerPlan::addLinesInGivenOrder( } void LayerPlan::addLinesMonotonic( - const Polygons& area, - const Polygons& polygons, + const Shape& area, + const OpenLinesSet& lines, const GCodePathConfig& config, const SpaceFillType space_fill_type, const AngleRadians monotonic_direction, @@ -1455,39 +1529,44 @@ void LayerPlan::addLinesMonotonic( const Ratio flow_ratio, const double fan_speed) { - const Polygons exclude_areas = area.tubeShape(exclude_distance, exclude_distance); + const Shape exclude_areas = area.createTubeShape(exclude_distance, exclude_distance); const coord_t exclude_dist2 = exclude_distance * exclude_distance; const Point2LL last_position = getLastPlannedPositionOrStartingPosition(); // First lay all adjacent lines next to each other, to have a sensible input to the monotonic part of the algorithm. - PathOrderOptimizer line_order(last_position); - for (const ConstPolygonRef polyline : polygons) + PathOrderOptimizer line_order(last_position); + for (const OpenPolyline& line : lines) { - line_order.addPolyline(polyline); + line_order.addPolyline(&line); } line_order.optimize(); - const auto is_inside_exclusion = [&exclude_areas, &exclude_dist2](ConstPolygonRef path) + const auto is_inside_exclusion = [&exclude_areas, &exclude_dist2](const OpenPolyline& path) { return vSize2(path[1] - path[0]) < exclude_dist2 && exclude_areas.inside((path[0] + path[1]) / 2); }; // Order monotonically, except for line-segments which stay in the excluded areas (read: close to the walls) consecutively. - PathOrderMonotonic order(monotonic_direction, max_adjacent_distance, last_position); - Polygons left_over; + PathOrderMonotonic order(monotonic_direction, max_adjacent_distance, last_position); + OpenLinesSet left_over; bool last_would_have_been_excluded = false; for (size_t line_idx = 0; line_idx < line_order.paths_.size(); ++line_idx) { - const ConstPolygonRef polyline = *line_order.paths_[line_idx].vertices_; + const OpenPolyline& polyline = *line_order.paths_[line_idx].vertices_; + if (! polyline.isValid()) + { + continue; + } + const bool inside_exclusion = is_inside_exclusion(polyline); const bool next_would_have_been_included = inside_exclusion && (line_idx < line_order.paths_.size() - 1 && is_inside_exclusion(*line_order.paths_[line_idx + 1].vertices_)); if (inside_exclusion && last_would_have_been_excluded && next_would_have_been_included) { - left_over.add(polyline); + left_over.push_back(polyline); } else { - order.addPolyline(polyline); + order.addPolyline(&polyline); } last_would_have_been_excluded = inside_exclusion; } @@ -1502,8 +1581,8 @@ void LayerPlan::addLinesMonotonic( void LayerPlan::spiralizeWallSlice( const GCodePathConfig& config, - ConstPolygonRef wall, - ConstPolygonRef last_wall, + const Polygon& wall, + const Polygon& last_wall, const int seam_vertex_idx, const int last_seam_vertex_idx, const bool is_top_layer, @@ -1533,8 +1612,8 @@ void LayerPlan::spiralizeWallSlice( } const int n_points = wall.size(); - Polygons last_wall_polygons; - last_wall_polygons.add(last_wall); + Shape last_wall_polygons; + last_wall_polygons.push_back(last_wall); const int max_dist2 = config.getLineWidth() * config.getLineWidth() * 4; // (2 * lineWidth)^2; double total_length = 0.0; // determine the length of the complete wall @@ -1605,7 +1684,7 @@ void LayerPlan::spiralizeWallSlice( if (smooth_contours && ! is_bottom_layer && wall_point_idx < n_points) { // now find the point on the last wall that is closest to p - ClosestPolygonPoint cpp = PolygonUtils::findClosest(p, last_wall_polygons); + ClosestPointPolygon cpp = PolygonUtils::findClosest(p, last_wall_polygons); // if we found a point and it's not further away than max_dist2, use it if (cpp.isValid() && vSize2(cpp.location_ - p) <= max_dist2) @@ -2600,19 +2679,43 @@ size_t LayerPlan::getExtruder() const return extruder_plans_.back().extruder_nr_; } -void LayerPlan::setBridgeWallMask(const Polygons& polys) +void LayerPlan::setBridgeWallMask(const Shape& polys) { bridge_wall_mask_ = polys; } -void LayerPlan::setOverhangMask(const Polygons& polys) +void LayerPlan::setOverhangMask(const Shape& polys) { overhang_mask_ = polys; } -void LayerPlan::setRoofingMask(const Polygons& polys) +void LayerPlan::setRoofingMask(const Shape& polys) { roofing_mask_ = polys; } +template void LayerPlan::addLinesByOptimizer( + const OpenLinesSet& lines, + const GCodePathConfig& config, + const SpaceFillType space_fill_type, + const bool enable_travel_optimization, + const coord_t wipe_dist, + const Ratio flow_ratio, + const std::optional near_start_location, + const double fan_speed, + const bool reverse_print_direction, + const std::unordered_multimap& order_requirements); + +template void LayerPlan::addLinesByOptimizer( + const ClosedLinesSet& lines, + const GCodePathConfig& config, + const SpaceFillType space_fill_type, + const bool enable_travel_optimization, + const coord_t wipe_dist, + const Ratio flow_ratio, + const std::optional near_start_location, + const double fan_speed, + const bool reverse_print_direction, + const std::unordered_multimap& order_requirements); + } // namespace cura diff --git a/src/MeshGroup.cpp b/src/MeshGroup.cpp index f71b2f7bd5..91fae7ae5c 100644 --- a/src/MeshGroup.cpp +++ b/src/MeshGroup.cpp @@ -293,6 +293,8 @@ bool loadMeshIntoMeshGroup(MeshGroup* meshgroup, const char* filename, const Mat spdlog::info("loading '{}' took {:03.3f} seconds", filename, load_timer.restart()); return true; } + spdlog::warn("loading '{}' failed", filename); + return false; } spdlog::warn("Unable to recognize the extension of the file. Currently only .stl and .STL are supported."); return false; diff --git a/src/Mold.cpp b/src/Mold.cpp index c5149467af..48b4fba93c 100644 --- a/src/Mold.cpp +++ b/src/Mold.cpp @@ -3,14 +3,17 @@ #include "Mold.h" +#include + #include "Application.h" //To get settings. #include "ExtruderTrain.h" #include "Scene.h" #include "Slice.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Point2LL.h" #include "settings/types/Ratio.h" #include "sliceDataStorage.h" #include "slicer.h" -#include "utils/Point2LL.h" namespace cura { @@ -46,11 +49,11 @@ void Mold::process(std::vector& slicer_list) } const coord_t layer_height = scene.current_mesh_group->settings.get("layer_height"); - std::vector mold_outline_above_per_mesh; // the outer outlines of the layer above without the original model(s) being cut out + std::vector mold_outline_above_per_mesh; // the outer outlines of the layer above without the original model(s) being cut out mold_outline_above_per_mesh.resize(slicer_list.size()); for (int layer_nr = layer_count - 1; layer_nr >= 0; layer_nr--) { - Polygons all_original_mold_outlines; // outlines of all models for which to generate a mold (insides of all molds) + Shape all_original_mold_outlines; // outlines of all models for which to generate a mold (insides of all molds) // first generate outlines for (unsigned int mesh_idx = 0; mesh_idx < slicer_list.size(); mesh_idx++) @@ -76,29 +79,29 @@ void Mold::process(std::vector& slicer_list) SlicerLayer& layer = slicer.layers[layer_nr]; - Polygons model_outlines = layer.polygons.unionPolygons(layer.openPolylines.offsetPolyLine(open_polyline_width / 2)); - layer.openPolylines.clear(); - all_original_mold_outlines.add(model_outlines); + Shape model_outlines = layer.polygons_.unionPolygons(layer.open_polylines_.offset(open_polyline_width / 2)); + layer.open_polylines_.clear(); + all_original_mold_outlines.push_back(model_outlines); if (angle >= 90) { - layer.polygons = model_outlines.offset(width, ClipperLib::jtRound); + layer.polygons_ = model_outlines.offset(width, ClipperLib::jtRound); } else { - Polygons& mold_outline_above = mold_outline_above_per_mesh[mesh_idx]; // the outside of the mold on the layer above - layer.polygons = mold_outline_above.offset(-inset).unionPolygons(model_outlines.offset(width, ClipperLib::jtRound)); + Shape& mold_outline_above = mold_outline_above_per_mesh[mesh_idx]; // the outside of the mold on the layer above + layer.polygons_ = mold_outline_above.offset(-inset).unionPolygons(model_outlines.offset(width, ClipperLib::jtRound)); } // add roofs if (roof_layer_count > 0 && layer_nr > 0) { LayerIndex layer_nr_below = std::max(0, static_cast(layer_nr - roof_layer_count)); - Polygons roofs = slicer.layers[layer_nr_below].polygons.offset(width, ClipperLib::jtRound); // TODO: don't compute offset twice! - layer.polygons = layer.polygons.unionPolygons(roofs); + Shape roofs = slicer.layers[layer_nr_below].polygons_.offset(width, ClipperLib::jtRound); // TODO: don't compute offset twice! + layer.polygons_ = layer.polygons_.unionPolygons(roofs); } - mold_outline_above_per_mesh[mesh_idx] = layer.polygons; + mold_outline_above_per_mesh[mesh_idx] = layer.polygons_; } all_original_mold_outlines = all_original_mold_outlines.unionPolygons(); @@ -114,7 +117,7 @@ void Mold::process(std::vector& slicer_list) } Slicer& slicer = *slicer_list[mesh_idx]; SlicerLayer& layer = slicer.layers[layer_nr]; - layer.polygons = layer.polygons.difference(all_original_mold_outlines); + layer.polygons_ = layer.polygons_.difference(all_original_mold_outlines); } } } diff --git a/src/PathOrderPath.cpp b/src/PathOrderPath.cpp deleted file mode 100644 index cfca513fa9..0000000000 --- a/src/PathOrderPath.cpp +++ /dev/null @@ -1,50 +0,0 @@ -// Copyright (c) 2023 UltiMaker -// CuraEngine is released under the terms of the AGPLv3 or higher. - -#include "PathOrdering.h" //The definitions we're implementing here. -#include "WallToolPaths.h" -#include "sliceDataStorage.h" //For SliceLayerPart. - -namespace cura -{ - -template<> -ConstPolygonRef PathOrdering::getVertexData() -{ - return *vertices_; -} - -template<> -ConstPolygonRef PathOrdering::getVertexData() -{ - return *vertices_; -} - -template<> -ConstPolygonRef PathOrdering::getVertexData() -{ - return vertices_->outline.outerPolygon(); -} - -template<> -ConstPolygonRef PathOrdering::getVertexData() -{ - return vertices_->outline.outerPolygon(); -} - -template<> -ConstPolygonRef PathOrdering::getVertexData() -{ - return vertices_->outline_.outerPolygon(); -} -template<> -ConstPolygonRef PathOrdering::getVertexData() -{ - if (! cached_vertices_) - { - cached_vertices_ = vertices_->toPolygon(); - } - return ConstPolygonRef(*cached_vertices_); -} - -} // namespace cura diff --git a/src/PrimeTower.cpp b/src/PrimeTower.cpp index 57dc4d9853..ecde02dffe 100644 --- a/src/PrimeTower.cpp +++ b/src/PrimeTower.cpp @@ -1,17 +1,17 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #include "PrimeTower.h" #include #include +#include #include #include "Application.h" //To get settings. #include "ExtruderTrain.h" #include "LayerPlan.h" -#include "PrintFeature.h" #include "Scene.h" #include "Slice.h" #include "gcodeExport.h" @@ -52,35 +52,37 @@ PrimeTower::PrimeTower() && scene.current_mesh_group->settings.get("prime_tower_size") > 10; would_have_actual_tower_ = enabled_; // Assume so for now. +} - extruder_count_ = scene.extruders.size(); - extruder_order_.resize(extruder_count_); - for (unsigned int extruder_nr = 0; extruder_nr < extruder_count_; extruder_nr++) +void PrimeTower::initializeExtruders(const std::vector& used_extruders) +{ + // Add used extruders in default order, then sort. + for (unsigned int extruder_nr = 0; extruder_nr < used_extruders.size(); extruder_nr++) { - extruder_order_[extruder_nr] = extruder_nr; // Start with default order, then sort. + if (used_extruders[extruder_nr]) + { + extruder_order_.push_back(extruder_nr); + } } + + extruder_count_ = extruder_order_.size(); + // Sort from high adhesion to low adhesion. - const Scene* scene_pointer = &scene; // Communicate to lambda via pointer to prevent copy. + const Scene& scene = Application::getInstance().current_slice_->scene; std::stable_sort( extruder_order_.begin(), extruder_order_.end(), - [scene_pointer](const unsigned int& extruder_nr_a, const unsigned int& extruder_nr_b) -> bool + [&scene](const unsigned int& extruder_nr_a, const unsigned int& extruder_nr_b) -> bool { - const Ratio adhesion_a = scene_pointer->extruders[extruder_nr_a].settings_.get("material_adhesion_tendency"); - const Ratio adhesion_b = scene_pointer->extruders[extruder_nr_b].settings_.get("material_adhesion_tendency"); + const Ratio adhesion_a = scene.extruders[extruder_nr_a].settings_.get("material_adhesion_tendency"); + const Ratio adhesion_b = scene.extruders[extruder_nr_b].settings_.get("material_adhesion_tendency"); return adhesion_a < adhesion_b; }); } -void PrimeTower::checkUsed(const SliceDataStorage& storage) +void PrimeTower::checkUsed() { - std::vector extruder_is_used = storage.getExtrudersUsed(); - size_t used_extruder_count = 0; - for (bool is_used : extruder_is_used) - { - used_extruder_count += is_used; - } - if (used_extruder_count <= 1) + if (extruder_count_ <= 1) { enabled_ = false; } @@ -100,7 +102,7 @@ void PrimeTower::generateGroundpoly() const coord_t x = mesh_group_settings.get("prime_tower_position_x"); const coord_t y = mesh_group_settings.get("prime_tower_position_y"); const coord_t tower_radius = tower_size / 2; - outer_poly_.add(PolygonUtils::makeCircle(Point2LL(x - tower_radius, y + tower_radius), tower_radius, TAU / CIRCLE_RESOLUTION)); + outer_poly_.push_back(PolygonUtils::makeCircle(Point2LL(x - tower_radius, y + tower_radius), tower_radius, TAU / CIRCLE_RESOLUTION)); middle_ = Point2LL(x - tower_size / 2, y + tower_size / 2); post_wipe_point_ = Point2LL(x - tower_size / 2, y + tower_size / 2); @@ -108,7 +110,7 @@ void PrimeTower::generateGroundpoly() void PrimeTower::generatePaths(const SliceDataStorage& storage) { - checkUsed(storage); + checkUsed(); const int raft_total_extra_layers = Raft::getTotalExtraLayers(); would_have_actual_tower_ = storage.max_print_height_second_to_last_extruder @@ -138,9 +140,13 @@ void PrimeTower::generatePaths_denseInfill(std::vector& cumulative_inse const coord_t base_height = std::max(scene.settings.get("prime_tower_base_height"), has_raft ? layer_height : 0); const double base_curve_magnitude = mesh_group_settings.get("prime_tower_base_curve_magnitude"); - prime_moves_.resize(extruder_count_); - base_extra_moves_.resize(extruder_count_); - inset_extra_moves_.resize(extruder_count_); + for (size_t extruder_nr : extruder_order_) + { + // By default, add empty moves for every extruder + prime_moves_[extruder_nr]; + base_extra_moves_[extruder_nr]; + inset_extra_moves_[extruder_nr]; + } coord_t cumulative_inset = 0; // Each tower shape is going to be printed inside the other. This is the inset we're doing for each extruder. for (size_t extruder_nr : extruder_order_) @@ -149,16 +155,16 @@ void PrimeTower::generatePaths_denseInfill(std::vector& cumulative_inse const coord_t required_volume = MM3_2INT(scene.extruders[extruder_nr].settings_.get("prime_tower_min_volume")); const Ratio flow = scene.extruders[extruder_nr].settings_.get("prime_tower_flow"); coord_t current_volume = 0; - Polygons& prime_moves = prime_moves_[extruder_nr]; + Shape& prime_moves = prime_moves_[extruder_nr]; // Create the walls of the prime tower. unsigned int wall_nr = 0; for (; current_volume < required_volume; wall_nr++) { // Create a new polygon with an offset from the outer polygon. - Polygons polygons = outer_poly_.offset(-cumulative_inset - wall_nr * line_width - line_width / 2); - prime_moves.add(polygons); - current_volume += polygons.polygonLength() * line_width * layer_height * flow; + Shape polygons = outer_poly_.offset(-cumulative_inset - wall_nr * line_width - line_width / 2); + prime_moves.push_back(polygons); + current_volume += polygons.length() * line_width * layer_height * flow; if (polygons.empty()) // Don't continue. We won't ever reach the required volume because it doesn't fit. { break; @@ -180,7 +186,6 @@ void PrimeTower::generatePaths_denseInfill(std::vector& cumulative_inse } extra_radius = line_width * extra_rings; outer_poly_base_.push_back(outer_poly_.offset(extra_radius)); - base_extra_moves_[extruder_nr].push_back(PolygonUtils::generateOutset(outer_poly_, extra_rings, line_width)); } } @@ -195,10 +200,10 @@ void PrimeTower::generatePaths_denseInfill(std::vector& cumulative_inse if (extruder_nr == extruder_order_.back() || method == PrimeTowerMethod::INTERLEAVED) { const coord_t line_width = scene.extruders[extruder_nr].settings_.get("prime_tower_line_width"); - Polygons pattern = PolygonUtils::generateInset(outer_poly_, line_width, cumulative_inset); + Shape pattern = PolygonUtils::generateInset(outer_poly_, line_width, cumulative_inset); if (! pattern.empty()) { - inset_extra_moves_[extruder_nr].add(pattern); + inset_extra_moves_[extruder_nr].push_back(pattern); } } } @@ -226,8 +231,6 @@ void PrimeTower::generatePaths_sparseInfill(const std::vector& cumulati if (method == PrimeTowerMethod::INTERLEAVED || method == PrimeTowerMethod::NORMAL) { - const size_t nb_extruders = scene.extruders.size(); - // Pre-compute radiuses of each extruder ring std::vector rings_radii; const coord_t tower_size = mesh_group_settings.get("prime_tower_size"); @@ -242,9 +245,9 @@ void PrimeTower::generatePaths_sparseInfill(const std::vector& cumulati // Generate all possible extruders combinations, e.g. if there are 4 extruders, we have combinations // 0 / 0-1 / 0-1-2 / 0-1-2-3 / 1 / 1-2 / 1-2-3 / 2 / 2-3 / 3 // A combination is represented by a bitmask - for (size_t first_extruder_idx = 0; first_extruder_idx < nb_extruders; ++first_extruder_idx) + for (size_t first_extruder_idx = 0; first_extruder_idx < extruder_count_; ++first_extruder_idx) { - size_t nb_extruders_sparse = method == PrimeTowerMethod::NORMAL ? first_extruder_idx + 1 : nb_extruders; + size_t nb_extruders_sparse = method == PrimeTowerMethod::NORMAL ? first_extruder_idx + 1 : extruder_count_; for (size_t last_extruder_idx = first_extruder_idx; last_extruder_idx < nb_extruders_sparse; ++last_extruder_idx) { @@ -255,12 +258,12 @@ void PrimeTower::generatePaths_sparseInfill(const std::vector& cumulati extruders_combination |= (1 << extruder_nr); } - std::map infills_for_combination; + std::map infills_for_combination; for (const ActualExtruder& actual_extruder : actual_extruders) { if (method == PrimeTowerMethod::INTERLEAVED || actual_extruder.number == extruder_order_.at(first_extruder_idx)) { - Polygons infill = generatePath_sparseInfill(first_extruder_idx, last_extruder_idx, rings_radii, actual_extruder.line_width, actual_extruder.number); + Shape infill = generatePath_sparseInfill(first_extruder_idx, last_extruder_idx, rings_radii, actual_extruder.line_width, actual_extruder.number); infills_for_combination[actual_extruder.number] = infill; } } @@ -271,7 +274,7 @@ void PrimeTower::generatePaths_sparseInfill(const std::vector& cumulati } } -Polygons PrimeTower::generatePath_sparseInfill( +Shape PrimeTower::generatePath_sparseInfill( const size_t first_extruder_idx, const size_t last_extruder_idx, const std::vector& rings_radii, @@ -285,7 +288,7 @@ Polygons PrimeTower::generatePath_sparseInfill( const coord_t radius_delta = outer_radius - inner_radius; const coord_t semi_line_width = line_width / 2; - Polygons pattern; + Shape pattern; // Split ring according to max bridging distance const size_t nb_rings = std::ceil(static_cast(radius_delta) / max_bridging_distance); @@ -300,7 +303,7 @@ Polygons PrimeTower::generatePath_sparseInfill( const size_t semi_nb_spokes = std::ceil((std::numbers::pi * ring_outer_radius) / max_bridging_distance); - pattern.add(PolygonUtils::makeWheel(middle_, ring_inner_radius, ring_outer_radius, semi_nb_spokes, ARC_RESOLUTION)); + pattern.push_back(PolygonUtils::makeWheel(middle_, ring_inner_radius, ring_outer_radius, semi_nb_spokes, ARC_RESOLUTION)); } } @@ -444,7 +447,7 @@ void PrimeTower::addToGcode_denseInfill(LayerPlan& gcode_layer, const size_t ext { // Actual prime pattern const GCodePathConfig& config = gcode_layer.configs_storage_.prime_tower_config_per_extruder[extruder_nr]; - const Polygons& pattern = prime_moves_[extruder_nr]; + const Shape& pattern = prime_moves_.at(extruder_nr); gcode_layer.addPolygonsByOptimizer(pattern, config); } } @@ -454,11 +457,11 @@ bool PrimeTower::addToGcode_base(LayerPlan& gcode_layer, const size_t extruder_n const size_t raft_total_extra_layers = Raft::getTotalExtraLayers(); LayerIndex absolute_layer_number = gcode_layer.getLayerNr() + raft_total_extra_layers; - const std::vector& pattern_extra_brim = base_extra_moves_[extruder_nr]; + const auto& pattern_extra_brim = base_extra_moves_.at(extruder_nr); if (absolute_layer_number < pattern_extra_brim.size()) { // Extra rings for stronger base - const Polygons& pattern = pattern_extra_brim[absolute_layer_number]; + const auto& pattern = pattern_extra_brim[absolute_layer_number]; if (! pattern.empty()) { const GCodePathConfig& config = gcode_layer.configs_storage_.prime_tower_config_per_extruder[extruder_nr]; @@ -477,7 +480,7 @@ bool PrimeTower::addToGcode_inset(LayerPlan& gcode_layer, const size_t extruder_ if (absolute_layer_number == 0) // Extra-adhesion on very first layer only { - const Polygons& pattern_extra_inset = inset_extra_moves_[extruder_nr]; + const Shape& pattern_extra_inset = inset_extra_moves_.at(extruder_nr); if (! pattern_extra_inset.empty()) { const GCodePathConfig& config = gcode_layer.configs_storage_.prime_tower_config_per_extruder[extruder_nr]; @@ -531,7 +534,7 @@ void PrimeTower::addToGcode_sparseInfill(LayerPlan& gcode_layer, const std::vect auto iterator_combination = sparse_pattern_per_extruders_.find(mask); if (iterator_combination != sparse_pattern_per_extruders_.end()) { - const std::map& infill_for_combination = iterator_combination->second; + const std::map& infill_for_combination = iterator_combination->second; auto iterator_extruder_nr = infill_for_combination.find(current_extruder_nr); if (iterator_extruder_nr != infill_for_combination.end()) @@ -596,7 +599,7 @@ void PrimeTower::subtractFromSupport(SliceDataStorage& storage) { for (size_t layer = 0; layer <= (size_t)storage.max_print_height_second_to_last_extruder + 1 && layer < storage.support.supportLayers.size(); layer++) { - const Polygons outside_polygon = getOuterPoly(layer).getOutsidePolygons(); + const Shape outside_polygon = getOuterPoly(layer).getOutsidePolygons(); AABB outside_polygon_boundary_box(outside_polygon); SupportLayer& support_layer = storage.support.supportLayers[layer]; // take the differences of the support infill parts and the prime tower area @@ -604,7 +607,7 @@ void PrimeTower::subtractFromSupport(SliceDataStorage& storage) } } -const Polygons& PrimeTower::getOuterPoly(const LayerIndex& layer_nr) const +const Shape& PrimeTower::getOuterPoly(const LayerIndex& layer_nr) const { const LayerIndex absolute_layer_nr = layer_nr + Raft::getTotalExtraLayers(); if (absolute_layer_nr < outer_poly_base_.size()) @@ -617,7 +620,7 @@ const Polygons& PrimeTower::getOuterPoly(const LayerIndex& layer_nr) const } } -const Polygons& PrimeTower::getGroundPoly() const +const Shape& PrimeTower::getGroundPoly() const { return getOuterPoly(-Raft::getTotalExtraLayers()); } @@ -629,8 +632,7 @@ void PrimeTower::gotoStartLocation(LayerPlan& gcode_layer, const int extruder_nr int current_start_location_idx = ((((extruder_nr + 1) * gcode_layer.getLayerNr()) % number_of_prime_tower_start_locations_) + number_of_prime_tower_start_locations_) % number_of_prime_tower_start_locations_; - const ClosestPolygonPoint wipe_location = prime_tower_start_locations_[current_start_location_idx]; - + const ClosestPointPolygon wipe_location = prime_tower_start_locations_[current_start_location_idx]; const ExtruderTrain& train = Application::getInstance().current_slice_->scene.extruders[extruder_nr]; const coord_t inward_dist = train.settings_.get("machine_nozzle_size") * 3 / 2; const coord_t start_dist = train.settings_.get("machine_nozzle_size") * 2; diff --git a/src/SkeletalTrapezoidation.cpp b/src/SkeletalTrapezoidation.cpp index d31aa52ce9..46454c3eb6 100644 --- a/src/SkeletalTrapezoidation.cpp +++ b/src/SkeletalTrapezoidation.cpp @@ -3,7 +3,7 @@ #include "SkeletalTrapezoidation.h" -#include +#include #include #include #include @@ -370,7 +370,7 @@ void SkeletalTrapezoidation::computeSegmentCellRange( } SkeletalTrapezoidation::SkeletalTrapezoidation( - const Polygons& polys, + const Shape& polys, const BeadingStrategy& beading_strategy, AngleRadians transitioning_angle, coord_t discretization_step_size, @@ -392,7 +392,7 @@ SkeletalTrapezoidation::SkeletalTrapezoidation( constructFromPolygons(polys); } -void SkeletalTrapezoidation::constructFromPolygons(const Polygons& polys) +void SkeletalTrapezoidation::constructFromPolygons(const Shape& polys) { vd_edge_to_he_edge_.clear(); vd_node_to_he_node_.clear(); @@ -402,7 +402,7 @@ void SkeletalTrapezoidation::constructFromPolygons(const Polygons& polys) std::vector segments; for (size_t poly_idx = 0; poly_idx < polys.size(); poly_idx++) { - ConstPolygonRef poly = polys[poly_idx]; + const Polygon& poly = polys[poly_idx]; for (size_t point_idx = 0; point_idx < poly.size(); point_idx++) { segments.emplace_back(&polys, poly_idx, point_idx); diff --git a/src/SkirtBrim.cpp b/src/SkirtBrim.cpp index 74935df98a..ba399fb721 100644 --- a/src/SkirtBrim.cpp +++ b/src/SkirtBrim.cpp @@ -8,12 +8,14 @@ #include "Application.h" #include "ExtruderTrain.h" #include "Slice.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Shape.h" #include "settings/EnumSettings.h" #include "settings/types/Ratio.h" #include "sliceDataStorage.h" #include "support.h" -#include "utils/PolylineStitcher.h" -#include "utils/Simplify.h" //Simplifying the brim/skirt at every inset. +#include "utils/MixedPolylineStitcher.h" +#include "utils/Simplify.h" namespace cura { @@ -21,7 +23,7 @@ namespace cura SkirtBrim::SkirtBrim(SliceDataStorage& storage) : storage_(storage) , adhesion_type_(Application::getInstance().current_slice_->scene.current_mesh_group->settings.get("adhesion_type")) - , has_ooze_shield_(storage.oozeShield.size() > 0 && storage.oozeShield[0].size() > 0) + , has_ooze_shield_(storage.ooze_shield.size() > 0 && storage.ooze_shield[0].size() > 0) , has_draft_shield_(storage.draft_protection_shield.size() > 0) , extruders_(Application::getInstance().current_slice_->scene.extruders) , extruder_count_(extruders_.size()) @@ -69,7 +71,7 @@ SkirtBrim::SkirtBrim(SliceDataStorage& storage) } } -std::vector SkirtBrim::generateBrimOffsetPlan(std::vector& starting_outlines) +std::vector SkirtBrim::generateBrimOffsetPlan(std::vector& starting_outlines) { std::vector all_brim_offsets; @@ -122,14 +124,14 @@ std::vector SkirtBrim::generateBrimOffsetPlan(std::vector starting_outlines(extruder_count_); + std::vector starting_outlines(extruder_count_); std::vector all_brim_offsets = generateBrimOffsetPlan(starting_outlines); - std::vector allowed_areas_per_extruder = generateAllowedAreas(starting_outlines); + std::vector allowed_areas_per_extruder = generateAllowedAreas(starting_outlines); // Apply 'approximate convex hull' if the adhesion is skirt _after_ any skirt but also prime-tower-brim adhesion. // Otherwise, the now expanded convex hull covered areas will mess with that brim. Fortunately this does not mess // with the other area calculation above, since they are either itself a simple/convex shape or relevant for brim. - Polygons covered_area = storage_.getLayerOutlines( + Shape covered_area = storage_.getLayerOutlines( 0, /*include_support*/ true, /*include_prime_tower*/ adhesion_type_ == EPlatformAdhesion::SKIRT); @@ -158,18 +160,17 @@ void SkirtBrim::generate() const Settings& global_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; const coord_t maximum_resolution = global_settings.get("meshfix_maximum_resolution"); const coord_t maximum_deviation = global_settings.get("meshfix_maximum_deviation"); + constexpr coord_t max_area_dev = 0u; // No area deviation applied for (int extruder_nr = 0; extruder_nr < extruder_count_; extruder_nr++) { - for (SkirtBrimLine& line : storage_.skirt_brim[extruder_nr]) + for (MixedLinesSet& lines : storage_.skirt_brim[extruder_nr]) { - constexpr coord_t max_area_dev = 0u; // No area deviation applied - line.open_polylines = Simplify(maximum_resolution, maximum_deviation, max_area_dev).polyline(line.open_polylines); - line.closed_polygons = Simplify(maximum_resolution, maximum_deviation, max_area_dev).polygon(line.closed_polygons); + lines = Simplify(maximum_resolution, maximum_deviation, max_area_dev).polyline(lines); } } } -std::vector SkirtBrim::generatePrimaryBrim(std::vector& all_brim_offsets, Polygons& covered_area, std::vector& allowed_areas_per_extruder) +std::vector SkirtBrim::generatePrimaryBrim(std::vector& all_brim_offsets, Shape& covered_area, std::vector& allowed_areas_per_extruder) { std::vector total_length(extruder_count_, 0U); @@ -180,7 +181,7 @@ std::vector SkirtBrim::generatePrimaryBrim(std::vector& all_bri { storage_.skirt_brim[offset.extruder_nr_].resize(offset.inset_idx_ + 1); } - SkirtBrimLine& output_location = storage_.skirt_brim[offset.extruder_nr_][offset.inset_idx_]; + MixedLinesSet& output_location = storage_.skirt_brim[offset.extruder_nr_][offset.inset_idx_]; const coord_t added_length = generateOffset(offset, covered_area, allowed_areas_per_extruder, output_location); if (added_length == 0) @@ -213,26 +214,26 @@ std::vector SkirtBrim::generatePrimaryBrim(std::vector& all_bri return total_length; } -coord_t SkirtBrim::generateOffset(const Offset& offset, Polygons& covered_area, std::vector& allowed_areas_per_extruder, SkirtBrimLine& result) +coord_t SkirtBrim::generateOffset(const Offset& offset, Shape& covered_area, std::vector& allowed_areas_per_extruder, MixedLinesSet& result) { coord_t length_added; - Polygons brim; + Shape brim; const ExtruderConfig& extruder_config = extruders_configs_[offset.extruder_nr_]; - if (std::holds_alternative(offset.reference_outline_or_index_)) + if (std::holds_alternative(offset.reference_outline_or_index_)) { - Polygons* reference_outline = std::get(offset.reference_outline_or_index_); + Shape* reference_outline = std::get(offset.reference_outline_or_index_); const coord_t offset_value = offset.offset_value_; - for (ConstPolygonRef polygon : *reference_outline) + for (const Polygon& polygon : *reference_outline) { const double area = polygon.area(); if (area > 0 && offset.outside_) { - brim.add(polygon.offset(offset_value, ClipperLib::jtRound)); + brim.push_back(polygon.offset(offset_value, ClipperLib::jtRound)); } else if (area < 0 && offset.inside_) { - brim.add(polygon.offset(-offset_value, ClipperLib::jtRound)); + brim.push_back(polygon.offset(-offset_value, ClipperLib::jtRound)); } } } @@ -241,42 +242,34 @@ coord_t SkirtBrim::generateOffset(const Offset& offset, Polygons& covered_area, const int reference_idx = std::get(offset.reference_outline_or_index_); const coord_t offset_dist = extruder_config.line_width_; - Polygons local_brim; - auto closed_polygons_brim = storage_.skirt_brim[offset.extruder_nr_][reference_idx].closed_polygons.offsetPolyLine(offset_dist, ClipperLib::jtRound, true); - local_brim.add(closed_polygons_brim); - - auto open_polylines_brim = storage_.skirt_brim[offset.extruder_nr_][reference_idx].open_polylines.offsetPolyLine(offset_dist, ClipperLib::jtRound); - local_brim.add(open_polylines_brim); - local_brim.unionPolygons(); - - brim.add(local_brim); + brim.push_back(storage_.skirt_brim[offset.extruder_nr_][reference_idx].offset(offset_dist, ClipperLib::jtRound)); } // limit brim lines to allowed areas, stitch them and store them in the result brim = Simplify(Application::getInstance().current_slice_->scene.extruders[offset.extruder_nr_].settings_).polygon(brim); - brim.toPolylines(); - Polygons brim_lines = allowed_areas_per_extruder[offset.extruder_nr_].intersectionPolyLines(brim, false); - length_added = brim_lines.polyLineLength(); + OpenLinesSet brim_lines = allowed_areas_per_extruder[offset.extruder_nr_].intersection(brim, false); + length_added = brim_lines.length(); - Polygons newly_covered = brim_lines.offsetPolyLine(extruder_config.line_width_ / 2 + 10, ClipperLib::jtRound); + Shape newly_covered = brim_lines.offset(extruder_config.line_width_ / 2 + 10, ClipperLib::jtRound); const coord_t max_stitch_distance = extruder_config.line_width_; - PolylineStitcher::stitch(brim_lines, result.open_polylines, result.closed_polygons, max_stitch_distance); - - // clean up too small lines - for (size_t line_idx = 0; line_idx < result.open_polylines.size();) - { - PolygonRef line = result.open_polylines[line_idx]; - if (line.shorterThan(min_brim_line_length)) - { - result.open_polylines.remove(line_idx); - } - else - { - line_idx++; - } - } + MixedPolylineStitcher::stitch(brim_lines, result, max_stitch_distance); + + // clean up too small lines (only open ones, which was done historically but may be a mistake) + result.erase( + std::remove_if( + result.begin(), + result.end(), + [](const PolylinePtr& line) + { + if (const std::shared_ptr open_line = dynamic_pointer_cast(line)) + { + return open_line->shorterThan(min_brim_line_length); + } + return false; + }), + result.end()); // update allowed_areas_per_extruder covered_area = covered_area.unionPolygons(newly_covered.unionPolygons()); @@ -291,9 +284,9 @@ coord_t SkirtBrim::generateOffset(const Offset& offset, Polygons& covered_area, return length_added; } -Polygons SkirtBrim::getFirstLayerOutline(const int extruder_nr /* = -1 */) +Shape SkirtBrim::getFirstLayerOutline(const int extruder_nr /* = -1 */) { - Polygons first_layer_outline; + Shape first_layer_outline; Settings& global_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; int reference_extruder_nr = skirt_brim_extruder_nr_; assert(! (reference_extruder_nr == -1 && extruder_nr == -1) && "We should only request the outlines of all layers when the brim is being generated for only one material"); @@ -306,7 +299,7 @@ Polygons SkirtBrim::getFirstLayerOutline(const int extruder_nr /* = -1 */) const LayerIndex layer_nr = 0; if (adhesion_type_ == EPlatformAdhesion::SKIRT) { - first_layer_outline = Polygons(); + first_layer_outline = Shape(); int skirt_height = 0; for (const auto& extruder : Application::getInstance().current_slice_->scene.extruders) { @@ -324,10 +317,10 @@ Polygons SkirtBrim::getFirstLayerOutline(const int extruder_nr /* = -1 */) first_layer_outline = first_layer_outline.unionPolygons(storage_.getLayerOutlines(i_layer, include_support, include_prime_tower, true)); } - Polygons shields; + Shape shields; if (has_ooze_shield_) { - shields = storage_.oozeShield[0]; + shields = storage_.ooze_shield[0]; } if (has_draft_shield_) { @@ -365,26 +358,26 @@ Polygons SkirtBrim::getFirstLayerOutline(const int extruder_nr /* = -1 */) // |+-+| |+--+| // +---+ +----+ const coord_t primary_extruder_skirt_brim_line_width = reference_extruder_config.line_width_; - Polygons model_brim_covered_area; + Shape model_brim_covered_area; // always leave a gap of an even number of brim lines, so that it fits if it's generating brim from both sides const coord_t offset = primary_extruder_skirt_brim_line_width * (primary_line_count + primary_line_count % 2); - for (ConstPolygonRef polygon : first_layer_outline) + for (const Polygon& polygon : first_layer_outline) { // Compute the fringe that the brim is going to cover around the model - Polygons outset; - Polygons inset; + Shape outset; + Shape inset; double area = polygon.area(); if (area > 0 && reference_extruder_config.outside_polys_) { outset = polygon.offset(offset, ClipperLib::jtRound); - inset.add(polygon); + inset.push_back(polygon); } else if (area < 0 && reference_extruder_config.inside_polys_) { - outset.add(polygon); + outset.push_back(polygon); inset = polygon.offset(-offset, ClipperLib::jtRound); } @@ -401,10 +394,10 @@ Polygons SkirtBrim::getFirstLayerOutline(const int extruder_nr /* = -1 */) for (const SupportInfillPart& support_infill_part : support_layer.support_infill_parts) { - first_layer_outline.add(support_infill_part.outline_); + first_layer_outline.push_back(support_infill_part.outline_); } - first_layer_outline.add(support_layer.support_bottom); - first_layer_outline.add(support_layer.support_roof); + first_layer_outline.push_back(support_layer.support_bottom); + first_layer_outline.push_back(support_layer.support_roof); } } constexpr coord_t join_distance = 20; @@ -419,7 +412,7 @@ Polygons SkirtBrim::getFirstLayerOutline(const int extruder_nr /* = -1 */) return first_layer_outline; } -void SkirtBrim::generateShieldBrim(Polygons& brim_covered_area, std::vector& allowed_areas_per_extruder) +void SkirtBrim::generateShieldBrim(Shape& brim_covered_area, std::vector& allowed_areas_per_extruder) { int extruder_nr = skirt_brim_extruder_nr_; if (extruder_nr < 0) @@ -446,10 +439,10 @@ void SkirtBrim::generateShieldBrim(Polygons& brim_covered_area, std::vector 0) { shield_brim = shield_brim.offset(-primary_extruder_skirt_brim_line_width); - storage_.skirt_brim[extruder_nr].back().closed_polygons.add( - shield_brim); // throw all polygons for the shileds onto one heap; because the brim lines are generated from both sides the order will not be important + storage_.skirt_brim[extruder_nr].back().push_back(shield_brim); // throw all polygons for the shileds onto one heap; because the brim lines are + // generated from both sides the order will not be important } } @@ -479,27 +472,27 @@ void SkirtBrim::generateShieldBrim(Polygons& brim_covered_area, std::vector& allowed_areas_per_extruder, std::vector& total_length) +void SkirtBrim::generateSecondarySkirtBrim(Shape& covered_area, std::vector& allowed_areas_per_extruder, std::vector& total_length) { constexpr coord_t bogus_total_offset = 0u; // Doesn't matter. The offsets won't be sorted here. constexpr bool is_last = false; // Doesn't matter. Isn't used in the algorithm below. for (int extruder_nr = 0; extruder_nr < extruder_count_; extruder_nr++) { bool first = true; - Polygons reference_outline = covered_area; + Shape reference_outline = covered_area; const ExtruderConfig& extruder_config = extruders_configs_[extruder_nr]; while (total_length[extruder_nr] < extruder_config.skirt_brim_minimal_length_) { @@ -528,7 +521,7 @@ void SkirtBrim::generateSecondarySkirtBrim(Polygons& covered_area, std::vector

SkirtBrim::generateAllowedAreas(const std::vector& starting_outlines) const +std::vector SkirtBrim::generateAllowedAreas(const std::vector& starting_outlines) const { constexpr LayerIndex layer_nr = 0; // For each extruder, pre-compute the areas covered by models/supports/prime tower struct ExtruderOutlines { - Polygons models_outlines; - Polygons supports_outlines; + Shape models_outlines; + Shape supports_outlines; }; std::vector covered_area_by_extruder; @@ -583,7 +576,7 @@ std::vector SkirtBrim::generateAllowedAreas(const std::vector allowed_areas_per_extruder(extruder_count_); + std::vector allowed_areas_per_extruder(extruder_count_); for (size_t extruder_nr = 0; extruder_nr < extruder_count_; extruder_nr++) { const ExtruderConfig& extruder_config = extruders_configs_[extruder_nr]; @@ -594,7 +587,7 @@ std::vector SkirtBrim::generateAllowedAreas(const std::vector SkirtBrim::generateAllowedAreas(const std::vector SkirtBrim::generateAllowedAreas(const std::vector -brim_line_width * brim_line_width * brim_area_minimum_hole_size_multiplier) { - brim_line.remove(n--); + brim_line.removeAt(n--); } } - storage_.support_brim.add(brim_line); + const bool brim_line_empty = brim_line.empty(); // Store before moving + storage_.support_brim.push_back(std::move(brim_line)); // In case of adhesion::NONE length of support brim is only the length of the brims formed for the support - const coord_t length = (adhesion_type_ == EPlatformAdhesion::NONE) ? skirt_brim_length : skirt_brim_length + storage_.support_brim.polygonLength(); + const coord_t length = (adhesion_type_ == EPlatformAdhesion::NONE) ? skirt_brim_length : skirt_brim_length + storage_.support_brim.length(); if (skirt_brim_number + 1 >= line_count && length > 0 && length < minimal_length) // Make brim or skirt have more lines when total length is too small. { line_count++; } - if (brim_line.empty()) + if (brim_line_empty) { // the fist layer of support is fully filled with brim break; } diff --git a/src/SupportInfillPart.cpp b/src/SupportInfillPart.cpp index fb504f5fb5..fc8e788a81 100644 --- a/src/SupportInfillPart.cpp +++ b/src/SupportInfillPart.cpp @@ -8,7 +8,7 @@ using namespace cura; -SupportInfillPart::SupportInfillPart(const PolygonsPart& outline, coord_t support_line_width, bool use_fractional_config, int inset_count_to_generate, coord_t custom_line_distance) +SupportInfillPart::SupportInfillPart(const SingleShape& outline, coord_t support_line_width, bool use_fractional_config, int inset_count_to_generate, coord_t custom_line_distance) : outline_(outline) , outline_boundary_box_(outline) , support_line_width_(support_line_width) diff --git a/src/TopSurface.cpp b/src/TopSurface.cpp index a717ccb3bb..8203ccb2ce 100644 --- a/src/TopSurface.cpp +++ b/src/TopSurface.cpp @@ -5,6 +5,7 @@ #include "ExtruderTrain.h" #include "LayerPlan.h" +#include "geometry/OpenPolyline.h" #include "infill.h" #include "sliceDataStorage.h" @@ -19,7 +20,7 @@ TopSurface::TopSurface() void TopSurface::setAreasFromMeshAndLayerNumber(SliceMeshStorage& mesh, size_t layer_number) { // The top surface is all parts of the mesh where there's no mesh above it, so find the layer above it first. - Polygons mesh_above; + Shape mesh_above; if (layer_number < mesh.layers.size() - 1) { mesh_above = mesh.layers[layer_number + 1].getOutlines(); @@ -86,7 +87,7 @@ bool TopSurface::ironing(const SliceDataStorage& storage, const SliceMeshStorage // Align the edge of the ironing line with the edge of the outer wall ironing_inset -= ironing_flow * line_width / 2; } - Polygons ironed_areas = areas.offset(ironing_inset); + Shape ironed_areas = areas.offset(ironing_inset); Infill infill_generator( pattern, @@ -107,8 +108,8 @@ bool TopSurface::ironing(const SliceDataStorage& storage, const SliceMeshStorage infill_origin, skip_line_stitching); std::vector ironing_paths; - Polygons ironing_polygons; - Polygons ironing_lines; + Shape ironing_polygons; + OpenLinesSet ironing_lines; infill_generator.generate(ironing_paths, ironing_polygons, ironing_lines, mesh.settings, layer.getLayerNr(), SectionType::IRONING); if (ironing_polygons.empty() && ironing_lines.empty() && ironing_paths.empty()) @@ -157,7 +158,7 @@ bool TopSurface::ironing(const SliceDataStorage& storage, const SliceMeshStorage { const coord_t max_adjacent_distance = line_spacing * 1.1; // Lines are considered adjacent - meaning they need to be printed in monotonic order - if spaced 1 line apart, with 10% extra play. - layer.addLinesMonotonic(Polygons(), ironing_lines, line_config, SpaceFillType::PolyLines, AngleRadians(direction), max_adjacent_distance); + layer.addLinesMonotonic(Shape(), ironing_lines, line_config, SpaceFillType::PolyLines, AngleRadians(direction), max_adjacent_distance); } added = true; } diff --git a/src/TreeModelVolumes.cpp b/src/TreeModelVolumes.cpp index fa8fe4bedf..a6d12804a6 100644 --- a/src/TreeModelVolumes.cpp +++ b/src/TreeModelVolumes.cpp @@ -26,7 +26,7 @@ TreeModelVolumes::TreeModelVolumes( size_t current_mesh_idx, double progress_multiplier, double progress_offset, - const std::vector& additional_excluded_areas) + const std::vector& additional_excluded_areas) : max_move_{ std::max(max_move - 2, coord_t(0)) } , // -2 to avoid rounding errors max_move_slow_{ std::max(max_move_slow - 2, coord_t(0)) } @@ -37,7 +37,7 @@ TreeModelVolumes::TreeModelVolumes( , machine_border_{ calculateMachineBorderCollision(storage.getMachineBorder()) } , machine_area_{ storage.getMachineBorder() } { - anti_overhang_ = std::vector(storage.support.supportLayers.size(), Polygons()); + anti_overhang_ = std::vector(storage.support.supportLayers.size(), Shape()); std::unordered_map mesh_to_layeroutline_idx; // Get, for all participating meshes, simplification settings, and support settings that can be set per mesh. @@ -64,7 +64,7 @@ TreeModelVolumes::TreeModelVolumes( if (! added) { mesh_to_layeroutline_idx[mesh_idx] = layer_outlines_.size(); - layer_outlines_.emplace_back(mesh.settings, std::vector(storage.support.supportLayers.size(), Polygons())); + layer_outlines_.emplace_back(mesh.settings, std::vector(storage.support.supportLayers.size(), Shape())); } } @@ -116,8 +116,8 @@ TreeModelVolumes::TreeModelVolumes( { return; // Can't break as parallel_for wont allow it, this is equivalent to a continue. } - Polygons outline = extractOutlineFromMesh(mesh_l, layer_idx); - layer_outlines_[mesh_to_layeroutline_idx[mesh_idx_l]].second[layer_idx].add(outline); + Shape outline = extractOutlineFromMesh(mesh_l, layer_idx); + layer_outlines_[mesh_to_layeroutline_idx[mesh_idx_l]].second[layer_idx].push_back(outline); }); } // Merge all the layer outlines together. @@ -140,17 +140,17 @@ TreeModelVolumes::TreeModelVolumes( { if (layer_idx < coord_t(additional_excluded_areas.size())) { - anti_overhang_[layer_idx].add(additional_excluded_areas[layer_idx]); + anti_overhang_[layer_idx].push_back(additional_excluded_areas[layer_idx]); } if (SUPPORT_TREE_AVOID_SUPPORT_BLOCKER) { - anti_overhang_[layer_idx].add(storage.support.supportLayers[layer_idx].anti_overhang); + anti_overhang_[layer_idx].push_back(storage.support.supportLayers[layer_idx].anti_overhang); } if (storage.primeTower.enabled_) { - anti_overhang_[layer_idx].add(storage.primeTower.getGroundPoly()); + anti_overhang_[layer_idx].push_back(storage.primeTower.getGroundPoly()); } anti_overhang_[layer_idx] = anti_overhang_[layer_idx].unionPolygons(); }); @@ -328,10 +328,10 @@ void TreeModelVolumes::precalculate(coord_t max_layer) dur_col_avo); } -const Polygons& TreeModelVolumes::getCollision(coord_t radius, LayerIndex layer_idx, bool min_xy_dist) +const Shape& TreeModelVolumes::getCollision(coord_t radius, LayerIndex layer_idx, bool min_xy_dist) { const coord_t orig_radius = radius; - std::optional> result; + std::optional> result; if (! min_xy_dist) { radius += current_min_xy_dist_delta_; @@ -361,10 +361,10 @@ const Polygons& TreeModelVolumes::getCollision(coord_t radius, LayerIndex layer_ return getCollision(orig_radius, layer_idx, min_xy_dist); } -const Polygons& TreeModelVolumes::getCollisionHolefree(coord_t radius, LayerIndex layer_idx, bool min_xy_dist) +const Shape& TreeModelVolumes::getCollisionHolefree(coord_t radius, LayerIndex layer_idx, bool min_xy_dist) { const coord_t orig_radius = radius; - std::optional> result; + std::optional> result; if (! min_xy_dist) { radius += current_min_xy_dist_delta_; @@ -391,7 +391,7 @@ const Polygons& TreeModelVolumes::getCollisionHolefree(coord_t radius, LayerInde return getCollisionHolefree(orig_radius, layer_idx, min_xy_dist); } -const Polygons& TreeModelVolumes::getAccumulatedPlaceable0(LayerIndex layer_idx) +const Shape& TreeModelVolumes::getAccumulatedPlaceable0(LayerIndex layer_idx) { { std::lock_guard critical_section_support_max_layer_nr(*critical_accumulated_placeables_cache_radius_0_); @@ -404,7 +404,7 @@ const Polygons& TreeModelVolumes::getAccumulatedPlaceable0(LayerIndex layer_idx) return getAccumulatedPlaceable0(layer_idx); } -const Polygons& TreeModelVolumes::getAvoidance(coord_t radius, LayerIndex layer_idx, AvoidanceType type, bool to_model, bool min_xy_dist) +const Shape& TreeModelVolumes::getAvoidance(coord_t radius, LayerIndex layer_idx, AvoidanceType type, bool to_model, bool min_xy_dist) { if (layer_idx == 0) // What on the layer directly above buildplate do i have to avoid to reach the buildplate ... { @@ -413,7 +413,7 @@ const Polygons& TreeModelVolumes::getAvoidance(coord_t radius, LayerIndex layer_ const coord_t orig_radius = radius; - std::optional> result; + std::optional> result; radius += (min_xy_dist ? 0 : current_min_xy_dist_delta_); radius = ceilRadius(radius); @@ -425,7 +425,7 @@ const Polygons& TreeModelVolumes::getAvoidance(coord_t radius, LayerIndex layer_ const RadiusLayerPair key{ radius, layer_idx }; - std::unordered_map* cache_ptr = nullptr; + std::unordered_map* cache_ptr = nullptr; std::mutex* mutex_ptr = nullptr; switch (type) { @@ -489,9 +489,9 @@ const Polygons& TreeModelVolumes::getAvoidance(coord_t radius, LayerIndex layer_ return getAvoidance(orig_radius, layer_idx, type, to_model, min_xy_dist); // retrive failed and correct result was calculated. Now it has to be retrived. } -const Polygons& TreeModelVolumes::getPlaceableAreas(coord_t radius, LayerIndex layer_idx) +const Shape& TreeModelVolumes::getPlaceableAreas(coord_t radius, LayerIndex layer_idx) { - std::optional> result; + std::optional> result; const coord_t orig_radius = radius; radius = ceilRadius(radius); RadiusLayerPair key{ radius, layer_idx }; @@ -520,7 +520,7 @@ const Polygons& TreeModelVolumes::getPlaceableAreas(coord_t radius, LayerIndex l } -const Polygons& TreeModelVolumes::getWallRestriction(coord_t radius, LayerIndex layer_idx, bool min_xy_dist) +const Shape& TreeModelVolumes::getWallRestriction(coord_t radius, LayerIndex layer_idx, bool min_xy_dist) { if (layer_idx == 0) // Should never be requested as there will be no going below layer 0 ..., but just to be sure some semi-sane catch. Alternative would be empty Polygon. { @@ -530,12 +530,12 @@ const Polygons& TreeModelVolumes::getWallRestriction(coord_t radius, LayerIndex const coord_t orig_radius = radius; min_xy_dist = min_xy_dist && current_min_xy_dist_delta_ > 0; - std::optional> result; + std::optional> result; radius = ceilRadius(radius); const RadiusLayerPair key{ radius, layer_idx }; - std::unordered_map* cache_ptr = min_xy_dist ? &wall_restrictions_cache_min_ : &wall_restrictions_cache_; + std::unordered_map* cache_ptr = min_xy_dist ? &wall_restrictions_cache_min_ : &wall_restrictions_cache_; { std::lock_guard critical_section(min_xy_dist ? *critical_wall_restrictions_cache_min_ : *critical_wall_restrictions_cache_); result = getArea(*cache_ptr, key); @@ -568,23 +568,23 @@ bool TreeModelVolumes::checkSettingsEquality(const Settings& me, const Settings& return TreeSupportSettings(me) == TreeSupportSettings(other); } -Polygons TreeModelVolumes::extractOutlineFromMesh(const SliceMeshStorage& mesh, LayerIndex layer_idx) const +Shape TreeModelVolumes::extractOutlineFromMesh(const SliceMeshStorage& mesh, LayerIndex layer_idx) const { // Similar to SliceDataStorage.getLayerOutlines but only for one mesh instead of for all of them. constexpr bool external_polys_only = false; - Polygons total; + Shape total; if (mesh.settings.get("infill_mesh") || mesh.settings.get("anti_overhang_mesh")) { - return Polygons(); + return Shape(); } const SliceLayer& layer = mesh.layers[layer_idx]; layer.getOutlines(total, external_polys_only); if (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL) { - total = total.unionPolygons(layer.openPolyLines.offsetPolyLine(FUDGE_LENGTH * 2)); + total = total.unionPolygons(layer.open_polylines.offset(FUDGE_LENGTH * 2)); } const coord_t maximum_resolution = mesh.settings.get("meshfix_maximum_resolution"); const coord_t maximum_deviation = mesh.settings.get("meshfix_maximum_deviation"); @@ -592,7 +592,7 @@ Polygons TreeModelVolumes::extractOutlineFromMesh(const SliceMeshStorage& mesh, return Simplify(maximum_resolution, maximum_deviation, maximum_area_deviation).polygon(total); } -LayerIndex TreeModelVolumes::getMaxCalculatedLayer(coord_t radius, const std::unordered_map& map) const +LayerIndex TreeModelVolumes::getMaxCalculatedLayer(coord_t radius, const std::unordered_map& map) const { LayerIndex max_layer = -1; @@ -620,12 +620,12 @@ void TreeModelVolumes::calculateCollision(const std::deque& key { const coord_t radius = keys[i].first; RadiusLayerPair key(radius, 0); - std::unordered_map data_outer; - std::unordered_map data_placeable_outer; + std::unordered_map data_outer; + std::unordered_map data_placeable_outer; for (const auto outline_idx : ranges::views::iota(0UL, layer_outlines_.size())) { - std::unordered_map data; - std::unordered_map data_placeable; + std::unordered_map data; + std::unordered_map data_placeable; const coord_t layer_height = layer_outlines_[outline_idx].first.get("layer_height"); const bool support_rests_on_this_model = layer_outlines_[outline_idx].first.get("support_type") == ESupportType::EVERYWHERE; @@ -652,15 +652,15 @@ void TreeModelVolumes::calculateCollision(const std::deque& key for (const auto layer_idx : ranges::views::iota(min_layer_bottom, max_required_layer + 1)) { key.second = layer_idx; - Polygons collision_areas = machine_border_; + Shape collision_areas = machine_border_; if (size_t(layer_idx) < layer_outlines_[outline_idx].second.size()) { - collision_areas.add(layer_outlines_[outline_idx].second[layer_idx]); + collision_areas.push_back(layer_outlines_[outline_idx].second[layer_idx]); } collision_areas = collision_areas.offset( radius + xy_distance); // jtRound is not needed here, as the overshoot can not cause errors in the algorithm, because no assumptions are made about the model. - data[key].add(collision_areas); // if a key does not exist when it is accessed it is added! + data[key].push_back(collision_areas); // if a key does not exist when it is accessed it is added! } // Add layers below, to ensure correct support_bottom_distance. Also save placeable areas of radius 0, if required for this mesh. @@ -669,18 +669,18 @@ void TreeModelVolumes::calculateCollision(const std::deque& key key.second = layer_idx; for (size_t layer_offset = 1; layer_offset <= z_distance_bottom_layers && layer_idx - coord_t(layer_offset) > min_layer_bottom; layer_offset++) { - data[key].add(data[RadiusLayerPair(radius, layer_idx - layer_offset)]); + data[key].push_back(data[RadiusLayerPair(radius, layer_idx - layer_offset)]); } // Placeable areas also have to be calculated when a collision has to be calculated if called outside of precalculate to prevent an infinite loop when they are // invalidly requested... if ((support_rests_on_this_model || precalculation_finished_ || ! precalculated_) && radius == 0 && layer_idx < coord_t(1 + keys[i].second)) { data[key] = data[key].unionPolygons(); - Polygons above = data[RadiusLayerPair(radius, layer_idx + 1)]; - above = above.unionPolygons(max_anti_overhang_layer >= layer_idx + 1 ? anti_overhang_[layer_idx] : Polygons()); + Shape above = data[RadiusLayerPair(radius, layer_idx + 1)]; + above = above.unionPolygons(max_anti_overhang_layer >= layer_idx + 1 ? anti_overhang_[layer_idx] : Shape()); // Empty polygons on condition: Just to be sure the area is correctly unioned as otherwise difference may behave unexpectedly. - Polygons placeable = data[key].unionPolygons().difference(above); + Shape placeable = data[key].unionPolygons().difference(above); data_placeable[RadiusLayerPair(radius, layer_idx + 1)] = data_placeable[RadiusLayerPair(radius, layer_idx + 1)].unionPolygons(placeable); } } @@ -719,9 +719,9 @@ void TreeModelVolumes::calculateCollision(const std::deque& key const coord_t required_range_x = coord_t(xy_distance - ((layer_offset - (z_distance_top_layers == 1 ? 0.5 : 0)) * xy_distance / z_distance_top_layers)); // ^^^ The conditional -0.5 ensures that plastic can never touch on the diagonal downward when the z_distance_top_layers = 1. // It is assumed to be better to not support an overhang<90� than to risk fusing to it. - data[key].add(layer_outlines_[outline_idx].second[layer_idx + layer_offset].offset(radius + required_range_x)); + data[key].push_back(layer_outlines_[outline_idx].second[layer_idx + layer_offset].offset(radius + required_range_x)); } - data[key] = data[key].unionPolygons(max_anti_overhang_layer >= layer_idx ? anti_overhang_[layer_idx].offset(radius) : Polygons()); + data[key] = data[key].unionPolygons(max_anti_overhang_layer >= layer_idx ? anti_overhang_[layer_idx].offset(radius) : Shape()); } for (const auto layer_idx : ranges::views::iota(static_cast(keys[i].second) + 1UL, max_required_layer + 1UL) | ranges::views::reverse) @@ -781,13 +781,13 @@ void TreeModelVolumes::calculateCollisionHolefree(const std::deque data; + std::unordered_map data; for (RadiusLayerPair key : keys) { // Logically increase the collision by increase_until_radius const coord_t radius = key.first; const coord_t increase_radius_ceil = ceilRadius(increase_until_radius_, false) - ceilRadius(radius, true); - Polygons col = getCollision(increase_until_radius_, layer_idx, false).offset(EPSILON - increase_radius_ceil, ClipperLib::jtRound).unionPolygons(); + Shape col = getCollision(increase_until_radius_, layer_idx, false).offset(EPSILON - increase_radius_ceil, ClipperLib::jtRound).unionPolygons(); // ^^^ That last 'unionPolygons' is important as otherwise holes(in form of lines that will increase to holes in a later step) can get unioned onto the area. col = simplifier_.polygon(col); data[RadiusLayerPair(radius, layer_idx)] = col; @@ -818,12 +818,12 @@ void TreeModelVolumes::calculateAccumulatedPlaceable0(const LayerIndex max_layer spdlog::debug("Requested calculation for value already calculated ?"); return; } - Polygons accumulated_placeable_0 + Shape accumulated_placeable_0 = start_layer == 1 ? machine_area_ : getAccumulatedPlaceable0(start_layer - 1).offset(FUDGE_LENGTH + (current_min_xy_dist_ + current_min_xy_dist_delta_)); // ^^^ The calculation here is done on the areas that are increased by xy_distance, but the result is saved without xy_distance, // so here it "restores" the previous state to continue calculating from about where it ended. // It would be better to ensure placeable areas of radius 0 do not include the xy distance, and removing the code compensating for it here and in calculatePlaceables. - std::vector> data(max_layer + 1, std::pair(-1, Polygons())); + std::vector> data(max_layer + 1, std::pair(-1, Shape())); for (LayerIndex layer = start_layer; layer <= max_layer; layer++) { @@ -867,26 +867,26 @@ void TreeModelVolumes::calculateCollisionAvoidance(const std::deque> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Polygons())); + std::vector> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Shape())); RadiusLayerPair key(radius, 0); - Polygons latest_avoidance = getAvoidance(radius, start_layer - 1, AvoidanceType::COLLISION, true, true); + Shape latest_avoidance = getAvoidance(radius, start_layer - 1, AvoidanceType::COLLISION, true, true); for (const LayerIndex layer : ranges::views::iota(static_cast(start_layer), max_required_layer + 1UL)) { key.second = layer; - Polygons col = getCollision(radius, layer, true); + Shape col = getCollision(radius, layer, true); latest_avoidance = safeOffset(latest_avoidance, -max_move_, ClipperLib::jtRound, -max_step_move, col); - Polygons placeable0RadiusCompensated = getAccumulatedPlaceable0(layer).offset(-std::max(radius, increase_until_radius_), ClipperLib::jtRound); + Shape placeable0RadiusCompensated = getAccumulatedPlaceable0(layer).offset(-std::max(radius, increase_until_radius_), ClipperLib::jtRound); latest_avoidance = latest_avoidance.difference(placeable0RadiusCompensated).unionPolygons(getCollision(radius, layer, true)); - Polygons next_latest_avoidance = simplifier_.polygon(latest_avoidance); + Shape next_latest_avoidance = simplifier_.polygon(latest_avoidance); latest_avoidance = next_latest_avoidance.unionPolygons(latest_avoidance); // ^^^ Ensure the simplification only causes the avoidance to become larger. // If the deviation of the simplification causes the avoidance to become smaller than it should be it can cause issues, if it is larger the worst case is that the // xy distance is effectively increased by deviation. If there would be an option to ensure the resulting polygon only gets larger by simplifying, it should improve // performance further. - data[layer] = std::pair(key, latest_avoidance); + data[layer] = std::pair(key, latest_avoidance); } { @@ -899,11 +899,11 @@ void TreeModelVolumes::calculateCollisionAvoidance(const std::deque= 0); - Polygons ret = me; + Shape ret = me; for (size_t i = 0; i < steps; ++i) { @@ -944,7 +944,7 @@ void TreeModelVolumes::calculateAvoidance(const std::deque& key const coord_t offset_speed = slow ? max_move_slow_ : max_move_; const coord_t max_step_move = std::max(1.9 * radius, current_min_xy_dist_ * 1.9); RadiusLayerPair key(radius, 0); - Polygons latest_avoidance; + Shape latest_avoidance; LayerIndex start_layer; { std::lock_guard critical_section(*(slow ? critical_avoidance_cache_slow_ : holefree ? critical_avoidance_cache_holefree_ : critical_avoidance_cache_)); @@ -956,7 +956,7 @@ void TreeModelVolumes::calculateAvoidance(const std::deque& key return; } start_layer = std::max(start_layer, LayerIndex(1)); // Ensure StartLayer is at least 1 as if no avoidance was calculated getMaxCalculatedLayer returns -1 - std::vector> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Polygons())); + std::vector> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Shape())); latest_avoidance = getAvoidance(radius, start_layer - 1, type, false, true); // minDist as the delta was already added, also avoidance for layer 0 will return the collision. @@ -965,7 +965,7 @@ void TreeModelVolumes::calculateAvoidance(const std::deque& key for (const LayerIndex layer : ranges::views::iota(static_cast(start_layer), max_required_layer + 1UL)) { key.second = layer; - Polygons col; + Shape col; if ((slow && radius < increase_until_radius_ + current_min_xy_dist_delta_) || holefree) { col = getCollisionHolefree(radius, layer, true); @@ -976,13 +976,13 @@ void TreeModelVolumes::calculateAvoidance(const std::deque& key } latest_avoidance = safeOffset(latest_avoidance, -offset_speed, ClipperLib::jtRound, -max_step_move, col); - Polygons next_latest_avoidance = simplifier_.polygon(latest_avoidance); + Shape next_latest_avoidance = simplifier_.polygon(latest_avoidance); latest_avoidance = next_latest_avoidance.unionPolygons(latest_avoidance); // ^^^ Ensure the simplification only causes the avoidance to become larger. // If the deviation of the simplification causes the avoidance to become smaller than it should be it can cause issues, if it is larger the worst case is that the // xy distance is effectively increased by deviation. If there would be an option to ensure the resulting polygon only gets larger by simplifying, it should improve // performance further. - data[layer] = std::pair(key, latest_avoidance); + data[layer] = std::pair(key, latest_avoidance); } { @@ -1013,7 +1013,7 @@ void TreeModelVolumes::calculatePlaceables(const std::deque& ke { const coord_t radius = keys[key_idx].first; const LayerIndex max_required_layer = keys[key_idx].second; - std::vector> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Polygons())); + std::vector> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Shape())); RadiusLayerPair key(radius, 0); LayerIndex start_layer; @@ -1029,17 +1029,17 @@ void TreeModelVolumes::calculatePlaceables(const std::deque& ke if (start_layer == 0) { - data[0] = std::pair(key, machine_border_.difference(getCollision(radius, 0, true))); + data[0] = std::pair(key, machine_border_.difference(getCollision(radius, 0, true))); start_layer = 1; } for (const LayerIndex layer : ranges::views::iota(static_cast(start_layer), max_required_layer + 1)) { key.second = layer; - Polygons placeable = getPlaceableAreas(0, layer); + Shape placeable = getPlaceableAreas(0, layer); placeable = simplifier_.polygon(placeable); // it is faster to do this here in each thread than once in calculateCollision. placeable = placeable.offset(-(radius + (current_min_xy_dist_ + current_min_xy_dist_delta_))).unionPolygons(); - data[layer] = std::pair(key, placeable); + data[layer] = std::pair(key, placeable); } { @@ -1086,8 +1086,8 @@ void TreeModelVolumes::calculateAvoidanceToModel(const std::deque> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Polygons())); + Shape latest_avoidance; + std::vector> data(max_required_layer + 1, std::pair(RadiusLayerPair(radius, -1), Shape())); RadiusLayerPair key(radius, 0); LayerIndex start_layer; @@ -1113,7 +1113,7 @@ void TreeModelVolumes::calculateAvoidanceToModel(const std::deque(start_layer), max_required_layer + 1)) { key.second = layer; - Polygons col = getCollision(radius, layer, true); + Shape col = getCollision(radius, layer, true); if ((slow && radius < increase_until_radius_ + current_min_xy_dist_delta_) || holefree) { @@ -1125,13 +1125,13 @@ void TreeModelVolumes::calculateAvoidanceToModel(const std::deque(key, latest_avoidance); + data[layer] = std::pair(key, latest_avoidance); } { @@ -1202,8 +1202,8 @@ void TreeModelVolumes::calculateWallRestrictions(const std::deque data; - std::unordered_map data_min; + std::unordered_map data; + std::unordered_map data_min; { std::lock_guard critical_section(*critical_wall_restrictions_cache_); @@ -1218,12 +1218,12 @@ void TreeModelVolumes::calculateWallRestrictions(const std::deque 0) { - Polygons wall_restriction_min = simplifier_.polygon(getCollision(0, layer_idx, true).intersection(getCollision(radius, layer_idx_below, true))); + Shape wall_restriction_min = simplifier_.polygon(getCollision(0, layer_idx, true).intersection(getCollision(radius, layer_idx_below, true))); data_min.emplace(key, wall_restriction_min); } } @@ -1271,22 +1271,22 @@ coord_t TreeModelVolumes::ceilRadius(coord_t radius) const } template -const std::optional> TreeModelVolumes::getArea(const std::unordered_map& cache, const KEY key) const +const std::optional> TreeModelVolumes::getArea(const std::unordered_map& cache, const KEY key) const { const auto it = cache.find(key); if (it != cache.end()) { - return std::optional>{ it->second }; + return std::optional>{ it->second }; } else { - return std::optional>(); + return std::optional>(); } } -Polygons TreeModelVolumes::calculateMachineBorderCollision(const Polygons&& machine_border) +Shape TreeModelVolumes::calculateMachineBorderCollision(const Shape&& machine_border) { - Polygons machine_volume_border = machine_border.offset(MM2INT(1000.0)); // Put a border of 1 meter around the print volume so that we don't collide. + Shape machine_volume_border = machine_border.offset(MM2INT(1000.0)); // Put a border of 1 meter around the print volume so that we don't collide. machine_volume_border = machine_volume_border.difference(machine_border); // Subtract the actual volume from the collision area. return machine_volume_border; } diff --git a/src/TreeSupport.cpp b/src/TreeSupport.cpp index 2b8a9e03a1..21a34796de 100644 --- a/src/TreeSupport.cpp +++ b/src/TreeSupport.cpp @@ -117,11 +117,11 @@ void TreeSupport::generateSupportAreas(SliceDataStorage& storage) storage.support.supportLayers .size()); // Value is the area where support may be placed. As this is calculated in CreateLayerPathing it is saved and reused in drawAreas. - additional_required_support_area = std::vector(storage.support.supportLayers.size(), Polygons()); + additional_required_support_area = std::vector(storage.support.supportLayers.size(), Shape()); spdlog::info("Processing support tree mesh group {} of {} containing {} meshes.", counter + 1, grouped_meshes.size(), grouped_meshes[counter].second.size()); - std::vector exclude(storage.support.supportLayers.size()); + std::vector exclude(storage.support.supportLayers.size()); auto t_start = std::chrono::high_resolution_clock::now(); // get all already existing support areas and exclude them @@ -130,12 +130,12 @@ void TreeSupport::generateSupportAreas(SliceDataStorage& storage) LayerIndex(storage.support.supportLayers.size()), [&](const LayerIndex layer_idx) { - Polygons exlude_at_layer; - exlude_at_layer.add(storage.support.supportLayers[layer_idx].support_bottom); - exlude_at_layer.add(storage.support.supportLayers[layer_idx].support_roof); + Shape exlude_at_layer; + exlude_at_layer.push_back(storage.support.supportLayers[layer_idx].support_bottom); + exlude_at_layer.push_back(storage.support.supportLayers[layer_idx].support_roof); for (auto part : storage.support.supportLayers[layer_idx].support_infill_parts) { - exlude_at_layer.add(part.outline_); + exlude_at_layer.push_back(part.outline_); } exclude[layer_idx] = exlude_at_layer.unionPolygons(); scripta::log("tree_support_exclude", exclude[layer_idx], SectionType::SUPPORT, layer_idx); @@ -232,7 +232,7 @@ LayerIndex TreeSupport::precalculate(const SliceDataStorage& storage, std::vecto for (const auto layer_idx : ranges::views::iota(1UL, mesh.overhang_areas.size() - z_distance_top_layers) | ranges::views::reverse) { // Look for max relevant layer. - const Polygons& overhang = mesh.overhang_areas[layer_idx + z_distance_top_layers]; + const Shape& overhang = mesh.overhang_areas[layer_idx + z_distance_top_layers]; if (! overhang.empty()) { if (layer_idx > max_layer) // iterates over multiple meshes @@ -337,28 +337,27 @@ void TreeSupport::mergeHelper( continue; } - Polygons relevant_infl; - Polygons relevant_redu; + Shape relevant_infl; + Shape relevant_redu; if (merging_to_bp) { relevant_infl = to_bp_areas.count(influence.first) ? to_bp_areas.at(influence.first) - : Polygons(); // influence.first is a new element => not required to check if it was changed + : Shape(); // influence.first is a new element => not required to check if it was changed relevant_redu = insert_bp_areas.count(reduced_check.first) ? insert_bp_areas[reduced_check.first] - : (to_bp_areas.count(reduced_check.first) ? to_bp_areas.at(reduced_check.first) : Polygons()); + : (to_bp_areas.count(reduced_check.first) ? to_bp_areas.at(reduced_check.first) : Shape()); } else { - relevant_infl = to_model_areas.count(influence.first) ? to_model_areas.at(influence.first) : Polygons(); - relevant_redu = insert_model_areas.count(reduced_check.first) - ? insert_model_areas[reduced_check.first] - : (to_model_areas.count(reduced_check.first) ? to_model_areas.at(reduced_check.first) : Polygons()); + relevant_infl = to_model_areas.count(influence.first) ? to_model_areas.at(influence.first) : Shape(); + relevant_redu = insert_model_areas.count(reduced_check.first) ? insert_model_areas[reduced_check.first] + : (to_model_areas.count(reduced_check.first) ? to_model_areas.at(reduced_check.first) : Shape()); } const bool red_bigger = config.getCollisionRadius(reduced_check.first) > config.getCollisionRadius(influence.first); - std::pair smaller_rad = red_bigger ? std::pair(influence.first, relevant_infl) - : std::pair(reduced_check.first, relevant_redu); - std::pair bigger_rad = red_bigger ? std::pair(reduced_check.first, relevant_redu) - : std::pair(influence.first, relevant_infl); + std::pair smaller_rad + = red_bigger ? std::pair(influence.first, relevant_infl) : std::pair(reduced_check.first, relevant_redu); + std::pair bigger_rad + = red_bigger ? std::pair(reduced_check.first, relevant_redu) : std::pair(influence.first, relevant_infl); const coord_t real_radius_delta = std::abs(config.getRadius(bigger_rad.first) - config.getRadius(smaller_rad.first)); const coord_t smaller_collision_radius = config.getCollisionRadius(smaller_rad.first); @@ -378,7 +377,7 @@ void TreeSupport::mergeHelper( // a branch (of the larger collision radius) placed in this intersection, has already engulfed the branch of the smaller collision radius. // Because of this a merge may happen even if the influence areas (that represent possible center points of branches) do not intersect yet. // Remember that collision radius <= real radius as otherwise this assumption would be false. - const Polygons small_rad_increased_by_big_minus_small = TreeSupportUtils::safeOffsetInc( + const Shape small_rad_increased_by_big_minus_small = TreeSupportUtils::safeOffsetInc( smaller_rad.second, real_radius_delta, volumes_.getCollision(smaller_collision_radius, layer_idx - 1, use_min_radius), @@ -387,7 +386,7 @@ void TreeSupport::mergeHelper( 0, config.support_line_distance / 2, &config.simplifier); - Polygons intersect = small_rad_increased_by_big_minus_small.intersection(bigger_rad.second); + Shape intersect = small_rad_increased_by_big_minus_small.intersection(bigger_rad.second); if (intersect.area() > 1) // dont use empty as a line is not empty, but for this use-case it very well may be (and would be one layer down as union does not keep lines) @@ -428,11 +427,11 @@ void TreeSupport::mergeHelper( const auto getIntersectInfluence = [&](const PropertyAreasUnordered& insert_infl, const PropertyAreas& infl_areas) { - const Polygons infl_small = insert_infl.count(smaller_rad.first) ? insert_infl.at(smaller_rad.first) - : (infl_areas.count(smaller_rad.first) ? infl_areas.at(smaller_rad.first) : Polygons()); - const Polygons infl_big = insert_infl.count(bigger_rad.first) ? insert_infl.at(bigger_rad.first) - : (infl_areas.count(bigger_rad.first) ? infl_areas.at(bigger_rad.first) : Polygons()); - const Polygons small_rad_increased_by_big_minus_small_infl = TreeSupportUtils::safeOffsetInc( + const Shape infl_small = insert_infl.count(smaller_rad.first) ? insert_infl.at(smaller_rad.first) + : (infl_areas.count(smaller_rad.first) ? infl_areas.at(smaller_rad.first) : Shape()); + const Shape infl_big = insert_infl.count(bigger_rad.first) ? insert_infl.at(bigger_rad.first) + : (infl_areas.count(bigger_rad.first) ? infl_areas.at(bigger_rad.first) : Shape()); + const Shape small_rad_increased_by_big_minus_small_infl = TreeSupportUtils::safeOffsetInc( infl_small, real_radius_delta, volumes_.getCollision(smaller_collision_radius, layer_idx - 1, use_min_radius), @@ -445,11 +444,11 @@ void TreeSupport::mergeHelper( infl_big); // If the one with the bigger radius with the lower radius removed overlaps we can merge. }; - Polygons intersect_influence; + Shape intersect_influence; intersect_influence = TreeSupportUtils::safeUnion(intersect, getIntersectInfluence(insert_influence, influence_areas)); // Rounding errors again. Do not ask me where or why. - Polygons intersect_to_model; + Shape intersect_to_model; if (merging_to_bp && config.support_rests_on_model) { intersect_to_model = getIntersectInfluence(insert_model_areas, to_model_areas); @@ -473,7 +472,7 @@ void TreeSupport::mergeHelper( erase.emplace_back(reduced_check.first); erase.emplace_back(influence.first); - const Polygons merge + const Shape merge = intersect.unionPolygons(intersect_to_model).offset(config.getRadius(key), ClipperLib::jtRound).difference(volumes_.getCollision(0, layer_idx - 1)); // ^^^ Regular union should be preferable here as Polygons tend to only become smaller through rounding errors (smaller!=has smaller area as holes have a // negative area.). @@ -553,7 +552,7 @@ void TreeSupport::mergeInfluenceAreas(PropertyAreasUnordered& to_bp_areas, Prope [&](size_t idx) // +=2 as in the beginning only uneven buckets will be filled { idx = idx * 2 + 1; // this is eqivalent to a parallel for(size_t idx=1;idx& input_pair : buckets_area[idx]) + for (const std::pair& input_pair : buckets_area[idx]) { AABB outer_support_wall_aabb = AABB(input_pair.second); outer_support_wall_aabb.expand(config.getRadius(input_pair.first)); @@ -601,16 +600,16 @@ void TreeSupport::mergeInfluenceAreas(PropertyAreasUnordered& to_bp_areas, Prope influence_areas.erase(del); } - for (const std::pair& tup : insert_main[i]) + for (const std::pair& tup : insert_main[i]) { to_bp_areas.emplace(tup); } - for (const std::pair& tup : insert_secondary[i]) + for (const std::pair& tup : insert_secondary[i]) { to_model_areas.emplace(tup); } - for (const std::pair& tup : insert_influence[i]) + for (const std::pair& tup : insert_influence[i]) { influence_areas.emplace(tup); } @@ -640,15 +639,15 @@ std::optional TreeSupport::increaseSingleArea( AreaIncreaseSettings settings, LayerIndex layer_idx, TreeSupportElement* parent, - const Polygons& relevant_offset, - Polygons& to_bp_data, - Polygons& to_model_data, - Polygons& increased, + const Shape& relevant_offset, + Shape& to_bp_data, + Shape& to_model_data, + Shape& increased, const coord_t overspeed, const bool mergelayer) { TreeSupportElement current_elem(parent); // Also increases DTT by one. - Polygons check_layer_data; + Shape check_layer_data; if (settings.increase_radius_) { current_elem.effective_radius_height_ += 1; @@ -726,13 +725,13 @@ std::optional TreeSupport::increaseSingleArea( return true; } - Polygons to_bp_data_2; + Shape to_bp_data_2; if (current_elem.to_buildplate_) { // Regular union as output will not be used later => this area should always be a subset of the safeUnion one. to_bp_data_2 = increased.difference(volumes_.getAvoidance(next_radius, layer_idx - 1, settings.type_, false, settings.use_min_distance_)).unionPolygons(); } - Polygons to_model_data_2; + Shape to_model_data_2; if (config.support_rests_on_model && ! current_elem.to_buildplate_) { to_model_data_2 = increased @@ -744,7 +743,7 @@ std::optional TreeSupport::increaseSingleArea( settings.use_min_distance_)) .unionPolygons(); } - Polygons check_layer_data_2 = current_elem.to_buildplate_ ? to_bp_data_2 : to_model_data_2; + Shape check_layer_data_2 = current_elem.to_buildplate_ ? to_bp_data_2 : to_model_data_2; return check_layer_data_2.area() > 1; }; @@ -773,8 +772,8 @@ std::optional TreeSupport::increaseSingleArea( // wall of the bowl if possible. if (config.getCollisionRadius(current_elem) < config.increase_radius_until_radius && config.getCollisionRadius(current_elem) < config.getRadius(current_elem)) { - Polygons new_to_bp_data; - Polygons new_to_model_data; + Shape new_to_bp_data; + Shape new_to_model_data; if (current_elem.to_buildplate_) { @@ -860,7 +859,7 @@ std::optional TreeSupport::increaseSingleArea( { if (current_elem.to_buildplate_) { - Polygons limited_to_bp = to_bp_data.intersection((current_elem.influence_area_limit_area_)); + Shape limited_to_bp = to_bp_data.intersection((current_elem.influence_area_limit_area_)); if (limited_to_bp.area() > 1) { to_bp_data = limited_to_bp; @@ -870,7 +869,7 @@ std::optional TreeSupport::increaseSingleArea( } else { - Polygons limited_to_model_data = to_model_data.intersection((current_elem.influence_area_limit_area_)); + Shape limited_to_model_data = to_model_data.intersection((current_elem.influence_area_limit_area_)); if (limited_to_model_data.area() > 1) { to_bp_data = to_bp_data.intersection((current_elem.influence_area_limit_area_)); @@ -908,10 +907,10 @@ void TreeSupport::increaseAreas( TreeSupportElement* parent = last_layer[idx]; TreeSupportElement elem(parent); // Also increases dtt. // Abstract representation of the model outline. If an influence area would move through it, it could teleport through a wall. - const Polygons wall_restriction = volumes_.getWallRestriction(config.getCollisionRadius(*parent), layer_idx, parent->use_min_xy_dist_); + const Shape wall_restriction = volumes_.getWallRestriction(config.getCollisionRadius(*parent), layer_idx, parent->use_min_xy_dist_); - Polygons to_bp_data; - Polygons to_model_data; + Shape to_bp_data; + Shape to_model_data; coord_t radius = config.getCollisionRadius(elem); // When the radius increases, the outer "support wall" of the branch will have been moved farther away from the center (as this is the definition of radius). @@ -971,8 +970,8 @@ void TreeSupport::increaseAreas( const coord_t fast_speed = config.maximum_move_distance + extra_speed; const coord_t slow_speed = config.maximum_move_distance_slow + extra_speed + extra_slow_speed; - Polygons offset_slow; - Polygons offset_fast; + Shape offset_slow; + Shape offset_fast; bool add = false; bool bypass_merge = false; @@ -1092,7 +1091,7 @@ void TreeSupport::increaseAreas( false); // Only do not move when already in a no hole avoidance with the regular xy distance. } - Polygons inc_wo_collision; + Shape inc_wo_collision; // Check whether it is faster to calculate the area increased with the fast speed independently from the slow area, or time could be saved by reusing the slow area to // calculate the fast one. Calculated by comparing the steps saved when calculating independently with the saved steps when not. const bool offset_independent_faster = (radius / safe_movement_distance - (((config.maximum_move_distance + extra_speed) < (radius + safe_movement_distance)) ? 1 : 0)) @@ -1154,8 +1153,8 @@ void TreeSupport::increaseAreas( { // If the area becomes for whatever reason something that clipper sees as a line, offset would stop working, so ensure that even if if wrongly would be a line, // it still actually has an area that can be increased - Polygons lines_offset = TreeSupportUtils::toPolylines(*parent->area_).offsetPolyLine(EPSILON); - Polygons base_error_area = parent->area_->unionPolygons(lines_offset); + Shape lines_offset = TreeSupportUtils::toPolylines(*parent->area_).offset(EPSILON); + Shape base_error_area = parent->area_->unionPolygons(lines_offset); result = increaseSingleArea(settings, layer_idx, parent, base_error_area, to_bp_data, to_model_data, inc_wo_collision, settings.increase_speed_, mergelayer); if (fast_speed < settings.increase_speed_) @@ -1242,7 +1241,7 @@ void TreeSupport::increaseAreas( if (add) { - Polygons max_influence_area = TreeSupportUtils::safeUnion( + Shape max_influence_area = TreeSupportUtils::safeUnion( inc_wo_collision.difference(volumes_.getCollision(radius, layer_idx - 1, elem.use_min_xy_dist_)), TreeSupportUtils::safeUnion(to_bp_data, to_model_data)); // ^^^ Note: union seems useless, but some rounding errors somewhere can cause to_bp_data to be slightly bigger than it should be @@ -1251,7 +1250,7 @@ void TreeSupport::increaseAreas( std::lock_guard critical_section_newLayer(critical_sections); if (bypass_merge) { - Polygons* new_area = new Polygons(max_influence_area); + Shape* new_area = new Shape(max_influence_area); TreeSupportElement* next = new TreeSupportElement(elem, new_area); bypass_merge_areas.emplace_back(next); } @@ -1348,10 +1347,10 @@ void TreeSupport::createLayerPathing(std::vector>& new_element = ! move_bounds[layer_idx - 1].empty(); // Save calculated elements to output, and allocate Polygons on heap, as they will not be changed again. - for (std::pair tup : influence_areas) + for (std::pair tup : influence_areas) { const TreeSupportElement elem = tup.first; - Polygons* new_area = new Polygons(TreeSupportUtils::safeUnion(tup.second)); + Shape* new_area = new Shape(TreeSupportUtils::safeUnion(tup.second)); TreeSupportElement* next = new TreeSupportElement(elem, new_area); move_bounds[layer_idx - 1].emplace(next); @@ -1426,13 +1425,13 @@ bool TreeSupport::setToModelContact(std::vector>& set = true; } - Polygons valid_place_area; + Shape valid_place_area; // Check for every layer upwards, up to the point where this influence area was created (either by initial insert or merge) if the branch could be placed on it, and highest // up layer index. for (LayerIndex layer_check = layer_idx; check->next_height_ >= layer_check; layer_check++) { - Polygons check_valid_place_area = check->area_->intersection(volumes_.getPlaceableAreas(config.getCollisionRadius(*check), layer_check)); + Shape check_valid_place_area = check->area_->intersection(volumes_.getPlaceableAreas(config.getCollisionRadius(*check), layer_check)); if (! check_valid_place_area.empty()) { @@ -1503,7 +1502,7 @@ bool TreeSupport::setToModelContact(std::vector>& else // can not add graceful => just place it here and hope for the best { Point2LL best = first_elem->next_position_; - Polygons valid_place_area + Shape valid_place_area = first_elem->area_->difference(volumes_.getAvoidance(config.getCollisionRadius(first_elem), layer_idx, AvoidanceType::COLLISION, first_elem->use_min_xy_dist_)); if (! valid_place_area.inside(best, true)) @@ -1643,7 +1642,7 @@ void TreeSupport::createNodesFromArea(std::vector> void TreeSupport::generateBranchAreas( std::vector>& linear_data, - std::vector>& layer_tree_polygons, + std::vector>& layer_tree_polygons, const std::map& inverse_tree_order) { double progress_total = TREE_PROGRESS_PRECALC_AVO + TREE_PROGRESS_PRECALC_COLL + TREE_PROGRESS_GENERATE_NODES + TREE_PROGRESS_AREA_CALC; @@ -1655,11 +1654,11 @@ void TreeSupport::generateBranchAreas( for (Point2LL vertex : base_circle) { vertex = Point2LL(vertex.X * config.branch_radius / TreeSupportBaseCircle::base_radius, vertex.Y * config.branch_radius / TreeSupportBaseCircle::base_radius); - branch_circle.add(vertex); + branch_circle.push_back(vertex); } } - std::vector linear_inserts(linear_data.size()); + std::vector linear_inserts(linear_data.size()); const size_t progress_inserts_check_interval = std::max(linear_data.size() / progress_report_steps, size_t(1)); std::mutex critical_sections; @@ -1697,9 +1696,9 @@ void TreeSupport::generateBranchAreas( } coord_t max_speed_sqd = 0; - std::function generateArea = [&](coord_t offset) + std::function generateArea = [&](coord_t offset) { - Polygons poly; + Shape poly; for (std::pair movement : movement_directions) { @@ -1723,9 +1722,9 @@ void TreeSupport::generateBranchAreas( for (Point2LL vertex : branch_circle) { vertex = Point2LL(matrix[0] * vertex.X + matrix[1] * vertex.Y, matrix[2] * vertex.X + matrix[3] * vertex.Y); - circle.add(center_position + vertex); + circle.push_back(center_position + vertex); } - poly.add(circle.offset(0)); + poly.push_back(circle.offset(0)); } poly = poly.unionPolygons() @@ -1747,7 +1746,7 @@ void TreeSupport::generateBranchAreas( { // Simulate the path the nozzle will take on the outermost wall. // If multiple parts exist, the outer line will not go all around the support part potentially causing support material to be printed mid air. - Polygons nozzle_path = linear_inserts[idx].offset(-config.support_line_width / 2); + Shape nozzle_path = linear_inserts[idx].offset(-config.support_line_width / 2); if (nozzle_path.splitIntoParts(false).size() > 1) { // Just try to make the area a tiny bit larger. @@ -1757,8 +1756,8 @@ void TreeSupport::generateBranchAreas( // if larger area did not fix the problem, all parts off the nozzle path that do not contain the center point are removed, hoping for the best if (nozzle_path.splitIntoParts(false).size() > 1) { - Polygons polygons_with_correct_center; - for (PolygonsPart part : nozzle_path.splitIntoParts(false)) + Shape polygons_with_correct_center; + for (SingleShape part : nozzle_path.splitIntoParts(false)) { if (part.inside(elem->result_on_layer_, true)) { @@ -1800,7 +1799,7 @@ void TreeSupport::generateBranchAreas( } } -void TreeSupport::smoothBranchAreas(std::vector>& layer_tree_polygons) +void TreeSupport::smoothBranchAreas(std::vector>& layer_tree_polygons) { double progress_total = TREE_PROGRESS_PRECALC_AVO + TREE_PROGRESS_PRECALC_COLL + TREE_PROGRESS_GENERATE_NODES + TREE_PROGRESS_AREA_CALC + TREE_PROGRESS_GENERATE_BRANCH_AREAS; const coord_t max_radius_change_per_layer = 1 + config.support_line_width / 2; // This is the upper limit a radius may change per layer. +1 to avoid rounding errors. @@ -1808,15 +1807,15 @@ void TreeSupport::smoothBranchAreas(std::vector(layer_tree_polygons.size(), 1UL) - 1UL)) { - std::vector> processing; + std::vector> processing; processing.insert(processing.end(), layer_tree_polygons[layer_idx].begin(), layer_tree_polygons[layer_idx].end()); - std::vector>> update_next(processing.size()); // With this a lock can be avoided. + std::vector>> update_next(processing.size()); // With this a lock can be avoided. cura::parallel_for( 0, processing.size(), [&](const size_t processing_idx) { - std::pair data_pair = processing[processing_idx]; + std::pair data_pair = processing[processing_idx]; coord_t max_outer_wall_distance = 0; bool do_something = false; @@ -1834,21 +1833,21 @@ void TreeSupport::smoothBranchAreas(std::vectorparents_) { if (config.getRadius(*parent) != config.getCollisionRadius(*parent)) { update_next[processing_idx].emplace_back( - std::pair(parent, layer_tree_polygons[layer_idx + 1][parent].intersection(max_allowed_area))); + std::pair(parent, layer_tree_polygons[layer_idx + 1][parent].intersection(max_allowed_area))); } } } }); - for (std::vector> data_vector : update_next) + for (std::vector> data_vector : update_next) { - for (std::pair data_pair : data_vector) + for (std::pair data_pair : data_vector) { layer_tree_polygons[layer_idx + 1][data_pair.first] = data_pair.second; } @@ -1864,25 +1863,25 @@ void TreeSupport::smoothBranchAreas(std::vector updated_last_iteration; for (const auto layer_idx : ranges::views::iota(0UL, std::max(layer_tree_polygons.size(), 1UL) - 1UL) | ranges::views::reverse) { - std::vector> processing; + std::vector> processing; processing.insert(processing.end(), layer_tree_polygons[layer_idx].begin(), layer_tree_polygons[layer_idx].end()); - std::vector> update_next( + std::vector> update_next( processing.size(), - std::pair(nullptr, Polygons())); // With this a lock can be avoided. + std::pair(nullptr, Shape())); // With this a lock can be avoided. cura::parallel_for( 0, processing.size(), [&](const size_t processing_idx) { - std::pair data_pair = processing[processing_idx]; + std::pair data_pair = processing[processing_idx]; bool do_something = false; - Polygons max_allowed_area; + Shape max_allowed_area; for (size_t idx = 0; idx < data_pair.first->parents_.size(); idx++) { TreeSupportElement* parent = data_pair.first->parents_[idx]; const coord_t max_outer_line_increase = max_radius_change_per_layer; - Polygons result = layer_tree_polygons[layer_idx + 1][parent].offset(max_outer_line_increase); + Shape result = layer_tree_polygons[layer_idx + 1][parent].offset(max_outer_line_increase); const Point2LL direction = data_pair.first->result_on_layer_ - parent->result_on_layer_; // Move the polygons object. for (auto& outer : result) @@ -1892,22 +1891,22 @@ void TreeSupport::smoothBranchAreas(std::vector(data_pair.first, result); + update_next[processing_idx] = std::pair(data_pair.first, result); } } }); updated_last_iteration.clear(); - for (std::pair data_pair : update_next) + for (std::pair data_pair : update_next) { if (data_pair.first != nullptr) { @@ -1922,9 +1921,9 @@ void TreeSupport::smoothBranchAreas(std::vector>& layer_tree_polygons, + std::vector>& layer_tree_polygons, const std::vector>& linear_data, - std::vector>>& dropped_down_areas, + std::vector>>& dropped_down_areas, const std::map& inverse_tree_order) { cura::parallel_for( @@ -1937,7 +1936,7 @@ void TreeSupport::dropNonGraciousAreas( && ! elem->to_buildplate_; // If an element has no child, it connects to whatever is below as no support further down for it will exist. if (non_gracious_model_contact) { - Polygons rest_support = layer_tree_polygons[linear_data[idx].first][elem].intersection(volumes_.getAccumulatedPlaceable0(linear_data[idx].first)); + Shape rest_support = layer_tree_polygons[linear_data[idx].first][elem].intersection(volumes_.getAccumulatedPlaceable0(linear_data[idx].first)); for (LayerIndex counter = 1; rest_support.area() > 1 && counter < linear_data[idx].first; ++counter) { rest_support = rest_support.difference(volumes_.getCollision(0, linear_data[idx].first - counter)); @@ -1948,7 +1947,7 @@ void TreeSupport::dropNonGraciousAreas( } -void TreeSupport::filterFloatingLines(std::vector& support_layer_storage) +void TreeSupport::filterFloatingLines(std::vector& support_layer_storage) { const auto t_start = std::chrono::high_resolution_clock::now(); @@ -1956,7 +1955,7 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora const coord_t open_close_distance = config.fill_outline_gaps ? config.min_feature_size / 2 - 5 : config.min_wall_line_width / 2 - 5; // based on calculation in WallToolPath const double small_area_length = INT2MM(static_cast(config.support_line_width) / 2); - std::function reversePolygon = [&](Polygons& poly) + std::function reversePolygon = [&](Shape& poly) { for (size_t idx = 0; idx < poly.size(); idx++) { @@ -1965,7 +1964,7 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora }; - std::vector support_holes(support_layer_storage.size(), Polygons()); + std::vector support_holes(support_layer_storage.size(), Shape()); // Extract all holes as polygon objects cura::parallel_for( 0, @@ -1978,33 +1977,33 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora .offset(-open_close_distance); support_layer_storage[layer_idx].removeSmallAreas(small_area_length * small_area_length, false); - std::vector parts = support_layer_storage[layer_idx].sortByNesting(); + std::vector parts = support_layer_storage[layer_idx].sortByNesting(); if (parts.size() <= 1) { return; } - Polygons holes_original; + Shape holes_original; for (const size_t idx : ranges::views::iota(1UL, parts.size())) { - Polygons area = parts[idx]; + Shape area = parts[idx]; reversePolygon(area); - holes_original.add(area); + holes_original.push_back(area); } support_holes[layer_idx] = holes_original; }); const auto t_union = std::chrono::high_resolution_clock::now(); - std::vector> holeparts(support_layer_storage.size()); + std::vector> holeparts(support_layer_storage.size()); // Split all holes into parts cura::parallel_for( 0, support_layer_storage.size(), [&](const LayerIndex layer_idx) { - for (Polygons hole : support_holes[layer_idx].splitIntoParts()) + for (Shape hole : support_holes[layer_idx].splitIntoParts()) { holeparts[layer_idx].emplace_back(hole); } @@ -2024,15 +2023,16 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora return; } - Polygons outer_walls - = TreeSupportUtils::toPolylines(support_layer_storage[layer_idx - 1].getOutsidePolygons()) - .tubeShape(closing_dist, 0); //.unionPolygons(volumes_.getCollision(0, layer_idx - 1, true).offset(-(config.support_line_width+config.xy_min_distance))); + Shape outer_walls = TreeSupportUtils::toPolylines(support_layer_storage[layer_idx - 1].getOutsidePolygons()) + .createTubeShape( + closing_dist, + 0); //.unionPolygons(volumes_.getCollision(0, layer_idx - 1, true).offset(-(config.support_line_width+config.xy_min_distance))); - Polygons holes_below; + Shape holes_below; for (auto poly : holeparts[layer_idx - 1]) { - holes_below.add(poly); + holes_below.push_back(poly); } for (auto [idx, hole] : holeparts[layer_idx] | ranges::views::enumerate) @@ -2060,7 +2060,7 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora const auto t_hole_rest_ordering = std::chrono::high_resolution_clock::now(); std::unordered_set removed_holes_by_idx; - std::vector valid_holes(support_holes.size(), Polygons()); + std::vector valid_holes(support_holes.size(), Shape()); // Check which holes have to be removed as they do not rest on anything. Only keep holes that have to be removed for (const size_t layer_idx : ranges::views::iota(1UL, support_holes.size())) { @@ -2093,8 +2093,8 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora } else { - valid_holes[layer_idx].add(hole); - holeparts[layer_idx][idx] = Polygons(); // all remaining holes will have to be removed later, so removing the hole means it is confirmed valid! + valid_holes[layer_idx].push_back(hole); + holeparts[layer_idx][idx] = Shape(); // all remaining holes will have to be removed later, so removing the hole means it is confirmed valid! } } removed_holes_by_idx = next_removed_holes_by_idx; @@ -2115,7 +2115,7 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora support_layer_storage[layer_idx] = support_layer_storage[layer_idx].getOutsidePolygons(); reversePolygon(valid_holes[layer_idx]); - support_layer_storage[layer_idx].add(valid_holes[layer_idx]); + support_layer_storage[layer_idx].push_back(valid_holes[layer_idx]); }); @@ -2136,9 +2136,9 @@ void TreeSupport::filterFloatingLines(std::vector& support_layer_stora } void TreeSupport::finalizeInterfaceAndSupportAreas( - std::vector& support_layer_storage, - std::vector& support_roof_storage, - std::vector& support_layer_storage_fractional, + std::vector& support_layer_storage, + std::vector& support_roof_storage, + std::vector& support_layer_storage_fractional, SliceDataStorage& storage) { InterfacePreference interface_pref = config.interface_preference; // InterfacePreference::SUPPORT_LINES_OVERWRITE_INTERFACE; @@ -2152,13 +2152,13 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( support_layer_storage.size(), [&](const LayerIndex layer_idx) { - Polygons fake_roof_lines; + Shape fake_roof_lines; for (FakeRoofArea& f_roof : fake_roof_areas[layer_idx]) { - fake_roof_lines.add( + fake_roof_lines.push_back( TreeSupportUtils::generateSupportInfillLines(f_roof.area_, config, false, layer_idx, f_roof.line_distance_, storage.support.cross_fill_provider, false) - .offsetPolyLine(config.support_line_width / 2)); + .offset(config.support_line_width / 2)); } fake_roof_lines = fake_roof_lines.unionPolygons(); @@ -2181,22 +2181,22 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( case InterfacePreference::INTERFACE_LINES_OVERWRITE_SUPPORT: { - Polygons interface_lines = TreeSupportUtils::generateSupportInfillLines( - storage.support.supportLayers[layer_idx].support_roof, - config, - true, - layer_idx, - config.support_roof_line_distance, - storage.support.cross_fill_provider, - true) - .offsetPolyLine(config.support_roof_line_width / 2); + Shape interface_lines = TreeSupportUtils::generateSupportInfillLines( + storage.support.supportLayers[layer_idx].support_roof, + config, + true, + layer_idx, + config.support_roof_line_distance, + storage.support.cross_fill_provider, + true) + .offset(config.support_roof_line_width / 2); support_layer_storage[layer_idx] = support_layer_storage[layer_idx].difference(interface_lines); } break; case InterfacePreference::SUPPORT_LINES_OVERWRITE_INTERFACE: { - Polygons tree_lines; + Shape tree_lines; tree_lines = tree_lines.unionPolygons(TreeSupportUtils::generateSupportInfillLines( support_layer_storage[layer_idx], config, @@ -2205,7 +2205,7 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( config.support_line_distance, storage.support.cross_fill_provider, true) - .offsetPolyLine(config.support_line_width / 2)); + .offset(config.support_line_width / 2)); storage.support.supportLayers[layer_idx].support_roof = storage.support.supportLayers[layer_idx].support_roof.difference(tree_lines); // Do not draw roof where the tree is. I prefer it this way as otherwise the roof may cut of a branch from its support below. } @@ -2219,8 +2219,8 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( // Subtract support floors from the support area and add them to the support floor instead. if (config.support_bottom_layers > 0 && ! support_layer_storage[layer_idx].empty()) { - Polygons floor_layer = storage.support.supportLayers[layer_idx].support_bottom; - Polygons layer_outset = support_layer_storage[layer_idx].offset(config.support_bottom_offset).difference(volumes_.getCollision(0, layer_idx, false)); + Shape floor_layer = storage.support.supportLayers[layer_idx].support_bottom; + Shape layer_outset = support_layer_storage[layer_idx].offset(config.support_bottom_offset).difference(volumes_.getCollision(0, layer_idx, false)); size_t layers_below = 0; while (layers_below <= config.support_bottom_layers) { @@ -2229,7 +2229,7 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( = static_cast(std::max(0, (static_cast(layer_idx) - static_cast(layers_below)) - static_cast(config.z_distance_bottom_layers))); constexpr bool no_support = false; constexpr bool no_prime_tower = false; - floor_layer.add(layer_outset.intersection(storage.getLayerOutlines(sample_layer, no_support, no_prime_tower))); + floor_layer.push_back(layer_outset.intersection(storage.getLayerOutlines(sample_layer, no_support, no_prime_tower))); if (layers_below < config.support_bottom_layers) { layers_below = std::min(layers_below + 1UL, config.support_bottom_layers); @@ -2250,8 +2250,7 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( support_layer_storage.size(), [&](const LayerIndex layer_idx) { - constexpr bool convert_every_part = true; // Convert every part into a PolygonsPart for the support. - + constexpr bool convert_every_part = true; // Convert every part into a SingleShape for the support. storage.support.supportLayers[layer_idx] .fillInfillParts(support_layer_storage[layer_idx], config.support_line_width, config.support_wall_count, false, convert_every_part); @@ -2259,7 +2258,7 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( // This only works because fractional support is always just projected upwards regular support or skin. // Also technically violates skin height, but there is no good way to prevent that. - Polygons fractional_support; + Shape fractional_support; if (layer_idx > 0) { @@ -2298,10 +2297,10 @@ void TreeSupport::finalizeInterfaceAndSupportAreas( void TreeSupport::drawAreas(std::vector>& move_bounds, SliceDataStorage& storage) { - std::vector support_layer_storage(move_bounds.size()); - std::vector support_layer_storage_fractional(move_bounds.size()); - std::vector support_roof_storage_fractional(move_bounds.size()); - std::vector support_roof_storage(move_bounds.size()); + std::vector support_layer_storage(move_bounds.size()); + std::vector support_layer_storage_fractional(move_bounds.size()); + std::vector support_roof_storage_fractional(move_bounds.size()); + std::vector support_roof_storage(move_bounds.size()); std::map inverse_tree_order; // In the tree structure only the parents can be accessed. Inverse this to be able to access the children. std::vector> @@ -2332,8 +2331,8 @@ void TreeSupport::drawAreas(std::vector>& move_bou } - // Reorder the processed data by layers again. The map also could be a vector>: - std::vector> layer_tree_polygons(move_bounds.size()); + // Reorder the processed data by layers again. The map also could be a vector>: + std::vector> layer_tree_polygons(move_bounds.size()); const auto t_start = std::chrono::high_resolution_clock::now(); // Generate the circles that will be the branches. @@ -2346,16 +2345,16 @@ void TreeSupport::drawAreas(std::vector>& move_bou const auto t_smooth = std::chrono::high_resolution_clock::now(); // Drop down all trees that connect non gracefully with the model. - std::vector>> dropped_down_areas(linear_data.size()); + std::vector>> dropped_down_areas(linear_data.size()); dropNonGraciousAreas(layer_tree_polygons, linear_data, dropped_down_areas, inverse_tree_order); const auto t_drop = std::chrono::high_resolution_clock::now(); // single threaded combining all dropped down support areas to the right layers. ONLY COPYS DATA! for (const coord_t i : ranges::views::iota(0UL, dropped_down_areas.size())) { - for (std::pair pair : dropped_down_areas[i]) + for (std::pair pair : dropped_down_areas[i]) { - support_layer_storage[pair.first].add(pair.second); + support_layer_storage[pair.first].push_back(pair.second); } } @@ -2365,7 +2364,7 @@ void TreeSupport::drawAreas(std::vector>& move_bou layer_tree_polygons.size(), [&](const size_t layer_idx) { - for (std::pair data_pair : layer_tree_polygons[layer_idx]) + for (std::pair data_pair : layer_tree_polygons[layer_idx]) { if (data_pair.first->missing_roof_layers_ > data_pair.first->distance_to_top_ && TreeSupportUtils::generateSupportInfillLines(data_pair.second, config, true, layer_idx, config.support_roof_line_distance, nullptr, true).empty()) @@ -2394,22 +2393,22 @@ void TreeSupport::drawAreas(std::vector>& move_bou layer_tree_polygons.size(), [&](const size_t layer_idx) { - for (std::pair data_pair : layer_tree_polygons[layer_idx]) + for (std::pair data_pair : layer_tree_polygons[layer_idx]) { if (data_pair.first->parents_.empty() && ! data_pair.first->supports_roof_ && layer_idx + 1 < support_roof_storage_fractional.size() && config.z_distance_top % config.layer_height > 0) { if (data_pair.first->missing_roof_layers_ > data_pair.first->distance_to_top_) { - support_roof_storage_fractional[layer_idx + 1].add(data_pair.second); + support_roof_storage_fractional[layer_idx + 1].push_back(data_pair.second); } else { - support_layer_storage_fractional[layer_idx + 1].add(data_pair.second); + support_layer_storage_fractional[layer_idx + 1].push_back(data_pair.second); } } - ((data_pair.first->missing_roof_layers_ > data_pair.first->distance_to_top_) ? support_roof_storage : support_layer_storage)[layer_idx].add(data_pair.second); + ((data_pair.first->missing_roof_layers_ > data_pair.first->distance_to_top_) ? support_roof_storage : support_layer_storage)[layer_idx].push_back(data_pair.second); } if (layer_idx + 1 < support_roof_storage_fractional.size()) { @@ -2422,7 +2421,7 @@ void TreeSupport::drawAreas(std::vector>& move_bou { if (support_layer_storage.size() > layer_idx) { - support_layer_storage[layer_idx].add(additional_required_support_area[layer_idx]); + support_layer_storage[layer_idx].push_back(additional_required_support_area[layer_idx]); } scripta::log("tree_support_layer_storage", support_layer_storage[layer_idx], SectionType::SUPPORT, layer_idx); } diff --git a/src/TreeSupportTipGenerator.cpp b/src/TreeSupportTipGenerator.cpp index 00ce1c46d9..0a2af0e233 100644 --- a/src/TreeSupportTipGenerator.cpp +++ b/src/TreeSupportTipGenerator.cpp @@ -1,24 +1,24 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "TreeSupportTipGenerator.h" #include +#include #include -#include +#include #include #include #include #include -#include #include #include "Application.h" //To get settings. #include "TreeSupportUtils.h" +#include "geometry/OpenPolyline.h" #include "infill/SierpinskiFillProvider.h" #include "settings/EnumSettings.h" -#include "utils/Simplify.h" #include "utils/ThreadPool.h" #include "utils/algorithm.h" #include "utils/math.h" //For round_up_divide and PI. @@ -57,9 +57,9 @@ TreeSupportTipGenerator::TreeSupportTipGenerator(const SliceMeshStorage& mesh, T , tip_roof_size_(force_tip_to_roof_ ? INT2MM2(config_.min_radius * config_.min_radius) * std::numbers::pi : 0) , force_minimum_roof_area_(use_fake_roof_ || SUPPORT_TREE_MINIMUM_ROOF_AREA_HARD_LIMIT) , already_inserted_(mesh.overhang_areas.size()) - , support_roof_drawn_(mesh.overhang_areas.size(), Polygons()) - , support_roof_drawn_fractional_(mesh.overhang_areas.size(), Polygons()) - , roof_tips_drawn_(mesh.overhang_areas.size(), Polygons()) + , support_roof_drawn_(mesh.overhang_areas.size(), Shape()) + , support_roof_drawn_fractional_(mesh.overhang_areas.size(), Shape()) + , roof_tips_drawn_(mesh.overhang_areas.size(), Shape()) { const double support_overhang_angle = mesh.settings.get("support_angle"); const coord_t max_overhang_speed = (support_overhang_angle < TAU / 4) ? (coord_t)(tan(support_overhang_angle) * config_.layer_height) : std::numeric_limits::max(); @@ -105,7 +105,7 @@ TreeSupportTipGenerator::TreeSupportTipGenerator(const SliceMeshStorage& mesh, T } -std::vector TreeSupportTipGenerator::convertLinesToInternal(Polygons polylines, LayerIndex layer_idx) +std::vector TreeSupportTipGenerator::convertLinesToInternal(const OpenLinesSet& polylines, LayerIndex layer_idx) { // NOTE: The volumes below (on which '.inside(p, true)' is called each time below) are the same each time. The values being calculated here are strictly local as well. // So they could in theory be pre-calculated here (outside of the loop). However, when I refatored it to be that way, it seemed to cause deadlocks each time for some @@ -115,7 +115,7 @@ std::vector TreeSupportTipGenerator::c std::vector result; // Also checks if the position is valid, if it is NOT, it deletes that point - for (const auto& line : polylines) + for (const OpenPolyline& line : polylines) { LineInformation res_line; for (const Point2LL& p : line) @@ -159,17 +159,17 @@ std::vector TreeSupportTipGenerator::c return result; } -Polygons TreeSupportTipGenerator::convertInternalToLines(std::vector lines) +OpenLinesSet TreeSupportTipGenerator::convertInternalToLines(std::vector lines) { - Polygons result; + OpenLinesSet result; for (const LineInformation& line : lines) { - Polygon path; + OpenPolyline path; for (const auto& point_data : line) { - path.add(point_data.first); + path.push_back(point_data.first); } - result.add(path); + result.push_back(path); } return result; } @@ -251,23 +251,24 @@ std::pair, std::vector>>>(keep, set_free); } -Polygons TreeSupportTipGenerator::ensureMaximumDistancePolyline(const Polygons& input, coord_t distance, size_t min_points, bool enforce_distance) const +OpenLinesSet TreeSupportTipGenerator::ensureMaximumDistancePolyline(const OpenLinesSet& input, coord_t distance, size_t min_points, bool enforce_distance) const { - Polygons result; - for (auto part : input) + OpenLinesSet result; + for (OpenPolyline part : input) { if (part.size() == 0) { continue; } - const coord_t length = Polygon(part).offset(0).polyLineLength(); - Polygon line; + + const coord_t length = part.length(); + OpenPolyline line; coord_t current_distance = std::max(distance, coord_t(FUDGE_LENGTH * 2)); if (length < 2 * distance && min_points <= 1) { - ClosestPolygonPoint middle_point(part[0], 0, part); + ClosestPoint middle_point(part[0], 0, &part); middle_point = PolygonUtils::walk(middle_point, coord_t(length / 2)); - line.add(middle_point.location_); + line.push_back(middle_point.location_); } else { @@ -301,7 +302,7 @@ Polygons TreeSupportTipGenerator::ensureMaximumDistancePolyline(const Polygons& { line.clear(); Point2LL current_point = part[0]; - line.add(part[0]); + line.push_back(part[0]); bool should_add_endpoint = min_points > 1 || vSize2(part[0] - part[optimal_end_index]) > (current_distance * current_distance); bool added_endpoint = ! should_add_endpoint; // If no endpoint should be added all endpoints are already added. @@ -319,7 +320,7 @@ Polygons TreeSupportTipGenerator::ensureMaximumDistancePolyline(const Polygons& current_index = optimal_end_index; current_point = part[optimal_end_index]; added_endpoint = true; - line.add(part[optimal_end_index]); + line.push_back(part[optimal_end_index]); continue; } @@ -337,7 +338,7 @@ Polygons TreeSupportTipGenerator::ensureMaximumDistancePolyline(const Polygons& if (! enforce_distance || min_distance_to_existing_point_sqd >= (current_distance * current_distance)) { // viable point was found. Add to possible result. - line.add(next_point.location); + line.push_back(next_point.location); current_point = next_point.location; current_index = next_point.pos; next_distance = current_distance; @@ -367,13 +368,13 @@ Polygons TreeSupportTipGenerator::ensureMaximumDistancePolyline(const Polygons& if (! added_endpoint) { - line.add(part[optimal_end_index]); + line.push_back(part[optimal_end_index]); } current_distance *= 0.9; } } - result.add(line); + result.push_back(line); } return result; } @@ -407,7 +408,7 @@ std::shared_ptr TreeSupportTipGenerator::generateCrossFi return nullptr; } -void TreeSupportTipGenerator::dropOverhangAreas(const SliceMeshStorage& mesh, std::vector& result, bool roof) +void TreeSupportTipGenerator::dropOverhangAreas(const SliceMeshStorage& mesh, std::vector& result, bool roof) { std::mutex critical; @@ -422,13 +423,13 @@ void TreeSupportTipGenerator::dropOverhangAreas(const SliceMeshStorage& mesh, st return; // This is a continue if imagined in a loop context. } - Polygons relevant_forbidden = volumes_.getCollision(roof ? 0 : config_.getRadius(0), layer_idx, ! xy_overrides_); + Shape relevant_forbidden = volumes_.getCollision(roof ? 0 : config_.getRadius(0), layer_idx, ! xy_overrides_); // ^^^ Take the least restrictive avoidance possible // Technically this also makes support blocker smaller, which is wrong as they do not have a xy_distance, but it should be good enough. - Polygons model_outline = volumes_.getCollision(0, layer_idx, ! xy_overrides_).offset(-config_.xy_min_distance, ClipperLib::jtRound); + Shape model_outline = volumes_.getCollision(0, layer_idx, ! xy_overrides_).offset(-config_.xy_min_distance, ClipperLib::jtRound); - Polygons overhang_regular = TreeSupportUtils::safeOffsetInc( + Shape overhang_regular = TreeSupportUtils::safeOffsetInc( mesh.overhang_areas[layer_idx + z_distance_delta_], roof ? roof_outset_ : support_outset_, relevant_forbidden, @@ -437,19 +438,19 @@ void TreeSupportTipGenerator::dropOverhangAreas(const SliceMeshStorage& mesh, st 1, config_.support_line_distance / 2, &config_.simplifier); - Polygons remaining_overhang = mesh.overhang_areas[layer_idx + z_distance_delta_] - .offset(roof ? roof_outset_ : support_outset_) - .difference(overhang_regular) - .intersection(relevant_forbidden) - .difference(model_outline); + Shape remaining_overhang = mesh.overhang_areas[layer_idx + z_distance_delta_] + .offset(roof ? roof_outset_ : support_outset_) + .difference(overhang_regular) + .intersection(relevant_forbidden) + .difference(model_outline); for (size_t lag_ctr = 1; lag_ctr <= max_overhang_insert_lag_ && layer_idx - coord_t(lag_ctr) >= 1 && ! remaining_overhang.empty(); lag_ctr++) { { std::lock_guard critical_section_storage(critical); - result[layer_idx - lag_ctr].add(remaining_overhang); + result[layer_idx - lag_ctr].push_back(remaining_overhang); } - Polygons relevant_forbidden_below = volumes_.getCollision(roof ? 0 : config_.getRadius(0), layer_idx - lag_ctr, ! xy_overrides_).offset(EPSILON); + Shape relevant_forbidden_below = volumes_.getCollision(roof ? 0 : config_.getRadius(0), layer_idx - lag_ctr, ! xy_overrides_).offset(EPSILON); remaining_overhang = remaining_overhang.intersection(relevant_forbidden_below).unionPolygons().difference(model_outline); } }); @@ -465,9 +466,9 @@ void TreeSupportTipGenerator::dropOverhangAreas(const SliceMeshStorage& mesh, st void TreeSupportTipGenerator::calculateRoofAreas(const cura::SliceMeshStorage& mesh) { - std::vector potential_support_roofs(mesh.overhang_areas.size(), Polygons()); + std::vector potential_support_roofs(mesh.overhang_areas.size(), Shape()); std::mutex critical_potential_support_roofs; - std::vector dropped_overhangs(mesh.overhang_areas.size(), Polygons()); + std::vector dropped_overhangs(mesh.overhang_areas.size(), Shape()); if (xy_overrides_) { @@ -486,19 +487,19 @@ void TreeSupportTipGenerator::calculateRoofAreas(const cura::SliceMeshStorage& m // Roof does not have a radius, so remove it using offset. Note that there is no 0 radius avoidance, and it would not be identical with the avoidance offset with // -radius. This is intentional here, as support roof is still valid if only a part of the tip may reach it. - Polygons forbidden_here = volumes_ - .getAvoidance( - config_.getRadius(0), - layer_idx, - (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, - config_.support_rests_on_model, - ! xy_overrides_) - .offset(-config_.getRadius(0), ClipperLib::jtRound); + Shape forbidden_here = volumes_ + .getAvoidance( + config_.getRadius(0), + layer_idx, + (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, + config_.support_rests_on_model, + ! xy_overrides_) + .offset(-config_.getRadius(0), ClipperLib::jtRound); // todo Since arachnea the assumption that an area smaller then line_width is not printed is no longer true all such safeOffset should have config.support_line_width // replaced with another setting. It should still work in most cases, but it should be possible to create a situation where a overhang outset lags though a wall. I will // take a look at this later. - Polygons full_overhang_area = TreeSupportUtils::safeOffsetInc( + Shape full_overhang_area = TreeSupportUtils::safeOffsetInc( mesh.full_overhang_areas[layer_idx + z_distance_delta_].unionPolygons(dropped_overhangs[layer_idx]), roof_outset_, forbidden_here, @@ -510,14 +511,14 @@ void TreeSupportTipGenerator::calculateRoofAreas(const cura::SliceMeshStorage& m for (LayerIndex dtt_roof = 0; dtt_roof < support_roof_layers_ && layer_idx - dtt_roof >= 1; dtt_roof++) { - const Polygons forbidden_next = volumes_ - .getAvoidance( - config_.getRadius(0), - layer_idx - (dtt_roof + 1), - (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, - config_.support_rests_on_model, - ! xy_overrides_) - .offset(-config_.getRadius(0), ClipperLib::jtRound); + const Shape forbidden_next = volumes_ + .getAvoidance( + config_.getRadius(0), + layer_idx - (dtt_roof + 1), + (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, + config_.support_rests_on_model, + ! xy_overrides_) + .offset(-config_.getRadius(0), ClipperLib::jtRound); full_overhang_area = full_overhang_area.difference(forbidden_next); @@ -529,10 +530,10 @@ void TreeSupportTipGenerator::calculateRoofAreas(const cura::SliceMeshStorage& m if (full_overhang_area.area() > EPSILON) { std::lock_guard critical_section_potential_support_roofs(critical_potential_support_roofs); - potential_support_roofs[layer_idx - dtt_roof].add((full_overhang_area)); + potential_support_roofs[layer_idx - dtt_roof].push_back((full_overhang_area)); if (dtt_roof == 0) { - support_roof_drawn_fractional_[layer_idx].add(full_overhang_area); + support_roof_drawn_fractional_[layer_idx].push_back(full_overhang_area); } } else @@ -559,7 +560,7 @@ void TreeSupportTipGenerator::calculateRoofAreas(const cura::SliceMeshStorage& m .unionPolygons(potential_support_roofs[layer_idx]); }); - std::vector additional_support_roofs(mesh.overhang_areas.size(), Polygons()); + std::vector additional_support_roofs(mesh.overhang_areas.size(), Shape()); cura::parallel_for( 0, @@ -570,18 +571,18 @@ void TreeSupportTipGenerator::calculateRoofAreas(const cura::SliceMeshStorage& m { // Roof does not have a radius, so remove it using offset. Note that there is no 0 radius avoidance, and it would not be identical with the avoidance offset with // -radius. This is intentional here, as support roof is still valid if only a part of the tip may reach it. - Polygons forbidden_here = volumes_ - .getAvoidance( - config_.getRadius(0), - layer_idx, - (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, - config_.support_rests_on_model, - ! xy_overrides_) - .offset(-(config_.getRadius(0)), ClipperLib::jtRound); + Shape forbidden_here = volumes_ + .getAvoidance( + config_.getRadius(0), + layer_idx, + (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, + config_.support_rests_on_model, + ! xy_overrides_) + .offset(-(config_.getRadius(0)), ClipperLib::jtRound); if (! force_minimum_roof_area_) { - Polygons fuzzy_area = Polygons(); + Shape fuzzy_area = Shape(); // the roof will be combined with roof above and below, to see if a part of this roof may be part of a valid roof further up/down. // This prevents the situation where a roof gets removed even tough its area would contribute to a (better) printable roof area further down. @@ -589,25 +590,25 @@ void TreeSupportTipGenerator::calculateRoofAreas(const cura::SliceMeshStorage& m -LayerIndex{ std::min(layer_idx, LayerIndex{ support_roof_layers_ }) }, LayerIndex{ std::min(LayerIndex{ potential_support_roofs.size() - layer_idx }, LayerIndex{ support_roof_layers_ + 1 }) })) { - fuzzy_area.add(support_roof_drawn_[layer_idx + layer_offset]); - fuzzy_area.add(potential_support_roofs[layer_idx + layer_offset]); + fuzzy_area.push_back(support_roof_drawn_[layer_idx + layer_offset]); + fuzzy_area.push_back(potential_support_roofs[layer_idx + layer_offset]); } fuzzy_area = fuzzy_area.unionPolygons(); fuzzy_area.removeSmallAreas(std::max(minimum_roof_area_, tip_roof_size_)); - for (Polygons potential_roof : potential_support_roofs[layer_idx].difference(forbidden_here).splitIntoParts()) + for (Shape potential_roof : potential_support_roofs[layer_idx].difference(forbidden_here).splitIntoParts()) { if (! potential_roof.intersection(fuzzy_area).empty()) { - additional_support_roofs[layer_idx].add(potential_roof); + additional_support_roofs[layer_idx].push_back(potential_roof); } } } else { - Polygons valid_roof = potential_support_roofs[layer_idx].difference(forbidden_here); + Shape valid_roof = potential_support_roofs[layer_idx].difference(forbidden_here); valid_roof.removeSmallAreas(std::max(minimum_roof_area_, tip_roof_size_)); - additional_support_roofs[layer_idx].add(valid_roof); + additional_support_roofs[layer_idx].push_back(valid_roof); } } }); @@ -642,11 +643,11 @@ void TreeSupportTipGenerator::addPointAsInfluenceArea( } Polygon circle; Polygon base_circle = TreeSupportBaseCircle::getBaseCircle(); - for (Point2LL corner : base_circle) + for (const Point2LL& corner : base_circle) { - circle.add(p.first + corner); + circle.push_back(p.first + corner); } - Polygons area = circle.offset(0); + Shape area = circle.offset(0); { std::lock_guard critical_section_movebounds(critical_move_bounds_); if (! already_inserted_[insert_layer].count(p.first / ((config_.min_radius + 1) / 10))) @@ -667,7 +668,7 @@ void TreeSupportTipGenerator::addPointAsInfluenceArea( skip_ovalisation, support_tree_limit_branch_reach_, support_tree_branch_reach_limit_); - elem->area_ = new Polygons(area); + elem->area_ = new Shape(area); for (Point2LL target : additional_ovalization_targets) { @@ -699,11 +700,11 @@ void TreeSupportTipGenerator::addLinesAsInfluenceAreas( std::function)> evaluateRoofWillGenerate = [&](std::pair p) { Polygon roof_circle; - for (Point2LL corner : TreeSupportBaseCircle::getBaseCircle()) + for (const Point2LL& corner : TreeSupportBaseCircle::getBaseCircle()) { - roof_circle.add(p.first + corner * std::max(config_.min_radius / TreeSupportBaseCircle::base_radius, coord_t(1))); + roof_circle.push_back(p.first + corner * std::max(config_.min_radius / TreeSupportBaseCircle::base_radius, coord_t(1))); } - Polygons area = roof_circle.offset(0); + Shape area = roof_circle.offset(0); return ! TreeSupportUtils::generateSupportInfillLines(area, config_, true, insert_layer_idx - dtt_roof_tip, support_roof_line_distance_, cross_fill_provider_, true) .empty(); }; @@ -732,27 +733,27 @@ void TreeSupportTipGenerator::addLinesAsInfluenceAreas( } // Add all tips as roof to the roof storage. - Polygons added_roofs; + Shape added_roofs; for (LineInformation line : lines) { for (std::pair p : line) { Polygon roof_circle; - for (Point2LL corner : TreeSupportBaseCircle::getBaseCircle()) + for (const Point2LL& corner : TreeSupportBaseCircle::getBaseCircle()) { - roof_circle.add(p.first + corner * std::max(config_.min_radius / TreeSupportBaseCircle::base_radius, coord_t(1))); + roof_circle.push_back(p.first + corner * std::max(config_.min_radius / TreeSupportBaseCircle::base_radius, coord_t(1))); } - added_roofs.add(roof_circle); + added_roofs.push_back(roof_circle); } } added_roofs = added_roofs.unionPolygons(); { std::lock_guard critical_section_roof(critical_roof_tips_); - roof_tips_drawn_[insert_layer_idx - dtt_roof_tip].add(added_roofs); + roof_tips_drawn_[insert_layer_idx - dtt_roof_tip].push_back(added_roofs); if (dtt_roof_tip == 0) { - support_roof_drawn_fractional_[insert_layer_idx].add(added_roofs); + support_roof_drawn_fractional_[insert_layer_idx].push_back(added_roofs); } } } @@ -795,7 +796,7 @@ void TreeSupportTipGenerator::addLinesAsInfluenceAreas( void TreeSupportTipGenerator::removeUselessAddedPoints( std::vector>& move_bounds, SliceDataStorage& storage, - std::vector& additional_support_areas) + std::vector& additional_support_areas) { cura::parallel_for( 0, @@ -805,9 +806,9 @@ void TreeSupportTipGenerator::removeUselessAddedPoints( if (layer_idx + 1 < storage.support.supportLayers.size()) { std::vector to_be_removed; - Polygons roof_on_layer_above = use_fake_roof_ ? support_roof_drawn_[layer_idx + 1] - : storage.support.supportLayers[layer_idx + 1].support_roof.unionPolygons(additional_support_areas[layer_idx + 1]); - Polygons roof_on_layer + Shape roof_on_layer_above = use_fake_roof_ ? support_roof_drawn_[layer_idx + 1] + : storage.support.supportLayers[layer_idx + 1].support_roof.unionPolygons(additional_support_areas[layer_idx + 1]); + Shape roof_on_layer = use_fake_roof_ ? support_roof_drawn_[layer_idx] : storage.support.supportLayers[layer_idx].support_roof.unionPolygons(additional_support_areas[layer_idx]); for (TreeSupportElement* elem : move_bounds[layer_idx]) @@ -846,7 +847,7 @@ void TreeSupportTipGenerator::generateTips( SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector>& move_bounds, - std::vector& additional_support_areas, + std::vector& additional_support_areas, std::vector>& placed_fake_roof_areas) { std::vector> new_tips(move_bounds.size()); @@ -877,7 +878,7 @@ void TreeSupportTipGenerator::generateTips( return; // This is a continue if imagined in a loop context. } - Polygons relevant_forbidden = volumes_.getAvoidance( + Shape relevant_forbidden = volumes_.getAvoidance( config_.getRadius(0), layer_idx, (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, @@ -888,7 +889,7 @@ void TreeSupportTipGenerator::generateTips( = relevant_forbidden.offset(EPSILON) .unionPolygons(); // Prevent rounding errors down the line, points placed directly on the line of the forbidden area may not be added otherwise. - std::function generateLines = [&](const Polygons& area, bool roof, LayerIndex generate_layer_idx) + std::function generateLines = [&](const Shape& area, bool roof, LayerIndex generate_layer_idx) { coord_t upper_line_distance = support_supporting_branch_distance_; coord_t line_distance = std::max(roof ? support_roof_line_distance_ : support_tree_branch_distance_, upper_line_distance); @@ -905,21 +906,21 @@ void TreeSupportTipGenerator::generateTips( }; - std::vector> overhang_processing; + std::vector> overhang_processing; // ^^^ Every overhang has saved if a roof should be generated for it. // This can NOT be done in the for loop as an area may NOT have a roof even if it is larger than the minimum_roof_area when it is only larger because of the support // horizontal expansion and it would not have a roof if the overhang is offset by support roof horizontal expansion instead. (At least this is the current behavior // of the regular support) - Polygons core_overhang = mesh.overhang_areas[layer_idx + z_distance_delta_]; + Shape core_overhang = mesh.overhang_areas[layer_idx + z_distance_delta_]; if (support_roof_layers_ && layer_idx + 1 < support_roof_drawn_.size()) { core_overhang = core_overhang.difference(support_roof_drawn_[layer_idx]); - for (Polygons roof_part : support_roof_drawn_[layer_idx + 1] - .difference(support_roof_drawn_[layer_idx]) - .splitIntoParts(true)) // If there is a roof, the roof will be one layer above the tips. + for (Shape roof_part : support_roof_drawn_[layer_idx + 1] + .difference(support_roof_drawn_[layer_idx]) + .splitIntoParts(true)) // If there is a roof, the roof will be one layer above the tips. { //^^^Technically one should also subtract the avoidance of radius 0 (similarly how calculated in calculateRoofArea), as there can be some rounding errors // introduced since then. But this does not fully prevent some rounding errors either way, so just handle the error later. @@ -927,7 +928,7 @@ void TreeSupportTipGenerator::generateTips( } } - Polygons overhang_regular = TreeSupportUtils::safeOffsetInc( + Shape overhang_regular = TreeSupportUtils::safeOffsetInc( core_overhang, support_outset_, relevant_forbidden, @@ -936,8 +937,7 @@ void TreeSupportTipGenerator::generateTips( 1, config_.support_line_distance / 2, &config_.simplifier); - Polygons remaining_overhang - = core_overhang.offset(support_outset_).difference(overhang_regular.offset(config_.support_line_width * 0.5)).intersection(relevant_forbidden); + Shape remaining_overhang = core_overhang.offset(support_outset_).difference(overhang_regular.offset(config_.support_line_width * 0.5)).intersection(relevant_forbidden); // Offset ensures that areas that could be supported by a part of a support line, are not considered unsupported overhang @@ -951,7 +951,7 @@ void TreeSupportTipGenerator::generateTips( ? std::min(config_.support_line_width / 8, extra_outset - extra_total_offset_acc) : std::min(circle_length_to_half_linewidth_change, extra_outset - extra_total_offset_acc); extra_total_offset_acc += offset_current_step; - Polygons overhang_offset = TreeSupportUtils::safeOffsetInc( + Shape overhang_offset = TreeSupportUtils::safeOffsetInc( overhang_regular, 1.5 * extra_total_offset_acc, volumes_.getCollision(0, layer_idx, true), @@ -962,7 +962,7 @@ void TreeSupportTipGenerator::generateTips( &config_.simplifier); remaining_overhang = remaining_overhang.difference(overhang_offset.unionPolygons(support_roof_drawn_[layer_idx].offset(1.5 * extra_total_offset_acc))) .unionPolygons(); // overhang_offset is combined with roof, as all area that has a roof, is already supported by said roof. - Polygons next_overhang = TreeSupportUtils::safeOffsetInc( + Shape next_overhang = TreeSupportUtils::safeOffsetInc( remaining_overhang, extra_total_offset_acc, volumes_.getCollision(0, layer_idx, true), @@ -979,7 +979,7 @@ void TreeSupportTipGenerator::generateTips( // (while adding them an infinite amount of layers down would technically be closer the setting description, it would not produce reasonable results. ) if (xy_overrides_) { - for (Polygons& remaining_overhang_part : remaining_overhang.splitIntoParts(false)) + for (Shape& remaining_overhang_part : remaining_overhang.splitIntoParts(false)) { if (remaining_overhang_part.area() <= MM2_2INT(minimum_support_area_)) { @@ -987,7 +987,7 @@ void TreeSupportTipGenerator::generateTips( } std::vector overhang_lines; - Polygons polylines = ensureMaximumDistancePolyline(generateLines(remaining_overhang_part, false, layer_idx), config_.min_radius, 1, false); + OpenLinesSet polylines = ensureMaximumDistancePolyline(generateLines(remaining_overhang_part, false, layer_idx), config_.min_radius, 1, false); // ^^^ Support_line_width to form a line here as otherwise most will be unsupported. // Technically this violates branch distance, but not only is this the only reasonable choice, // but it ensures consistent behavior as some infill patterns generate each line segment as its own polyline part causing a similar line forming behavior. @@ -1011,7 +1011,7 @@ void TreeSupportTipGenerator::generateTips( for (size_t lag_ctr = 1; lag_ctr <= max_overhang_insert_lag_ && ! overhang_lines.empty() && layer_idx - coord_t(lag_ctr) >= 1; lag_ctr++) { // get least restricted avoidance for layer_idx-lag_ctr - Polygons relevant_forbidden_below = volumes_.getAvoidance( + Shape relevant_forbidden_below = volumes_.getAvoidance( config_.getRadius(0), layer_idx - lag_ctr, (only_gracious_ || ! config_.support_rests_on_model) ? AvoidanceType::FAST : AvoidanceType::COLLISION, @@ -1054,23 +1054,23 @@ void TreeSupportTipGenerator::generateTips( overhang_regular.removeSmallAreas(minimum_support_area_); - for (Polygons support_part : overhang_regular.splitIntoParts(true)) + for (Shape support_part : overhang_regular.splitIntoParts(true)) { overhang_processing.emplace_back(support_part, false); } - for (std::pair overhang_pair : overhang_processing) + for (std::pair overhang_pair : overhang_processing) { const bool roof_allowed_for_this_part = overhang_pair.second; - Polygons overhang_outset = overhang_pair.first; - const size_t min_support_points = std::max(coord_t(1), std::min(coord_t(EPSILON), overhang_outset.polygonLength() / connect_length_)); + Shape overhang_outset = overhang_pair.first; + const size_t min_support_points = std::max(coord_t(1), std::min(coord_t(EPSILON), overhang_outset.length() / connect_length_)); std::vector overhang_lines; bool only_lines = true; // The tip positions are determined here. // todo can cause inconsistent support density if a line exactly aligns with the model - Polygons polylines = ensureMaximumDistancePolyline( + OpenLinesSet polylines = ensureMaximumDistancePolyline( generateLines(overhang_outset, roof_allowed_for_this_part, layer_idx + roof_allowed_for_this_part), ! roof_allowed_for_this_part ? config_.min_radius * 2 : use_fake_roof_ ? support_supporting_branch_distance_ @@ -1089,7 +1089,7 @@ void TreeSupportTipGenerator::generateTips( only_lines = false; // Add the outer wall (of the overhang) to ensure it is correct supported instead. // Try placing the support points in a way that they fully support the outer wall, instead of just the with half of the support line width. - Polygons reduced_overhang_outset = overhang_outset.offset(-config_.support_line_width / 2.2); + Shape reduced_overhang_outset = overhang_outset.offset(-config_.support_line_width / 2.2); // ^^^ It's assumed that even small overhangs are over one line width wide, so lets try to place the support points in a way that the full support area // generated from them will support the overhang. // (If this is not done it may only be half). This WILL NOT be the case when supporting an angle of about < 60� so there is a fallback, as some support is @@ -1114,7 +1114,7 @@ void TreeSupportTipGenerator::generateTips( if (overhang_lines.empty()) // some error handling and logging { - Polygons enlarged_overhang_outset = overhang_outset.offset(config_.getRadius(0) + FUDGE_LENGTH / 2, ClipperLib::jtRound).difference(relevant_forbidden); + Shape enlarged_overhang_outset = overhang_outset.offset(config_.getRadius(0) + FUDGE_LENGTH / 2, ClipperLib::jtRound).difference(relevant_forbidden); polylines = ensureMaximumDistancePolyline(TreeSupportUtils::toPolylines(enlarged_overhang_outset), connect_length_, min_support_points, true); overhang_lines = convertLinesToInternal(polylines, layer_idx); @@ -1157,11 +1157,11 @@ void TreeSupportTipGenerator::generateTips( if (layer_idx < additional_support_areas.size() && TreeSupportUtils::generateSupportInfillLines(roof_area, config_, true, layer_idx, support_roof_line_distance_, cross_fill_provider_, false).empty()) { - additional_support_areas[layer_idx].add(roof_area); + additional_support_areas[layer_idx].push_back(roof_area); } else { - storage.support.supportLayers[layer_idx].support_roof.add(roof_area); + storage.support.supportLayers[layer_idx].support_roof.push_back(roof_area); } } @@ -1177,7 +1177,7 @@ void TreeSupportTipGenerator::generateTips( if (config_.z_distance_top % config_.layer_height != 0 && layer_idx > 0) { // Fake roof tips would just be tips, so no need to add them here as all polygons in roof_tips_drawn_ will be empty! - Polygons all_roof_fractional = support_roof_drawn_fractional_[layer_idx - 1].intersection(support_roof_drawn_[layer_idx - 1]); + Shape all_roof_fractional = support_roof_drawn_fractional_[layer_idx - 1].intersection(support_roof_drawn_[layer_idx - 1]); placed_fake_roof_areas[layer_idx].emplace_back(all_roof_fractional, support_roof_line_distance_, true); } } @@ -1185,17 +1185,17 @@ void TreeSupportTipGenerator::generateTips( { if (config_.z_distance_top % config_.layer_height != 0 && layer_idx > 0) { - Polygons all_roof_below = support_roof_drawn_[layer_idx - 1].unionPolygons(roof_tips_drawn_[layer_idx - 1]); - Polygons all_roof_fractional = support_roof_drawn_fractional_[layer_idx - 1].intersection(all_roof_below); + Shape all_roof_below = support_roof_drawn_[layer_idx - 1].unionPolygons(roof_tips_drawn_[layer_idx - 1]); + Shape all_roof_fractional = support_roof_drawn_fractional_[layer_idx - 1].intersection(all_roof_below); storage.support.supportLayers[layer_idx].support_fractional_roof = storage.support.supportLayers[layer_idx].support_fractional_roof.unionPolygons(all_roof_fractional); // Fractional roof is a modifier applied to a roof area, which means if only the fractional roof area is set, there will be nothing as there is no roof to // modify. Because of that the fractional roof has ALSO to be added to the roof. - storage.support.supportLayers[layer_idx].support_roof.add(all_roof_fractional); + storage.support.supportLayers[layer_idx].support_roof.push_back(all_roof_fractional); } - Polygons all_roof = support_roof_drawn_[layer_idx].unionPolygons(roof_tips_drawn_[layer_idx]); + Shape all_roof = support_roof_drawn_[layer_idx].unionPolygons(roof_tips_drawn_[layer_idx]); storage.support.supportLayers[layer_idx].support_roof = storage.support.supportLayers[layer_idx].support_roof.unionPolygons(all_roof); } } diff --git a/src/WallToolPaths.cpp b/src/WallToolPaths.cpp index 055377060d..b144d0ebfb 100644 --- a/src/WallToolPaths.cpp +++ b/src/WallToolPaths.cpp @@ -13,7 +13,7 @@ #include "ExtruderTrain.h" #include "SkeletalTrapezoidation.h" -#include "utils/PolylineStitcher.h" +#include "utils/ExtrusionLineStitcher.h" #include "utils/Simplify.h" #include "utils/SparsePointGrid.h" //To stitch the inner contour. #include "utils/actions/smooth.h" @@ -23,7 +23,7 @@ namespace cura { WallToolPaths::WallToolPaths( - const Polygons& outline, + const Shape& outline, const coord_t nominal_bead_width, const size_t inset_count, const coord_t wall_0_inset, @@ -47,7 +47,7 @@ WallToolPaths::WallToolPaths( } WallToolPaths::WallToolPaths( - const Polygons& outline, + const Shape& outline, const coord_t bead_width_0, const coord_t bead_width_x, const size_t inset_count, @@ -87,7 +87,7 @@ const std::vector& WallToolPaths::generate() // Simplify outline for boost::voronoi consumption. Absolutely no self intersections or near-self intersections allowed: // TODO: Open question: Does this indeed fix all (or all-but-one-in-a-million) cases for manifold but otherwise possibly complex polygons? - Polygons prepared_outline = outline_.offset(-open_close_distance).offset(open_close_distance * 2).offset(-open_close_distance); + Shape prepared_outline = outline_.offset(-open_close_distance).offset(open_close_distance * 2).offset(-open_close_distance); scripta::log("prepared_outline_0", prepared_outline, section_type_, layer_idx_); prepared_outline.removeSmallAreas(small_area_length_ * small_area_length_, false); prepared_outline = Simplify(settings_).polygon(prepared_outline); @@ -95,9 +95,9 @@ const std::vector& WallToolPaths::generate() { // No need to smooth support walls auto smoother = actions::smooth(settings_); - for (auto& polygon : prepared_outline) + for (Polygon& polygon : prepared_outline) { - polygon = smoother(polygon); + polygon.setPoints(smoother(polygon.getPoints())); } } @@ -253,7 +253,7 @@ void WallToolPaths::stitchToolPaths(std::vector& toolpaths, VariableWidthLines stitched_polylines; VariableWidthLines closed_polygons; - PolylineStitcher::stitch(wall_lines, stitched_polylines, closed_polygons, stitch_distance); + ExtrusionLineStitcher::stitch(wall_lines, stitched_polylines, closed_polygons, stitch_distance); wall_lines = stitched_polylines; // replace input toolpaths with stitched polylines for (ExtrusionLine& wall_polygon : closed_polygons) @@ -286,7 +286,7 @@ void WallToolPaths::removeSmallLines(std::vector& toolpaths) { min_width = std::min(min_width, j.w_); } - if (line.is_odd_ && ! line.is_closed_ && shorterThan(line, min_width / 2)) + if (line.is_odd_ && ! line.is_closed_ && line.shorterThan(min_width / 2)) { // remove line line = std::move(inset.back()); inset.erase(--inset.end()); @@ -417,7 +417,7 @@ void WallToolPaths::separateOutInnerContour() inner_contour_ = inner_contour_.processEvenOdd(); } -const Polygons& WallToolPaths::getInnerContour() +const Shape& WallToolPaths::getInnerContour() { if (! toolpaths_generated_ && inset_count_ > 0) { diff --git a/src/WallsComputation.cpp b/src/WallsComputation.cpp index ea8f959746..752d1cf0c9 100644 --- a/src/WallsComputation.cpp +++ b/src/WallsComputation.cpp @@ -84,7 +84,8 @@ void WallsComputation::generateWalls(SliceLayerPart* part, SectionType section_t part->wall_toolpaths = wall_tool_paths.getToolPaths(); part->inner_area = wall_tool_paths.getInnerContour(); } - part->outline = PolygonsPart{ Simplify(settings_).polygon(part->outline) }; + + part->outline = SingleShape{ Simplify(settings_).polygon(part->outline) }; part->print_outline = part->outline; } diff --git a/src/bridge.cpp b/src/bridge.cpp index 554d81d5b1..55b7b8e8cd 100644 --- a/src/bridge.cpp +++ b/src/bridge.cpp @@ -3,31 +3,33 @@ #include "bridge.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" #include "settings/types/Ratio.h" #include "sliceDataStorage.h" #include "utils/AABB.h" -#include "utils/polygon.h" namespace cura { double bridgeAngle( const Settings& settings, - const Polygons& skin_outline, + const Shape& skin_outline, const SliceDataStorage& storage, const unsigned layer_nr, const unsigned bridge_layer, const SupportLayer* support_layer, - Polygons& supported_regions) + Shape& supported_regions) { assert(! skin_outline.empty()); AABB boundary_box(skin_outline); // To detect if we have a bridge, first calculate the intersection of the current layer with the previous layer. // This gives us the islands that the layer rests on. - Polygons islands; + Shape islands; - Polygons prev_layer_outline; // we also want the complete outline of the previous layer + Shape prev_layer_outline; // we also want the complete outline of the previous layer const Ratio sparse_infill_max_density = settings.get("bridge_sparse_infill_max_density"); @@ -44,17 +46,17 @@ double bridgeAngle( for (const SliceLayerPart& prev_layer_part : mesh.layers[layer_nr - bridge_layer].parts) { - Polygons solid_below(prev_layer_part.outline); + Shape solid_below(prev_layer_part.outline); if (bridge_layer == 1 && part_has_sparse_infill) { solid_below = solid_below.difference(prev_layer_part.getOwnInfillArea()); } - prev_layer_outline.add(solid_below); // not intersected with skin + prev_layer_outline.push_back(solid_below); // not intersected with skin if (! boundary_box.hit(prev_layer_part.boundaryBox)) continue; - islands.add(skin_outline.intersection(solid_below)); + islands.push_back(skin_outline.intersection(solid_below)); } } } @@ -73,12 +75,12 @@ double bridgeAngle( AABB support_roof_bb(support_layer->support_roof); if (boundary_box.hit(support_roof_bb)) { - prev_layer_outline.add(support_layer->support_roof); // not intersected with skin + prev_layer_outline.push_back(support_layer->support_roof); // not intersected with skin - Polygons supported_skin(skin_outline.intersection(support_layer->support_roof)); + Shape supported_skin(skin_outline.intersection(support_layer->support_roof)); if (! supported_skin.empty()) { - supported_regions.add(supported_skin); + supported_regions.push_back(supported_skin); } } } @@ -89,12 +91,12 @@ double bridgeAngle( AABB support_part_bb(support_part.getInfillArea()); if (boundary_box.hit(support_part_bb)) { - prev_layer_outline.add(support_part.getInfillArea()); // not intersected with skin + prev_layer_outline.push_back(support_part.getInfillArea()); // not intersected with skin - Polygons supported_skin(skin_outline.intersection(support_part.getInfillArea())); + Shape supported_skin(skin_outline.intersection(support_part.getInfillArea())); if (! supported_skin.empty()) { - supported_regions.add(supported_skin); + supported_regions.push_back(supported_skin); } } } @@ -111,45 +113,43 @@ double bridgeAngle( if (support_threshold > 0 && (supported_regions.area() / (skin_outline.area() + 1)) < support_threshold) { - Polygons bb_poly; - bb_poly.add(boundary_box.toPolygon()); + Shape bb_poly; + bb_poly.push_back(boundary_box.toPolygon()); // airBelow is the region below the skin that is not supported, it extends well past the boundary of the skin. // It needs to be shrunk slightly so that the vertices of the skin polygon that would otherwise fall exactly on // the air boundary do appear to be supported const coord_t bb_max_dim = std::max(boundary_box.max_.X - boundary_box.min_.X, boundary_box.max_.Y - boundary_box.min_.Y); - const Polygons air_below(bb_poly.offset(bb_max_dim).difference(prev_layer_outline).offset(-10)); + const Shape air_below(bb_poly.offset(bb_max_dim).difference(prev_layer_outline).offset(-10)); - Polygons skin_perimeter_lines; - for (ConstPolygonRef poly : skin_outline) + OpenLinesSet skin_perimeter_lines; + for (const Polygon& poly : skin_outline) { - if (poly.empty()) - continue; - skin_perimeter_lines.add(poly); - skin_perimeter_lines.back().emplace_back(poly.front()); + if (! poly.empty()) + { + skin_perimeter_lines.emplace_back(poly.toPseudoOpenPolyline()); + } } - Polygons skin_perimeter_lines_over_air(air_below.intersectionPolyLines(skin_perimeter_lines)); + OpenLinesSet skin_perimeter_lines_over_air(air_below.intersection(skin_perimeter_lines)); if (skin_perimeter_lines_over_air.size()) { // one or more edges of the skin region are unsupported, determine the longest coord_t max_dist2 = 0; double line_angle = -1; - for (PolygonRef air_line : skin_perimeter_lines_over_air) + for (const OpenPolyline& air_line : skin_perimeter_lines_over_air) { - Point2LL p0 = air_line[0]; - for (unsigned i = 1; i < air_line.size(); ++i) + for (auto iterator = air_line.beginSegments(); iterator != air_line.endSegments(); ++iterator) { - const Point2LL& p1(air_line[i]); - coord_t dist2 = vSize2(p0 - p1); + const Point2LL vector = (*iterator).start - (*iterator).end; + coord_t dist2 = vSize2(vector); if (dist2 > max_dist2) { max_dist2 = dist2; - line_angle = angle(p0 - p1); + line_angle = angle(vector); } - p0 = p1; } } return line_angle; diff --git a/src/communication/ArcusCommunication.cpp b/src/communication/ArcusCommunication.cpp index 80f27db79d..0f0fd4e6d3 100644 --- a/src/communication/ArcusCommunication.cpp +++ b/src/communication/ArcusCommunication.cpp @@ -33,11 +33,11 @@ #include "communication/ArcusCommunicationPrivate.h" //Our PIMPL. #include "communication/Listener.h" //To listen to the Arcus socket. #include "communication/SliceDataStruct.h" //To store sliced layer data. +#include "geometry/Polygon.h" #include "plugins/slots.h" #include "settings/types/LayerIndex.h" //To point to layers. #include "settings/types/Velocity.h" //To send to layer view how fast stuff is printing. #include "utils/channel.h" -#include "utils/polygon.h" namespace cura { @@ -236,7 +236,7 @@ class ArcusCommunication::PathCompiler * \param thickness The layer thickness of the polygon. * \param velocity How fast the polygon is printed. */ - void sendPolygon(const PrintFeatureType& print_feature_type, const ConstPolygonRef& polygon, const coord_t& width, const coord_t& thickness, const Velocity& velocity) + void sendPolygon(const PrintFeatureType& print_feature_type, const Polygon& polygon, const coord_t& width, const coord_t& thickness, const Velocity& velocity) { if (polygon.size() < 2) // Don't send single points or empty polygons. { @@ -444,19 +444,14 @@ void ArcusCommunication::sendOptimizedLayerData() data.slice_data.clear(); } -void ArcusCommunication::sendPolygon( - const PrintFeatureType& type, - const ConstPolygonRef& polygon, - const coord_t& line_width, - const coord_t& line_thickness, - const Velocity& velocity) +void ArcusCommunication::sendPolygon(const PrintFeatureType& type, const Polygon& polygon, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity) { path_compiler->sendPolygon(type, polygon, line_width, line_thickness, velocity); } -void ArcusCommunication::sendPolygons(const PrintFeatureType& type, const Polygons& polygons, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity) +void ArcusCommunication::sendPolygons(const PrintFeatureType& type, const Shape& polygons, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity) { - for (const std::vector& polygon : polygons) + for (const Polygon& polygon : polygons) { path_compiler->sendPolygon(type, polygon, line_width, line_thickness, velocity); } diff --git a/src/communication/CommandLine.cpp b/src/communication/CommandLine.cpp index 3c31f1f6cd..12e04adc13 100644 --- a/src/communication/CommandLine.cpp +++ b/src/communication/CommandLine.cpp @@ -8,12 +8,16 @@ #include #include //To check if files exist. #include //For std::accumulate. +#include #include //Loading JSON documents to get settings from them. #include #include #include #include +#include +#include #include +#include #include #include @@ -22,6 +26,7 @@ #include "Application.h" //To get the extruders for material estimates. #include "ExtruderTrain.h" #include "FffProcessor.h" //To start a slice and get time estimates. +#include "MeshGroup.h" #include "Slice.h" #include "utils/Matrix4x3D.h" //For the mesh_rotation_matrix setting. #include "utils/format/filesystem_path.h" @@ -66,10 +71,10 @@ void CommandLine::sendLineTo(const PrintFeatureType&, const Point2LL&, const coo void CommandLine::sendOptimizedLayerData() { } -void CommandLine::sendPolygon(const PrintFeatureType&, const ConstPolygonRef&, const coord_t&, const coord_t&, const Velocity&) +void CommandLine::sendPolygon(const PrintFeatureType&, const Polygon&, const coord_t&, const coord_t&, const Velocity&) { } -void CommandLine::sendPolygons(const PrintFeatureType&, const Polygons&, const coord_t&, const coord_t&, const Velocity&) +void CommandLine::sendPolygons(const PrintFeatureType&, const Shape&, const coord_t&, const coord_t&, const Velocity&) { } void CommandLine::setExtruderForSend(const ExtruderTrain&) @@ -354,6 +359,127 @@ void CommandLine::sliceNext() last_settings->add(key, value); break; } + case 'r': + { + /* + * read in resolved values from a json file. The json format of the file resolved settings is the following: + * + * ``` + * { + * "global": [SETTINGS], + * "extruder.0": [SETTINGS], + * "extruder.1": [SETTINGS], + * "model.stl": [SETTINGS] + * } + * ``` + * where `[SETTINGS]` follow the schema + * ``` + * { + * [key: string]: bool | string | number | number[] | number[][] + * } + * ``` + * There can be any number of extruders (denoted with `extruder.n`) and any number of models (denoted with `[modelname].stl`). + * The key of the model values will also be the filename of the relevant model, when running CuraEngine with this option the + * model file with that same name _must_ be in the same folder as the resolved settings json. + */ + + argument_index++; + if (argument_index >= arguments_.size()) + { + spdlog::error("Missing setting name and value with -r argument."); + exit(1); + } + argument = arguments_[argument_index]; + const auto settings = readResolvedJsonValues(std::filesystem::path{ argument }); + + if (! settings.has_value()) + { + spdlog::error("Failed to load JSON file: {}", argument); + exit(1); + } + + constexpr std::string_view global_identifier = "global"; + constexpr std::string_view extruder_identifier = "extruder."; + constexpr std::string_view model_identifier = "model."; + constexpr std::string_view limit_to_extruder_identifier = "limit_to_extruder"; + + // Split the settings into global, extruder and model settings. This is needed since the order in which the settings are applied is important. + // first global settings, then extruder settings, then model settings. The order of these stacks is not enforced in the JSON files. + std::unordered_map global_settings; + container_setting_map extruder_settings; + container_setting_map model_settings; + std::unordered_map limit_to_extruder; + + for (const auto& [key, values] : settings.value()) + { + if (key == global_identifier) + { + global_settings = values; + } + else if (key.starts_with(extruder_identifier)) + { + extruder_settings[key] = values; + } + else if (key == limit_to_extruder_identifier) + { + limit_to_extruder = values; + } + else + { + model_settings[key] = values; + } + } + + for (const auto& [setting_key, setting_value] : global_settings) + { + slice.scene.settings.add(setting_key, setting_value); + } + + for (const auto& [key, values] : extruder_settings) + { + const auto extruder_nr = std::stoi(key.substr(extruder_identifier.size())); + while (slice.scene.extruders.size() <= static_cast(extruder_nr)) + { + slice.scene.extruders.emplace_back(extruder_nr, &slice.scene.settings); + } + for (const auto& [setting_key, setting_value] : values) + { + slice.scene.extruders[extruder_nr].settings_.add(setting_key, setting_value); + } + } + + for (const auto& [key, values] : model_settings) + { + const auto& model_name = key; + + cura::MeshGroup mesh_group; + for (const auto& [setting_key, setting_value] : values) + { + mesh_group.settings.add(setting_key, setting_value); + } + + const auto transformation = mesh_group.settings.get("mesh_rotation_matrix"); + const auto extruder_nr = mesh_group.settings.get("extruder_nr"); + + if (! loadMeshIntoMeshGroup(&mesh_group, model_name.c_str(), transformation, slice.scene.extruders[extruder_nr].settings_)) + { + spdlog::error("Failed to load model: {}. (error number {})", model_name, errno); + exit(1); + } + + slice.scene.mesh_groups.push_back(std::move(mesh_group)); + } + for (const auto& [key, value] : limit_to_extruder) + { + const auto extruder_nr = std::stoi(value.substr(extruder_identifier.size())); + if (extruder_nr >= 0) + { + slice.scene.limit_to_extruder[key] = &slice.scene.extruders[extruder_nr]; + } + } + + break; + } default: { spdlog::error("Unknown option: -{}", argument[1]); @@ -586,6 +712,55 @@ void CommandLine::loadJSONSettings(const rapidjson::Value& element, Settings& se } } +std::optional CommandLine::readResolvedJsonValues(const std::filesystem::path& json_filename) +{ + std::ifstream file(json_filename, std::ios::binary); + if (! file) + { + spdlog::error("Couldn't open JSON file: {}", json_filename); + return std::nullopt; + } + + std::vector read_buffer(std::istreambuf_iterator(file), {}); + rapidjson::MemoryStream memory_stream(read_buffer.data(), read_buffer.size()); + + rapidjson::Document json_document; + json_document.ParseStream(memory_stream); + if (json_document.HasParseError()) + { + spdlog::error("Error parsing JSON (offset {}): {}", json_document.GetErrorOffset(), GetParseError_En(json_document.GetParseError())); + return std::nullopt; + } + + return readResolvedJsonValues(json_document); +} + +std::optional CommandLine::readResolvedJsonValues(const rapidjson::Document& document) +{ + if (! document.IsObject()) + { + return std::nullopt; + } + + container_setting_map result; + for (rapidjson::Value::ConstMemberIterator resolved_key = document.MemberBegin(); resolved_key != document.MemberEnd(); resolved_key++) + { + std::unordered_map values; + for (rapidjson::Value::ConstMemberIterator resolved_value = resolved_key->value.MemberBegin(); resolved_value != resolved_key->value.MemberEnd(); resolved_value++) + { + std::string value_string; + if (! jsonValue2Str(resolved_value->value, value_string)) + { + spdlog::warn("Unrecognized data type in JSON setting {}", resolved_value->name.GetString()); + continue; + } + values.emplace(resolved_value->name.GetString(), value_string); + } + result.emplace(resolved_key->name.GetString(), std::move(values)); + } + return result; +} + std::string CommandLine::findDefinitionFile(const std::string& definition_id, const std::vector& search_directories) { for (const auto& search_directory : search_directories) diff --git a/src/gcodeExport.cpp b/src/gcodeExport.cpp index 8a3cc11538..98d95d3f07 100644 --- a/src/gcodeExport.cpp +++ b/src/gcodeExport.cpp @@ -1,12 +1,12 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "gcodeExport.h" -#include +#include #include #include -#include +#include #include @@ -18,6 +18,7 @@ #include "WipeScriptConfig.h" #include "communication/Communication.h" //To send layer view data. #include "settings/types/LayerIndex.h" +#include "sliceDataStorage.h" #include "utils/Date.h" #include "utils/string.h" // MMtoStream, PrecisionedDouble diff --git a/src/geometry/ClosedPolyline.cpp b/src/geometry/ClosedPolyline.cpp new file mode 100644 index 0000000000..c4890c48f1 --- /dev/null +++ b/src/geometry/ClosedPolyline.cpp @@ -0,0 +1,57 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/ClosedPolyline.h" + +#include + +#include "geometry/OpenPolyline.h" + +namespace cura +{ + +size_t ClosedPolyline::segmentsCount() const +{ + if (explicitely_closed_) + { + return size() >= 3 ? size() - 1 : 0; + } + return size() >= 2 ? size() : 0; +} + +bool ClosedPolyline::isValid() const +{ + return size() >= (explicitely_closed_ ? 4 : 3); +} + +bool ClosedPolyline::inside(const Point2LL& p, bool border_result) const +{ + int res = ClipperLib::PointInPolygon(p, getPoints()); + if (res == -1) + { + return border_result; + } + return res == 1; +} + +bool ClosedPolyline::inside(const ClipperLib::Path& polygon) const +{ + return ranges::all_of( + *this, + [&polygon](const auto& point) + { + return ClipperLib::PointInPolygon(point, polygon); + }); +} + +OpenPolyline ClosedPolyline::toPseudoOpenPolyline() const +{ + OpenPolyline open_polyline(getPoints()); + if (hasClosingSegment()) + { + open_polyline.push_back(open_polyline.getPoints().front()); + } + return open_polyline; +} + +} // namespace cura diff --git a/src/geometry/LinesSet.cpp b/src/geometry/LinesSet.cpp new file mode 100644 index 0000000000..039931812a --- /dev/null +++ b/src/geometry/LinesSet.cpp @@ -0,0 +1,349 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/LinesSet.h" + +#include + +#include "geometry/ClosedLinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" + +namespace cura +{ + +template +bool LinesSet::checkAdd(const LineType& line, CheckNonEmptyParam check_non_empty) +{ + switch (check_non_empty) + { + case CheckNonEmptyParam::EvenIfEmpty: + return true; + case CheckNonEmptyParam::OnlyIfNotEmpty: + return ! line.empty(); + case CheckNonEmptyParam::OnlyIfValid: + return line.isValid(); + } + + return false; +} + +template +void LinesSet::push_back(const LineType& line, CheckNonEmptyParam check_non_empty) +{ + if (checkAdd(line, check_non_empty)) + { + lines_.push_back(line); + } +} + +template +void LinesSet::push_back(LineType&& line, CheckNonEmptyParam check_non_empty) +{ + if (checkAdd(line, check_non_empty)) + { + lines_.push_back(std::move(line)); + } +} + +template +template +void LinesSet::push_back(LinesSet&& lines_set) +{ + reserve(size() + lines_set.size()); + for (OtherLineType& line : lines_set) + { + emplace_back(std::move(line)); + } +} + +template +size_t LinesSet::pointCount() const +{ + return std::accumulate( + lines_.begin(), + lines_.end(), + 0ULL, + [](size_t total, const LineType& line) + { + return total + line.size(); + }); +} + +template<> +void OpenLinesSet::addSegment(const Point2LL& from, const Point2LL& to) +{ + lines_.emplace_back(std::initializer_list{ from, to }); +} + +template +void LinesSet::removeAt(size_t index) +{ + if (lines_.size() == 1) + { + lines_.clear(); + } + else if (lines_.size() > 1) + { + assert(index < lines_.size()); + if (index < lines_.size() - 1) + { + lines_[index] = std::move(lines_.back()); + } + lines_.resize(lines_.size() - 1); + } +} + +template +void LinesSet::splitIntoSegments(OpenLinesSet& result) const +{ + for (const LineType& line : lines_) + { + line.splitIntoSegments(result); + } +} + +template +OpenLinesSet LinesSet::splitIntoSegments() const +{ + OpenLinesSet result; + for (const LineType& line : lines_) + { + line.splitIntoSegments(result); + } + return result; +} + +template +coord_t LinesSet::length() const +{ + return std::accumulate( + lines_.begin(), + lines_.end(), + 0LL, + [](coord_t total, const LineType& line) + { + return total += line.length(); + }); +} + +template +Shape LinesSet::createTubeShape(const coord_t inner_offset, const coord_t outer_offset) const +{ + return offset(outer_offset).difference(offset(-inner_offset)); +} + +template +void LinesSet::translate(const Point2LL& delta) +{ + if (delta.X != 0 || delta.Y != 0) + { + for (LineType& line : getLines()) + { + line.translate(delta); + } + } +} + +template<> +Shape LinesSet::offset(coord_t distance, ClipperLib::JoinType join_type, double miter_limit) const +{ + if (empty()) + { + return {}; + } + if (distance == 0) + { + Shape result; + for (const ClosedPolyline& line : getLines()) + { + result.emplace_back(line.getPoints(), line.isExplicitelyClosed()); + } + return result; + } + ClipperLib::Paths ret; + ClipperLib::ClipperOffset clipper(miter_limit, 10.0); + addPaths(clipper, join_type, ClipperLib::etClosedLine); + clipper.MiterLimit = miter_limit; + clipper.Execute(ret, static_cast(distance)); + return Shape{ std::move(ret) }; +} + +template<> +Shape LinesSet::offset(coord_t distance, ClipperLib::JoinType join_type, double miter_limit) const +{ + if (empty()) + { + return {}; + } + if (distance == 0) + { + return { getLines() }; + } + ClipperLib::Paths ret; + ClipperLib::ClipperOffset clipper(miter_limit, 10.0); + Shape(getLines()).unionPolygons().addPaths(clipper, join_type, ClipperLib::etClosedPolygon); + clipper.MiterLimit = miter_limit; + clipper.Execute(ret, static_cast(distance)); + return Shape{ std::move(ret) }; +} + +template<> +Shape OpenLinesSet::offset(coord_t distance, ClipperLib::JoinType join_type, double miter_limit) const +{ + if (empty() || distance == 0) + { + return {}; + } + + ClipperLib::ClipperOffset clipper(miter_limit, 10.0); + const ClipperLib::EndType end_type{ join_type == ClipperLib::jtMiter ? ClipperLib::etOpenSquare : ClipperLib::etOpenRound }; + addPaths(clipper, join_type, end_type); + clipper.MiterLimit = miter_limit; + ClipperLib::Paths result_paths; + clipper.Execute(result_paths, static_cast(distance)); + + return Shape{ std::move(result_paths) }; +} + +template +void LinesSet::removeDegenerateVerts() +{ + for (size_t poly_idx = 0; poly_idx < lines_.size(); poly_idx++) + { + LineType& poly = lines_[poly_idx]; + const bool for_polyline = (dynamic_cast(&poly) != nullptr); + ClipperLib::Path result; + + auto is_degenerate = [](const Point2LL& last, const Point2LL& now, const Point2LL& next) + { + Point2LL last_line = now - last; + Point2LL next_line = next - now; + return dot(last_line, next_line) == -1 * vSize(last_line) * vSize(next_line); + }; + + // With polylines, skip the first and last vertex. + const size_t start_vertex = for_polyline ? 1 : 0; + const size_t end_vertex = for_polyline ? poly.size() - 1 : poly.size(); + for (size_t i = 0; i < start_vertex; ++i) + { + result.push_back(poly[i]); // Add everything before the start vertex. + } + + bool is_changed = false; + for (size_t idx = start_vertex; idx < end_vertex; idx++) + { + const Point2LL& last = (result.size() == 0) ? poly.back() : result.back(); + if (idx + 1 >= poly.size() && result.size() == 0) + { + break; + } + const Point2LL& next = (idx + 1 >= poly.size()) ? result[0] : poly[idx + 1]; + if (is_degenerate(last, poly[idx], next)) + { // lines are in the opposite direction + // don't add vert to the result + is_changed = true; + while (result.size() > 1 && is_degenerate(result[result.size() - 2], result.back(), next)) + { + result.pop_back(); + } + } + else + { + result.push_back(poly[idx]); + } + } + + for (size_t i = end_vertex; i < poly.size(); ++i) + { + result.push_back(poly[i]); // Add everything after the end vertex. + } + + if (is_changed) + { + if (for_polyline || result.size() > 2) + { + poly.setPoints(std::move(result)); + } + else + { + removeAt(poly_idx); + poly_idx--; // effectively the next iteration has the same poly_idx (referring to a new poly which is not yet processed) + } + } + } +} + +template +void LinesSet::addPaths(ClipperLib::Clipper& clipper, ClipperLib::PolyType poly_typ) const +{ + for (const LineType& line : getLines()) + { + // In this context, the "Closed" argument means "Is a surface" so it should be only + // true for actual filled polygons. Closed polylines are to be treated as lines here. + if constexpr (std::is_same::value) + { + clipper.AddPath(line.getPoints(), poly_typ, true); + } + else + { + clipper.AddPath(line.getPoints(), poly_typ, false); + } + } +} + +template +void LinesSet::addPaths(ClipperLib::ClipperOffset& clipper, ClipperLib::JoinType joint_type, ClipperLib::EndType endType) const +{ + for (const LineType& line : getLines()) + { + clipper.AddPath(line.getPoints(), joint_type, endType); + } +} + +template size_t OpenLinesSet::pointCount() const; +template void OpenLinesSet::removeAt(size_t index); +template void OpenLinesSet::splitIntoSegments(OpenLinesSet& result) const; +template OpenLinesSet OpenLinesSet::splitIntoSegments() const; +template coord_t OpenLinesSet::length() const; +template Shape OpenLinesSet::createTubeShape(const coord_t inner_offset, const coord_t outer_offset) const; +template void OpenLinesSet::translate(const Point2LL& delta); +template void OpenLinesSet::removeDegenerateVerts(); +template void OpenLinesSet::addPaths(ClipperLib::Clipper& clipper, ClipperLib::PolyType PolyTyp) const; +template void OpenLinesSet::addPaths(ClipperLib::ClipperOffset& clipper, ClipperLib::JoinType jointType, ClipperLib::EndType endType) const; +template void OpenLinesSet::push_back(const OpenPolyline& line, CheckNonEmptyParam checkNonEmpty); +template void OpenLinesSet::push_back(OpenPolyline&& line, CheckNonEmptyParam checkNonEmpty); +template void OpenLinesSet::push_back(OpenLinesSet&& lines_set); + +template size_t ClosedLinesSet::pointCount() const; +template void ClosedLinesSet::removeAt(size_t index); +template void ClosedLinesSet::splitIntoSegments(OpenLinesSet& result) const; +template OpenLinesSet ClosedLinesSet::splitIntoSegments() const; +template coord_t ClosedLinesSet::length() const; +template Shape ClosedLinesSet::createTubeShape(const coord_t inner_offset, const coord_t outer_offset) const; +template void ClosedLinesSet::translate(const Point2LL& delta); +template void ClosedLinesSet::removeDegenerateVerts(); +template void ClosedLinesSet::addPaths(ClipperLib::Clipper& clipper, ClipperLib::PolyType PolyTyp) const; +template void ClosedLinesSet::addPaths(ClipperLib::ClipperOffset& clipper, ClipperLib::JoinType jointType, ClipperLib::EndType endType) const; +template void ClosedLinesSet::push_back(const ClosedPolyline& line, CheckNonEmptyParam checkNonEmpty); +template void ClosedLinesSet::push_back(ClosedPolyline&& line, CheckNonEmptyParam checkNonEmpty); +template void ClosedLinesSet::push_back(ClosedLinesSet&& lines_set); +template void ClosedLinesSet::push_back(LinesSet&& lines_set); + +template size_t LinesSet::pointCount() const; +template void LinesSet::removeAt(size_t index); +template void LinesSet::splitIntoSegments(OpenLinesSet& result) const; +template OpenLinesSet LinesSet::splitIntoSegments() const; +template coord_t LinesSet::length() const; +template Shape LinesSet::createTubeShape(const coord_t inner_offset, const coord_t outer_offset) const; +template void LinesSet::translate(const Point2LL& delta); +template void LinesSet::removeDegenerateVerts(); +template void LinesSet::addPaths(ClipperLib::Clipper& clipper, ClipperLib::PolyType PolyTyp) const; +template void LinesSet::addPaths(ClipperLib::ClipperOffset& clipper, ClipperLib::JoinType jointType, ClipperLib::EndType endType) const; +template void LinesSet::push_back(const Polygon& line, CheckNonEmptyParam checkNonEmpty); +template void LinesSet::push_back(Polygon&& line, CheckNonEmptyParam checkNonEmpty); +template void LinesSet::push_back(LinesSet&& lines_set); + +} // namespace cura diff --git a/src/geometry/MixedLinesSet.cpp b/src/geometry/MixedLinesSet.cpp new file mode 100644 index 0000000000..c080becf41 --- /dev/null +++ b/src/geometry/MixedLinesSet.cpp @@ -0,0 +1,159 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/MixedLinesSet.h" + +#include + +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" + + +namespace cura +{ + +Shape MixedLinesSet::offset(coord_t distance, ClipperLib::JoinType join_type, double miter_limit) const +{ + if (distance == 0) + { + // Return a shape that contains only actual polygons + Shape result; + + for (const PolylinePtr& line : (*this)) + { + if (const std::shared_ptr polygon = dynamic_pointer_cast(line)) + { + result.push_back(*polygon); + } + } + + return result; + } + Shape polygons; + ClipperLib::ClipperOffset clipper(miter_limit, 10.0); + + for (const PolylinePtr& line : (*this)) + { + if (const std::shared_ptr polygon = dynamic_pointer_cast(line)) + { + // Union all polygons first and add them later + polygons.push_back(*polygon); + } + else + { + static auto end_type_fn = [&line, &join_type]() + { + if (line->hasClosingSegment()) + { + return ClipperLib::etClosedLine; + } + return join_type == ClipperLib::jtMiter ? ClipperLib::etOpenSquare : ClipperLib::etOpenRound; + }; + + const ClipperLib::EndType end_type{ end_type_fn() }; + clipper.AddPath(line->getPoints(), join_type, end_type); + } + } + + if (! polygons.empty()) + { + polygons = polygons.unionPolygons(); + + for (const Polygon& polygon : polygons) + { + clipper.AddPath(polygon.getPoints(), join_type, ClipperLib::etClosedPolygon); + } + } + + clipper.MiterLimit = miter_limit; + + ClipperLib::Paths result; + clipper.Execute(result, static_cast(distance)); + return Shape{ std::move(result) }; +} + +void MixedLinesSet::push_back(const OpenPolyline& line) +{ + std::vector::push_back(std::make_shared(line)); +} + +void MixedLinesSet::push_back(OpenPolyline&& line) +{ + std::vector::push_back(std::make_shared(std::move(line))); +} + +void MixedLinesSet::push_back(ClosedPolyline&& line) +{ + std::vector::push_back(std::make_shared(std::move(line))); +} + +void MixedLinesSet::push_back(const Polygon& line) +{ + std::vector::push_back(std::make_shared(line)); +} + +void MixedLinesSet::push_back(const std::shared_ptr& line) +{ + std::vector::push_back(line); +} + +void MixedLinesSet::push_back(const PolylinePtr& line) +{ + std::vector::push_back(line); +} + +void MixedLinesSet::push_back(OpenLinesSet&& lines_set) +{ + reserve(size() + lines_set.size()); + for (OpenPolyline& line : lines_set) + { + push_back(std::move(line)); + } +} + +void MixedLinesSet::push_back(const OpenLinesSet& lines_set) +{ + reserve(size() + lines_set.size()); + for (const OpenPolyline& line : lines_set) + { + push_back(line); + } +} + +void MixedLinesSet::push_back(ClosedLinesSet&& lines_set) +{ + reserve(size() + lines_set.size()); + for (ClosedPolyline& line : lines_set) + { + push_back(std::move(line)); + } +} + +void MixedLinesSet::push_back(const LinesSet& lines_set) +{ + reserve(size() + lines_set.size()); + for (const Polygon& line : lines_set) + { + push_back(line); + } +} + +void MixedLinesSet::push_back(const Shape& shape) +{ + push_back(static_cast&>(shape)); +} + +coord_t MixedLinesSet::length() const +{ + return std::accumulate( + begin(), + end(), + 0LL, + [](coord_t value, const PolylinePtr& line) + { + return value + line->length(); + }); +} + +} // namespace cura diff --git a/src/geometry/PartsView.cpp b/src/geometry/PartsView.cpp new file mode 100644 index 0000000000..5e080b2aea --- /dev/null +++ b/src/geometry/PartsView.cpp @@ -0,0 +1,60 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/PartsView.h" + +#include "geometry/Polygon.h" +#include "geometry/SingleShape.h" + +namespace cura +{ + +size_t PartsView::getPartContaining(size_t poly_idx, size_t* boundary_poly_idx) const +{ + const PartsView& partsView = *this; + for (size_t part_idx_now = 0; part_idx_now < partsView.size(); part_idx_now++) + { + const std::vector& partView = partsView[part_idx_now]; + if (partView.size() == 0) + { + continue; + } + std::vector::const_iterator result = std::find(partView.begin(), partView.end(), poly_idx); + if (result != partView.end()) + { + if (boundary_poly_idx) + { + *boundary_poly_idx = partView[0]; + } + return part_idx_now; + } + } + return NO_INDEX; +} + +SingleShape PartsView::assemblePart(size_t part_idx) const +{ + const PartsView& partsView = *this; + SingleShape ret; + if (part_idx != NO_INDEX) + { + for (size_t poly_idx_ff : partsView[part_idx]) + { + ret.push_back(polygons_[poly_idx_ff]); + } + } + return ret; +} + +SingleShape PartsView::assemblePartContaining(size_t poly_idx, size_t* boundary_poly_idx) const +{ + SingleShape ret; + size_t part_idx = getPartContaining(poly_idx, boundary_poly_idx); + if (part_idx != NO_INDEX) + { + return assemblePart(part_idx); + } + return ret; +} + +} // namespace cura diff --git a/src/geometry/PointsSet.cpp b/src/geometry/PointsSet.cpp new file mode 100644 index 0000000000..e839f7c55a --- /dev/null +++ b/src/geometry/PointsSet.cpp @@ -0,0 +1,51 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/PointsSet.h" + +#include "geometry/Point3Matrix.h" +#include "geometry/PointMatrix.h" + +namespace cura +{ + +PointsSet::PointsSet(const std::initializer_list& initializer) + : points_(initializer) +{ +} + +PointsSet::PointsSet(const ClipperLib::Path& points) + : points_(points) +{ +} + +PointsSet::PointsSet(ClipperLib::Path&& points) + : points_(std::move(points)) +{ +} + +void PointsSet::applyMatrix(const PointMatrix& matrix) +{ + for (Point2LL& point : points_) + { + point = matrix.apply(point); + } +} + +void PointsSet::applyMatrix(const Point3Matrix& matrix) +{ + for (Point2LL& point : points_) + { + point = matrix.apply(point); + } +} + +void PointsSet::translate(const Point2LL& translation) +{ + for (Point2LL& point : points_) + { + point += translation; + } +} + +} // namespace cura diff --git a/src/geometry/Polygon.cpp b/src/geometry/Polygon.cpp new file mode 100644 index 0000000000..83283b4000 --- /dev/null +++ b/src/geometry/Polygon.cpp @@ -0,0 +1,599 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/Polygon.h" + +#include +#include + +#include "geometry/Point3Matrix.h" +#include "geometry/Shape.h" +#include "utils/ListPolyIt.h" +#include "utils/linearAlg2D.h" + +namespace cura +{ + +Shape Polygon::intersection(const Polygon& other) const +{ + ClipperLib::Paths ret_paths; + ClipperLib::Clipper clipper(clipper_init); + clipper.AddPath(getPoints(), ClipperLib::ptSubject, true); + clipper.AddPath(other.getPoints(), ClipperLib::ptClip, true); + clipper.Execute(ClipperLib::ctIntersection, ret_paths); + return Shape{ std::move(ret_paths) }; +} + +void Polygon::smooth2(int remove_length, Polygon& result) const +{ + if (! empty()) + { + result.push_back(front()); + } + + for (unsigned int poly_idx = 1; poly_idx < size(); poly_idx++) + { + const Point2LL& last = getPoints()[poly_idx - 1]; + const Point2LL& now = getPoints()[poly_idx]; + const Point2LL& next = getPoints()[(poly_idx + 1) % size()]; + if (shorterThen(last - now, remove_length) && shorterThen(now - next, remove_length)) + { + poly_idx++; // skip the next line piece (dont escalate the removal of edges) + if (poly_idx < size()) + { + result.push_back(getPoints()[poly_idx]); + } + } + else + { + result.push_back(getPoints()[poly_idx]); + } + } +} + +void Polygon::smooth(int remove_length, Polygon& result) const +{ + // a typical zigzag with the middle part to be removed by removing (1) : + // + // 3 + // ^ + // | + // | + // inside | outside + // 1--->2 + // ^ + // | + // | + // | + // 0 + if (! empty()) + { + result.push_back(front()); + } + auto is_zigzag = [remove_length](const int64_t v02_size, const int64_t v12_size, const int64_t v13_size, const int64_t dot1, const int64_t dot2) + { + if (v12_size > remove_length) + { // v12 or v13 is too long + return false; + } + const bool p1_is_left_of_v02 = dot1 < 0; + if (! p1_is_left_of_v02) + { // removing p1 wouldn't smooth outward + return false; + } + const bool p2_is_left_of_v13 = dot2 > 0; + if (p2_is_left_of_v13) + { // l0123 doesn't constitute a zigzag ''|,, + return false; + } + if (-dot1 <= v02_size * v12_size / 2) + { // angle at p1 isn't sharp enough + return false; + } + if (-dot2 <= v13_size * v12_size / 2) + { // angle at p2 isn't sharp enough + return false; + } + return true; + }; + Point2LL v02 = getPoints()[2] - getPoints()[0]; + Point2LL v02_t = turn90CCW(v02); + int64_t v02_size = vSize(v02); + bool force_push = false; + for (unsigned int poly_idx = 1; poly_idx < size(); poly_idx++) + { + const Point2LL& p1 = getPoints().at(poly_idx); + const Point2LL& p2 = getPoints().at((poly_idx + 1) % size()); + const Point2LL& p3 = getPoints().at((poly_idx + 2) % size()); + // v02 computed in last iteration + // v02_size as well + const Point2LL v12 = p2 - p1; + const int64_t v12_size = vSize(v12); + const Point2LL v13 = p3 - p1; + const int64_t v13_size = vSize(v13); + + // v02T computed in last iteration + const int64_t dot1 = dot(v02_t, v12); + const Point2LL v13_t = turn90CCW(v13); + const int64_t dot2 = dot(v13_t, v12); + bool push_point = force_push || ! is_zigzag(v02_size, v12_size, v13_size, dot1, dot2); + force_push = false; + if (push_point) + { + result.push_back(p1); + } + else + { + // do not add the current one to the result + force_push = true; // ensure the next point is added; it cannot also be a zigzag + } + v02_t = v13_t; + v02 = v13; + v02_size = v13_size; + } +} + +void Polygon::smoothOutward(const AngleDegrees min_angle, int shortcut_length, Polygon& result) const +{ + // example of smoothed out corner: + // + // 6 + // ^ + // | + // inside | outside + // 2>3>4>5 + // ^ / . + // | / . + // 1 / . + // ^ / . + // |/ . + // | + // | + // 0 + + int shortcut_length2 = shortcut_length * shortcut_length; + double cos_min_angle = std::cos(min_angle / 180 * std::numbers::pi); + + ListPolygon poly; + ListPolyIt::convertPolygonToList(*this, poly); + + { // remove duplicate vertices + ListPolyIt p1_it(poly, poly.begin()); + do + { + ListPolyIt next = p1_it.next(); + if (vSize2(p1_it.p() - next.p()) < 100LL) + { + p1_it.remove(); + } + p1_it = next; + } while (p1_it != ListPolyIt(poly, poly.begin())); + } + + ListPolyIt p1_it(poly, poly.begin()); + do + { + const Point2LL p1 = p1_it.p(); + ListPolyIt p0_it = p1_it.prev(); + ListPolyIt p2_it = p1_it.next(); + const Point2LL p0 = p0_it.p(); + const Point2LL p2 = p2_it.p(); + + const Point2LL v10 = p0 - p1; + const Point2LL v12 = p2 - p1; + double cos_angle = INT2MM(INT2MM(dot(v10, v12))) / vSizeMM(v10) / vSizeMM(v12); + bool is_left_angle = LinearAlg2D::pointIsLeftOfLine(p1, p0, p2) > 0; + if (cos_angle > cos_min_angle && is_left_angle) + { + // angle is so sharp that it can be removed + Point2LL v02 = p2_it.p() - p0_it.p(); + if (vSize2(v02) >= shortcut_length2) + { + smoothCornerSimple(p0, p1, p2, p0_it, p1_it, p2_it, v10, v12, v02, shortcut_length, cos_angle); + } + else + { + bool remove_poly = smoothCornerComplex(p1, p0_it, p2_it, shortcut_length); // edits p0_it and p2_it! + if (remove_poly) + { + // don't convert ListPolygon into result + return; + } + } + // update: + p1_it = p2_it; // next point to consider for whether it's an internal corner + } + else + { + ++p1_it; + } + } while (p1_it != ListPolyIt(poly, poly.begin())); + + ListPolyIt::convertListPolygonToPolygon(poly, result); +} + +void Polygon::smoothCornerSimple( + const Point2LL& p0, + const Point2LL& p1, + const Point2LL& p2, + const ListPolyIt& p0_it, + const ListPolyIt& p1_it, + const ListPolyIt& p2_it, + const Point2LL& v10, + const Point2LL& v12, + const Point2LL& v02, + const int64_t shortcut_length, + double cos_angle) +{ + // 1----b---->2 + // ^ / + // | / + // | / + // |/ + // |a + // | + // 0 + // ideally a1_size == b1_size + if (vSize2(v02) <= shortcut_length * (shortcut_length + 10) // v02 is approximately shortcut length + || (cos_angle > 0.9999 && LinearAlg2D::getDist2FromLine(p2, p0, p1) < 20 * 20)) // p1 is degenerate + { + // handle this separately to avoid rounding problems below in the getPointOnLineWithDist function + p1_it.remove(); + // don't insert new elements + } + else + { + // compute the distance a1 == b1 to get vSize(ab)==shortcut_length with the given angle between v10 and v12 + // 1 + // /|\ . + // / | \ . + // / | \ . + // / | \ . + // a/____|____\b . + // m + // use trigonometry on the right-angled triangle am1 + double a1m_angle = acos(cos_angle) / 2; + const int64_t a1_size = shortcut_length / 2 / std::sin(a1m_angle); + if (a1_size * a1_size < vSize2(v10) && a1_size * a1_size < vSize2(v12)) + { + Point2LL a = p1 + normal(v10, a1_size); + Point2LL b = p1 + normal(v12, a1_size); +#ifdef ASSERT_INSANE_OUTPUT + assert(vSize(a) < 4000000); + assert(vSize(b) < 4000000); +#endif // #ifdef ASSERT_INSANE_OUTPUT + ListPolyIt::insertPointNonDuplicate(p0_it, p1_it, a); + ListPolyIt::insertPointNonDuplicate(p1_it, p2_it, b); + p1_it.remove(); + } + else if (vSize2(v12) < vSize2(v10)) + { + // b + // 1->2 + // ^ | + // | / + // | | + // |/ + // |a + // | + // 0 + const Point2LL& b = p2_it.p(); + Point2LL a; + bool success = LinearAlg2D::getPointOnLineWithDist(b, p1, p0, shortcut_length, a); + // v02 has to be longer than ab! + if (success) + { // if not success then assume a is negligibly close to 0, but rounding errors caused a problem +#ifdef ASSERT_INSANE_OUTPUT + assert(vSize(a) < 4000000); +#endif // #ifdef ASSERT_INSANE_OUTPUT + ListPolyIt::insertPointNonDuplicate(p0_it, p1_it, a); + } + p1_it.remove(); + } + else + { + // 1---------b----------->2 + // ^ ,-' + // | ,-' + // 0.-' + // a + const Point2LL& a = p0_it.p(); + Point2LL b; + bool success = LinearAlg2D::getPointOnLineWithDist(a, p1, p2, shortcut_length, b); + // v02 has to be longer than ab! + if (success) + { // if not success then assume b is negligibly close to 2, but rounding errors caused a problem +#ifdef ASSERT_INSANE_OUTPUT + assert(vSize(b) < 4000000); +#endif // #ifdef ASSERT_INSANE_OUTPUT + ListPolyIt::insertPointNonDuplicate(p1_it, p2_it, b); + } + p1_it.remove(); + } + } +} + +void Polygon::smoothOutwardStep( + const Point2LL& p1, + const int64_t shortcut_length2, + ListPolyIt& p0_it, + ListPolyIt& p2_it, + bool& forward_is_blocked, + bool& backward_is_blocked, + bool& forward_is_too_far, + bool& backward_is_too_far) +{ + const bool forward_has_converged = forward_is_blocked || forward_is_too_far; + const bool backward_has_converged = backward_is_blocked || backward_is_too_far; + const Point2LL p0 = p0_it.p(); + const Point2LL p2 = p2_it.p(); + bool walk_forward + = ! forward_has_converged && (backward_has_converged || (vSize2(p2 - p1) < vSize2(p0 - p1))); // whether to walk along the p1-p2 direction or in the p1-p0 direction + + if (walk_forward) + { + const ListPolyIt p2_2_it = p2_it.next(); + const Point2LL p2_2 = p2_2_it.p(); + bool p2_is_left = LinearAlg2D::pointIsLeftOfLine(p2, p0, p2_2) >= 0; + if (! p2_is_left) + { + forward_is_blocked = true; + return; + } + + const Point2LL v02_2 = p2_2 - p0_it.p(); + if (vSize2(v02_2) > shortcut_length2) + { + forward_is_too_far = true; + return; + } + + p2_it = p2_2_it; // make one step in the forward direction + backward_is_blocked = false; // invalidate data about backward walking + backward_is_too_far = false; + return; + } + const ListPolyIt p0_2_it = p0_it.prev(); + const Point2LL p0_2 = p0_2_it.p(); + bool p0_is_left = LinearAlg2D::pointIsLeftOfLine(p0, p0_2, p2) >= 0; + if (! p0_is_left) + { + backward_is_blocked = true; + return; + } + + const Point2LL v02_2 = p2_it.p() - p0_2; + if (vSize2(v02_2) > shortcut_length2) + { + backward_is_too_far = true; + return; + } + + p0_it = p0_2_it; // make one step in the backward direction + forward_is_blocked = false; // invalidate data about forward walking + forward_is_too_far = false; +} + +bool Polygon::smoothCornerComplex(const Point2LL& p1, ListPolyIt& p0_it, ListPolyIt& p2_it, const int64_t shortcut_length) +{ + // walk away from the corner until the shortcut > shortcut_length or it would smooth a piece inward + // - walk in both directions untill shortcut > shortcut_length + // - stop walking in one direction if it would otherwise cut off a corner in that direction + // - same in the other direction + // - stop if both are cut off + // walk by updating p0_it and p2_it + int64_t shortcut_length2 = shortcut_length * shortcut_length; + bool forward_is_blocked = false; + bool forward_is_too_far = false; + bool backward_is_blocked = false; + bool backward_is_too_far = false; + while (true) + { + const bool forward_has_converged = forward_is_blocked || forward_is_too_far; + const bool backward_has_converged = backward_is_blocked || backward_is_too_far; + if (forward_has_converged && backward_has_converged) + { + if (forward_is_too_far && backward_is_too_far && vSize2(p0_it.prev().p() - p2_it.next().p()) < shortcut_length2) + { + // o + // / \ . + // o o + // | | + // \ / . + // | | + // \ / . + // | | + // o o + --p0_it; + ++p2_it; + forward_is_too_far = false; // invalidate data + backward_is_too_far = false; // invalidate data + continue; + } + break; + } + smoothOutwardStep(p1, shortcut_length2, p0_it, p2_it, forward_is_blocked, backward_is_blocked, forward_is_too_far, backward_is_too_far); + if (p0_it.prev() == p2_it || p0_it == p2_it) + { // stop if we went all the way around the polygon + // this should only be the case for hole polygons (?) + if (forward_is_too_far && backward_is_too_far) + { + // in case p0_it.prev() == p2_it : + // / . + // / /| + // | becomes | | + // \ \| + // \ . + // in case p0_it == p2_it : + // / . + // / becomes /| + // \ \| + // \ . + break; + } + // this whole polygon can be removed + return true; + } + } + + const Point2LL v02 = p2_it.p() - p0_it.p(); + const int64_t v02_size2 = vSize2(v02); + // set the following: + // p0_it = start point of line + // p2_it = end point of line + if (std::abs(v02_size2 - shortcut_length2) < shortcut_length * 10) // i.e. if (size2 < l * (l+10) && size2 > l * (l-10)) + { // v02 is approximately shortcut length + // handle this separately to avoid rounding problems below in the getPointOnLineWithDist function + // p0_it and p2_it are already correct + } + else if (! backward_is_blocked && ! forward_is_blocked) + { // introduce two new points + // 1----b---->2 + // ^ / + // | / + // | / + // |/ + // |a + // | + // 0 + const auto v02_size = static_cast(std::sqrt(v02_size2)); + + const ListPolyIt p0_2_it = p0_it.prev(); + const ListPolyIt p2_2_it = p2_it.next(); + const Point2LL p2_2 = p2_2_it.p(); + const Point2LL p0_2 = p0_2_it.p(); + const Point2LL v02_2 = p0_2 - p2_2; + const int64_t v02_2_size = vSize(v02_2); + double progress + = std::min(1.0, INT2MM(shortcut_length - v02_size) / INT2MM(v02_2_size - v02_size)); // account for rounding error when v02_2_size is approx equal to v02_size + assert(progress >= 0.0f && progress <= 1.0f && "shortcut length must be between last length and new length"); + const Point2LL new_p0 = p0_it.p() + (p0_2 - p0_it.p()) * progress; + p0_it = ListPolyIt::insertPointNonDuplicate(p0_2_it, p0_it, new_p0); + const Point2LL new_p2 = p2_it.p() + (p2_2 - p2_it.p()) * progress; + p2_it = ListPolyIt::insertPointNonDuplicate(p2_it, p2_2_it, new_p2); + } + else if (! backward_is_blocked) + { // forward is blocked, back is open + // | + // 1->b + // ^ : + // | / + // 0 : + // |/ + // |a + // | + // 0_2 + const ListPolyIt p0_2_it = p0_it.prev(); + const Point2LL p0 = p0_it.p(); + const Point2LL p0_2 = p0_2_it.p(); + const Point2LL p2 = p2_it.p(); + Point2LL new_p0; + bool success = LinearAlg2D::getPointOnLineWithDist(p2, p0, p0_2, shortcut_length, new_p0); + // shortcut length must be possible given that last length was ok and new length is too long + if (success) + { +#ifdef ASSERT_INSANE_OUTPUT + assert(new_p0.X < 400000 && new_p0.Y < 400000); +#endif // #ifdef ASSERT_INSANE_OUTPUT + p0_it = ListPolyIt::insertPointNonDuplicate(p0_2_it, p0_it, new_p0); + } + else + { // if not then a rounding error occured + if (vSize(p2 - p0_2) < vSize2(p2 - p0)) + { + p0_it = p0_2_it; // start shortcut at 0 + } + } + } + else if (! forward_is_blocked) + { // backward is blocked, front is open + // 1----2----b----------->2_2 + // ^ ,-' + // | ,-' + //--0.-' + // a + const ListPolyIt p2_2_it = p2_it.next(); + const Point2LL p0 = p0_it.p(); + const Point2LL p2 = p2_it.p(); + const Point2LL p2_2 = p2_2_it.p(); + Point2LL new_p2; + bool success = LinearAlg2D::getPointOnLineWithDist(p0, p2, p2_2, shortcut_length, new_p2); + // shortcut length must be possible given that last length was ok and new length is too long + if (success) + { +#ifdef ASSERT_INSANE_OUTPUT + assert(new_p2.X < 400000 && new_p2.Y < 400000); +#endif // #ifdef ASSERT_INSANE_OUTPUT + p2_it = ListPolyIt::insertPointNonDuplicate(p2_it, p2_2_it, new_p2); + } + else + { // if not then a rounding error occured + if (vSize(p2_2 - p0) < vSize2(p2 - p0)) + { + p2_it = p2_2_it; // start shortcut at 0 + } + } + } + else + { + // | + // __|2 + // | / > shortcut cannot be of the desired length + // ___|/ . + // 0 + // both are blocked and p0_it and p2_it are already correct + } + // delete all cut off points + while (p0_it.next() != p2_it) + { + p0_it.next().remove(); + } + return false; +} + +Point2LL Polygon::centerOfMass() const +{ + if (! empty()) + { + Point2LL p0 = getPoints()[0]; + if (size() > 1) + { + double x{ 0 }; + double y{ 0 }; + for (size_t n = 1; n <= size(); n++) + { + Point2LL p1 = getPoints()[n % size()]; + auto second_factor = static_cast((p0.X * p1.Y) - (p1.X * p0.Y)); + + x += static_cast(p0.X + p1.X) * second_factor; + y += static_cast(p0.Y + p1.Y) * second_factor; + p0 = p1; + } + + double current_area = area(); + + x = x / 6 / current_area; + y = y / 6 / current_area; + + return { std::llrint(x), std::llrint(y) }; + } + return p0; + } + return {}; +} + +Shape Polygon::offset(int distance, ClipperLib::JoinType join_type, double miter_limit) const +{ + if (distance == 0) + { + return Shape({ *this }); + } + ClipperLib::Paths ret; + ClipperLib::ClipperOffset clipper(miter_limit, 10.0); + clipper.AddPath(getPoints(), join_type, ClipperLib::etClosedPolygon); + clipper.MiterLimit = miter_limit; + clipper.Execute(ret, distance); + return Shape{ std::move(ret) }; +} + +} // namespace cura diff --git a/src/geometry/Polyline.cpp b/src/geometry/Polyline.cpp new file mode 100644 index 0000000000..d64341eada --- /dev/null +++ b/src/geometry/Polyline.cpp @@ -0,0 +1,172 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/Polyline.h" + +#include +#include + +#include "geometry/LinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "settings/types/Angle.h" +#include "utils/linearAlg2D.h" + +namespace cura +{ + +void Polyline::removeColinearEdges(const AngleRadians max_deviation_angle) +{ + // TODO: Can be made more efficient (for example, use pointer-types for process-/skip-indices, so we can swap them without copy). + + size_t num_removed_in_iteration = 0; + do + { + num_removed_in_iteration = 0; + + std::vector process_indices(size(), true); + + bool go = true; + while (go) + { + go = false; + + const ClipperLib::Path& rpath = getPoints(); + const size_t pathlen = rpath.size(); + if (pathlen <= 3) + { + return; + } + + std::vector skip_indices(size(), false); + + std::vector new_path; + for (size_t point_idx = 0; point_idx < pathlen; ++point_idx) + { + // Don't iterate directly over process-indices, but do it this way, because there are points _in_ process-indices that should nonetheless be skipped: + if (! process_indices[point_idx]) + { + new_path.push_back(rpath[point_idx]); + continue; + } + + // Should skip the last point for this iteration if the old first was removed (which can be seen from the fact that the new first was skipped): + if (point_idx == (pathlen - 1) && skip_indices[0]) + { + skip_indices[new_path.size()] = true; + go = true; + new_path.push_back(rpath[point_idx]); + break; + } + + const Point2LL& prev = rpath[(point_idx - 1 + pathlen) % pathlen]; + const Point2LL& pt = rpath[point_idx]; + const Point2LL& next = rpath[(point_idx + 1) % pathlen]; + + double angle = LinearAlg2D::getAngleLeft(prev, pt, next); // [0 : 2 * pi] + if (angle >= std::numbers::pi) + { + angle -= std::numbers::pi; + } // map [pi : 2 * pi] to [0 : pi] + + // Check if the angle is within limits for the point to 'make sense', given the maximum deviation. + // If the angle indicates near-parallel segments ignore the point 'pt' + if (angle > max_deviation_angle && angle < std::numbers::pi - max_deviation_angle) + { + new_path.push_back(pt); + } + else if (point_idx != (pathlen - 1)) + { + // Skip the next point, since the current one was removed: + skip_indices[new_path.size()] = true; + go = true; + new_path.push_back(next); + ++point_idx; + } + } + setPoints(std::move(new_path)); + num_removed_in_iteration += pathlen - size(); + + process_indices.clear(); + process_indices.insert(process_indices.end(), skip_indices.begin(), skip_indices.end()); + } + } while (num_removed_in_iteration > 0); +} + +Polyline::const_segments_iterator Polyline::beginSegments() const +{ + return const_segments_iterator(begin(), begin(), end()); +} + +Polyline::const_segments_iterator Polyline::endSegments() const +{ + if (hasClosingSegment()) + { + return const_segments_iterator(end(), begin(), end()); + } + else + { + return const_segments_iterator(size() > 1 ? std::prev(end()) : end(), begin(), end()); + } +} + +Polyline::segments_iterator Polyline::beginSegments() +{ + return segments_iterator(begin(), begin(), end()); +} + +Polyline::segments_iterator Polyline::endSegments() +{ + if (hasClosingSegment()) + { + return segments_iterator(end(), begin(), end()); + } + else + { + return segments_iterator(size() > 1 ? std::prev(end()) : end(), begin(), end()); + } +} + +coord_t Polyline::length() const +{ + return std::accumulate( + beginSegments(), + endSegments(), + 0, + [](coord_t total, const const_segments_iterator::value_type& segment) + { + return total + vSize(segment.end - segment.start); + }); +} + +bool Polyline::shorterThan(const coord_t check_length) const +{ + coord_t length = 0; + auto iterator_segment = std::find_if( + beginSegments(), + endSegments(), + [&length, &check_length](const const_segments_iterator::value_type& segment) + { + length += vSize(segment.end - segment.start); + return length >= check_length; + }); + return iterator_segment == endSegments(); +} + +void Polyline::splitIntoSegments(OpenLinesSet& result) const +{ + result.reserve(result.size() + segmentsCount()); + for (auto it = beginSegments(); it != endSegments(); ++it) + { + result.emplace_back(OpenPolyline({ (*it).start, (*it).end })); + } +} + +OpenLinesSet Polyline::splitIntoSegments() const +{ + OpenLinesSet result; + splitIntoSegments(result); + return result; +} + +} // namespace cura diff --git a/src/geometry/Shape.cpp b/src/geometry/Shape.cpp new file mode 100644 index 0000000000..8156bb5fa2 --- /dev/null +++ b/src/geometry/Shape.cpp @@ -0,0 +1,965 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/Shape.h" + +#include +#include +#include +#include +#include + +#ifdef BUILD_TESTS +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#endif + +#include +#include + +#include "geometry/MixedLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/PartsView.h" +#include "geometry/Polygon.h" +#include "geometry/SingleShape.h" +#include "settings/types/Ratio.h" +#include "utils/OpenPolylineStitcher.h" +#include "utils/linearAlg2D.h" + +namespace cura +{ + +Shape::Shape(ClipperLib::Paths&& paths, bool explicitely_closed) +{ + emplace_back(std::move(paths), explicitely_closed); +} + +Shape::Shape(const std::vector& polygons) + : LinesSet(polygons) +{ +} + +void Shape::emplace_back(ClipperLib::Paths&& paths, bool explicitely_closed) +{ + reserve(size() + paths.size()); + for (ClipperLib::Path& path : paths) + { + emplace_back(std::move(path), explicitely_closed); + } +} + +void Shape::emplace_back(ClipperLib::Path&& path, bool explicitely_closed) +{ + static_cast*>(this)->emplace_back(std::move(path), explicitely_closed); +} + +Shape Shape::approxConvexHull(int extra_outset) const +{ + constexpr int overshoot = MM2INT(100); // 10cm (hard-coded value). + + Shape convex_hull; + // Perform the offset for each polygon one at a time. + // This is necessary because the polygons may overlap, in which case the offset could end up in an infinite loop. + // See http://www.angusj.com/delphi/clipper/documentation/Docs/Units/ClipperLib/Classes/ClipperOffset/_Body.htm + for (const Polygon& polygon : (*this)) + { + ClipperLib::Paths offset_result; + ClipperLib::ClipperOffset offsetter(1.2, 10.0); + offsetter.AddPath(polygon.getPoints(), ClipperLib::jtRound, ClipperLib::etClosedPolygon); + offsetter.Execute(offset_result, overshoot); + convex_hull.emplace_back(std::move(offset_result)); + } + + return convex_hull.unionPolygons().offset(-overshoot + extra_outset, ClipperLib::jtRound); +} + +void Shape::makeConvex() +{ + // early out if there is nothing to do + if (empty()) + { + return; + } + + // Andrew’s Monotone Chain Convex Hull Algorithm + std::vector points; + for (const Polygon& poly : getLines()) + { + points.insert(points.end(), poly.begin(), poly.end()); + } + + Polygon convexified; + auto make_sorted_poly_convex = [&convexified](std::vector& poly) + { + convexified.push_back(poly[0]); + + for (const auto window : poly | ranges::views::sliding(2)) + { + const Point2LL& current = window[0]; + const Point2LL& after = window[1]; + + if (LinearAlg2D::pointIsLeftOfLine(current, convexified.back(), after) < 0) + { + // Track backwards to make sure we haven't been in a concave pocket for multiple vertices already. + while (convexified.size() >= 2 + && (LinearAlg2D::pointIsLeftOfLine(convexified.back(), convexified[convexified.size() - 2], current) >= 0 + || LinearAlg2D::pointIsLeftOfLine(convexified.back(), convexified[convexified.size() - 2], convexified.front()) > 0)) + { + convexified.pop_back(); + } + convexified.push_back(current); + } + } + }; + + std::sort( + points.begin(), + points.end(), + [](const Point2LL& a, const Point2LL& b) + { + return a.X == b.X ? a.Y < b.Y : a.X < b.X; + }); + make_sorted_poly_convex(points); + std::reverse(points.begin(), points.end()); + make_sorted_poly_convex(points); + + setLines({ convexified }); +} + +Shape Shape::difference(const Shape& other) const +{ + if (empty()) + { + return {}; + } + if (other.empty()) + { + return *this; + } + ClipperLib::Paths ret; + ClipperLib::Clipper clipper(clipper_init); + addPaths(clipper, ClipperLib::ptSubject); + other.addPaths(clipper, ClipperLib::ptClip); + clipper.Execute(ClipperLib::ctDifference, ret); + return Shape(std::move(ret)); +} + +Shape Shape::unionPolygons(const Shape& other, ClipperLib::PolyFillType fill_type) const +{ + if (empty() && other.empty()) + { + return {}; + } + if (empty() && other.size() <= 1) + { + return other; + } + if (other.empty() && size() <= 1) + { + return *this; + } + ClipperLib::Paths ret; + ClipperLib::Clipper clipper(clipper_init); + addPaths(clipper, ClipperLib::ptSubject); + other.addPaths(clipper, ClipperLib::ptSubject); + clipper.Execute(ClipperLib::ctUnion, ret, fill_type, fill_type); + return Shape{ std::move(ret) }; +} + +Shape Shape::unionPolygons() const +{ + return unionPolygons(Shape()); +} + +Shape Shape::intersection(const Shape& other) const +{ + if (empty() || other.empty()) + { + return {}; + } + ClipperLib::Paths ret; + ClipperLib::Clipper clipper(clipper_init); + addPaths(clipper, ClipperLib::ptSubject); + other.addPaths(clipper, ClipperLib::ptClip); + clipper.Execute(ClipperLib::ctIntersection, ret); + return Shape{ std::move(ret) }; +} + +Shape Shape::offset(coord_t distance, ClipperLib::JoinType join_type, double miter_limit) const +{ + if (empty()) + { + return {}; + } + if (distance == 0) + { + return *this; + } + + ClipperLib::Paths ret; + ClipperLib::ClipperOffset clipper(miter_limit, 10.0); + unionPolygons().addPaths(clipper, join_type, ClipperLib::etClosedPolygon); + clipper.MiterLimit = miter_limit; + clipper.Execute(ret, static_cast(distance)); + return Shape{ std::move(ret) }; +} + +bool Shape::inside(const Point2LL& p, bool border_result) const +{ + int poly_count_inside = 0; + for (const Polygon& poly : *this) + { + const int is_inside_this_poly = ClipperLib::PointInPolygon(p, poly.getPoints()); + if (is_inside_this_poly == -1) + { + return border_result; + } + poly_count_inside += is_inside_this_poly; + } + return (poly_count_inside % 2) == 1; +} + +size_t Shape::findInside(const Point2LL& p, bool border_result) const +{ + if (empty()) + { + return 0; + } + + // NOTE: Keep these vectors fixed-size, they replace an (non-standard, sized at runtime) arrays. + std::vector min_x(size(), std::numeric_limits::max()); + std::vector crossings(size()); + + for (size_t poly_idx = 0; poly_idx < size(); poly_idx++) + { + const Polygon& poly = (*this)[poly_idx]; + Point2LL p0 = poly.back(); + for (const Point2LL& p1 : poly) + { + short comp = LinearAlg2D::pointLiesOnTheRightOfLine(p, p0, p1); + if (comp == 1) + { + crossings[poly_idx]++; + int64_t x; + if (p1.Y == p0.Y) + { + x = p0.X; + } + else + { + x = p0.X + (p1.X - p0.X) * (p.Y - p0.Y) / (p1.Y - p0.Y); + } + min_x[poly_idx] = std::min(x, min_x[poly_idx]); + } + else if (border_result && comp == 0) + { + return poly_idx; + } + p0 = p1; + } + } + + int64_t min_x_uneven = std::numeric_limits::max(); + size_t ret = NO_INDEX; + size_t n_unevens = 0; + for (size_t array_idx = 0; array_idx < size(); array_idx++) + { + if (crossings[array_idx] % 2 == 1) + { + n_unevens++; + if (min_x[array_idx] < min_x_uneven) + { + min_x_uneven = min_x[array_idx]; + ret = array_idx; + } + } + } + if (n_unevens % 2 == 0) + { + ret = NO_INDEX; + } + return ret; +} + +template +OpenLinesSet Shape::intersection(const LinesSet& polylines, bool restitch, const coord_t max_stitch_distance) const +{ + if (empty() || polylines.empty()) + { + return {}; + } + + OpenLinesSet split_polylines = polylines.splitIntoSegments(); + + ClipperLib::PolyTree result; + ClipperLib::Clipper clipper(clipper_init); + split_polylines.addPaths(clipper, ClipperLib::ptSubject); + addPaths(clipper, ClipperLib::ptClip); + clipper.Execute(ClipperLib::ctIntersection, result); + ClipperLib::Paths result_paths; + ClipperLib::OpenPathsFromPolyTree(result, result_paths); + + OpenLinesSet result_lines(std::move(result_paths)); + + if (restitch) + { + OpenLinesSet result_open_lines; + Shape result_closed_lines; + + const coord_t snap_distance = 10_mu; + OpenPolylineStitcher::stitch(result_lines, result_open_lines, result_closed_lines, max_stitch_distance, snap_distance); + + result_lines = std::move(result_open_lines); + // if open polylines got stitched into closed polylines, split them back up into open polylines again, because the result only admits open polylines + for (ClosedPolyline& closed_line : result_closed_lines) + { + if (! closed_line.empty()) + { + if (closed_line.size() > 2) + { + closed_line.push_back(closed_line.front()); + } + result_lines.emplace_back(std::move(closed_line.getPoints())); + } + } + } + + return result_lines; +} + +Shape Shape::xorPolygons(const Shape& other, ClipperLib::PolyFillType pft) const +{ + if (empty()) + { + return other; + } + if (other.empty()) + { + return *this; + } + ClipperLib::Paths ret; + ClipperLib::Clipper clipper(clipper_init); + addPaths(clipper, ClipperLib::ptSubject); + other.addPaths(clipper, ClipperLib::ptClip); + clipper.Execute(ClipperLib::ctXor, ret, pft); + return Shape{ std::move(ret) }; +} + +Shape Shape::execute(ClipperLib::PolyFillType pft) const +{ + ClipperLib::Paths ret; + ClipperLib::Clipper clipper(clipper_init); + addPaths(clipper, ClipperLib::ptSubject); + clipper.Execute(ClipperLib::ctXor, ret, pft); + return Shape{ std::move(ret) }; +} + +Shape Shape::offsetMulti(const std::vector& offset_dists) const +{ + // we need as many offset-dists as points + assert(pointCount() == offset_dists.size()); + + ClipperLib::Paths ret; + size_t i = 0; + for (const Polygon& poly_line : (*this) + | ranges::views::filter( + [](const Polygon& path) + { + return ! path.empty(); + })) + { + Polygon ret_poly_line; + + auto prev_p = poly_line.back(); + auto prev_dist = offset_dists[i + poly_line.size() - 1]; + + for (const Point2LL& p : poly_line) + { + auto offset_dist = offset_dists[i]; + + auto vec_dir = prev_p - p; + + constexpr coord_t min_vec_len = 10; + if (vSize2(vec_dir) > min_vec_len * min_vec_len) + { + auto offset_p1 = turn90CCW(normal(vec_dir, prev_dist)); + auto offset_p2 = turn90CCW(normal(vec_dir, offset_dist)); + + ret_poly_line.push_back(prev_p + offset_p1); + ret_poly_line.push_back(p + offset_p2); + } + + prev_p = p; + prev_dist = offset_dist; + i++; + } + + ret.push_back(std::move(ret_poly_line.getPoints())); + } + + ClipperLib::SimplifyPolygons(ret, ClipperLib::PolyFillType::pftPositive); + + return Shape(std::move(ret)); +} + +Shape Shape::getOutsidePolygons() const +{ + if (empty()) + { + return {}; + } + if (size() == 1) + { + return *this; + } + + Shape ret; + ClipperLib::Clipper clipper(clipper_init); + ClipperLib::PolyTree poly_tree; + addPaths(clipper, ClipperLib::ptSubject); + clipper.Execute(ClipperLib::ctUnion, poly_tree); + + for (size_t outer_poly_idx = 0; outer_poly_idx < static_cast(poly_tree.ChildCount()); outer_poly_idx++) + { + ClipperLib::PolyNode* child = poly_tree.Childs[outer_poly_idx]; + ret.emplace_back(std::move(child->Contour)); + } + return ret; +} + +void Shape::removeEmptyHolesProcessPolyTreeNode(const ClipperLib::PolyNode& node, const bool remove_holes, Shape& ret) const +{ + for (size_t outer_poly_idx = 0; outer_poly_idx < static_cast(node.ChildCount()); outer_poly_idx++) + { + ClipperLib::PolyNode* child = node.Childs[outer_poly_idx]; + if (remove_holes) + { + ret.emplace_back(std::move(child->Contour)); + } + for (size_t hole_node_idx = 0; hole_node_idx < static_cast(child->ChildCount()); hole_node_idx++) + { + ClipperLib::PolyNode& hole_node = *child->Childs[hole_node_idx]; + if ((hole_node.ChildCount() > 0) == remove_holes) + { + ret.emplace_back(std::move(hole_node.Contour)); + removeEmptyHolesProcessPolyTreeNode(hole_node, remove_holes, ret); + } + } + } +} + +void Shape::removeSmallAreas(const double min_area_size, const bool remove_holes) +{ + auto new_end = end(); + if (remove_holes) + { + for (auto it = begin(); it < new_end;) + { + // All polygons smaller than target are removed by replacing them with a polygon from the back of the vector + if (std::abs(INT2MM2(it->area())) < min_area_size) + { + *it = std::move(*--new_end); + continue; + } + it++; // Skipped on removal such that the polygon just swaped in is checked next + } + } + else + { + // For each polygon, computes the signed area, move small outlines at the end of the vector and keep references on small holes + std::vector small_holes; + for (auto it = begin(); it < new_end;) + { + double area = INT2MM2(it->area()); + if (std::abs(area) < min_area_size) + { + if (area >= 0) + { + --new_end; + if (it < new_end) + { + std::swap(*new_end, *it); + continue; + } + break; + } + small_holes.push_back(&(*it)); + } + it++; // Skipped on removal such that the polygon just swaped in is checked next + } + + // Removes small holes that have their first point inside one of the removed outlines + // Iterating in reverse ensures that unprocessed small holes won't be moved + const auto removed_outlines_start = new_end; + for (auto hole_it = small_holes.rbegin(); hole_it < small_holes.rend(); hole_it++) + { + for (auto outline_it = removed_outlines_start; outline_it < end(); outline_it++) + { + if (outline_it->inside((*hole_it)->front())) + { + **hole_it = std::move(*--new_end); + break; + } + } + } + } + resize(new_end - begin()); +} + +Shape Shape::removePolygon(const Shape& to_be_removed, int same_distance) const +{ + Shape result; + for (size_t poly_keep_idx = 0; poly_keep_idx < size(); poly_keep_idx++) + { + const Polygon& poly_keep = (*this)[poly_keep_idx]; + bool should_be_removed = false; + if (! poly_keep.empty()) + { + for (const Polygon& poly_rem : to_be_removed) + { + if (poly_rem.size() != poly_keep.size() || poly_rem.empty()) + { + continue; + } + + // find closest point, supposing this point aligns the two shapes in the best way + size_t closest_point_idx = 0; + coord_t smallest_dist2 = -1; + for (size_t point_rem_idx = 0; point_rem_idx < poly_rem.size(); point_rem_idx++) + { + coord_t dist2 = vSize2(poly_rem[point_rem_idx] - poly_keep[0]); + if (dist2 < smallest_dist2 || smallest_dist2 < 0) + { + smallest_dist2 = dist2; + closest_point_idx = point_rem_idx; + } + } + bool poly_rem_is_poly_keep = true; + // compare the two polygons on all points + if (smallest_dist2 > static_cast(same_distance * same_distance)) + { + continue; + } + for (size_t point_idx = 0; point_idx < poly_rem.size(); point_idx++) + { + coord_t dist2 = vSize2(poly_rem[(closest_point_idx + point_idx) % poly_rem.size()] - poly_keep[point_idx]); + if (dist2 > static_cast(same_distance * same_distance)) + { + poly_rem_is_poly_keep = false; + break; + } + } + if (poly_rem_is_poly_keep) + { + should_be_removed = true; + break; + } + } + } + if (! should_be_removed) + { + result.push_back(poly_keep); + } + } + return result; +} + +Shape Shape::processEvenOdd(ClipperLib::PolyFillType poly_fill_type) const +{ + ClipperLib::Paths ret; + ClipperLib::Clipper clipper(clipper_init); + addPaths(clipper, ClipperLib::ptSubject); + clipper.Execute(ClipperLib::ctUnion, ret, poly_fill_type); + return Shape{ std::move(ret) }; +} + +Shape Shape::smoothOutward(const AngleDegrees max_angle, int shortcut_length) const +{ + Shape ret; + for (const Polygon& poly : (*this)) + { + if (poly.size() < 3) + { + continue; + } + if (poly.size() == 3) + { + ret.push_back(poly); + continue; + } + poly.smoothOutward(max_angle, shortcut_length, ret.newLine()); + if (ret.back().size() < 3) + { + ret.resize(ret.size() - 1); + } + } + return ret; +} + +Shape Shape::smooth(int remove_length) const +{ + Shape ret; + for (const Polygon& poly : (*this)) + { + if (poly.size() < 3) + { + continue; + } + if (poly.size() == 3) + { + ret.push_back(poly); + continue; + } + poly.smooth(remove_length, ret.newLine()); + Polygon& back = ret.back(); + if (back.size() < 3) + { + back.resize(back.size() - 1); + } + } + return ret; +} + +Shape Shape::smooth2(int remove_length, int min_area) const +{ + Shape ret; + for (const Polygon& poly : (*this)) + { + if (poly.empty()) + { + continue; + } + if (poly.area() < min_area || poly.size() <= 5) // when optimally removing, a poly with 5 pieces results in a triangle. Smaller polys dont have area! + { + ret.push_back(poly); + continue; + } + if (poly.size() < 4) + { + ret.push_back(poly); + } + else + { + poly.smooth2(remove_length, ret.newLine()); + } + } + return ret; +} + +void Shape::removeColinearEdges(const AngleRadians max_deviation_angle) +{ + Shape& thiss = *this; + for (size_t p = 0; p < size(); p++) + { + thiss[p].removeColinearEdges(max_deviation_angle); + if (thiss[p].size() < 3) + { + removeAt(p); + p--; + } + } +} + +double Shape::area() const +{ + return std::accumulate( + begin(), + end(), + 0.0, + [](double total, const Polygon& poly) + { + // note: holes already have negative area + return total + poly.area(); + }); +} + +std::vector Shape::splitIntoParts(bool union_all) const +{ + std::vector ret; + ClipperLib::Clipper clipper(clipper_init); + ClipperLib::PolyTree result_poly_tree; + addPaths(clipper, ClipperLib::ptSubject); + if (union_all) + { + clipper.Execute(ClipperLib::ctUnion, result_poly_tree, ClipperLib::pftNonZero, ClipperLib::pftNonZero); + } + else + { + clipper.Execute(ClipperLib::ctUnion, result_poly_tree); + } + + splitIntoPartsProcessPolyTreeNode(&result_poly_tree, ret); + return ret; +} + +void Shape::splitIntoPartsProcessPolyTreeNode(ClipperLib::PolyNode* node, std::vector& ret) const +{ + for (size_t n = 0; n < static_cast(node->ChildCount()); n++) + { + ClipperLib::PolyNode* child = node->Childs[n]; + SingleShape part; + part.emplace_back(std::move(child->Contour)); + for (size_t i = 0; i < static_cast(child->ChildCount()); i++) + { + part.emplace_back(std::move(child->Childs[i]->Contour)); + splitIntoPartsProcessPolyTreeNode(child->Childs[i], ret); + } + ret.push_back(std::move(part)); + } +} + +std::vector Shape::sortByNesting() const +{ + std::vector ret; + ClipperLib::Clipper clipper(clipper_init); + ClipperLib::PolyTree result_poly_tree; + addPaths(clipper, ClipperLib::ptSubject); + clipper.Execute(ClipperLib::ctUnion, result_poly_tree); + + sortByNestingProcessPolyTreeNode(&result_poly_tree, 0, ret); + return ret; +} + +void Shape::sortByNestingProcessPolyTreeNode(ClipperLib::PolyNode* node, const size_t nesting_idx, std::vector& ret) const +{ + for (size_t n = 0; n < static_cast(node->ChildCount()); n++) + { + ClipperLib::PolyNode* child = node->Childs[n]; + if (nesting_idx >= ret.size()) + { + ret.resize(nesting_idx + 1); + } + ret[nesting_idx].emplace_back(std::move(child->Contour)); + sortByNestingProcessPolyTreeNode(child, nesting_idx + 1, ret); + } +} + +PartsView Shape::splitIntoPartsView(bool union_all) +{ + Shape reordered; + PartsView parts_view(*this); + ClipperLib::Clipper clipper(clipper_init); + ClipperLib::PolyTree result_poly_tree; + addPaths(clipper, ClipperLib::ptSubject); + if (union_all) + { + clipper.Execute(ClipperLib::ctUnion, result_poly_tree, ClipperLib::pftNonZero, ClipperLib::pftNonZero); + } + else + { + clipper.Execute(ClipperLib::ctUnion, result_poly_tree); + } + + splitIntoPartsViewProcessPolyTreeNode(parts_view, reordered, &result_poly_tree); + + (*this) = std::move(reordered); + return parts_view; +} + +void Shape::splitIntoPartsViewProcessPolyTreeNode(PartsView& partsView, Shape& reordered, ClipperLib::PolyNode* node) const +{ + for (size_t n = 0; n < static_cast(node->ChildCount()); n++) + { + ClipperLib::PolyNode* child = node->Childs[n]; + partsView.emplace_back(); + size_t pos = partsView.size() - 1; + partsView[pos].push_back(reordered.size()); + reordered.emplace_back(std::move(child->Contour)); + for (size_t i = 0; i < static_cast(child->ChildCount()); i++) + { + partsView[pos].push_back(reordered.size()); + reordered.emplace_back(std::move(child->Childs[i]->Contour)); + splitIntoPartsViewProcessPolyTreeNode(partsView, reordered, child->Childs[i]); + } + } +} + +Shape Shape::removeNearSelfIntersections() const +{ + using map_pt = mapbox::geometry::point; + using map_ring = mapbox::geometry::linear_ring; + using map_poly = mapbox::geometry::polygon; + using map_mpoly = mapbox::geometry::multi_polygon; + + map_mpoly mwpoly; + + mapbox::geometry::wagyu::wagyu wagyu; + + for (auto& polygon : splitIntoParts()) + { + mwpoly.emplace_back(); + map_poly& wpoly = mwpoly.back(); + for (auto& path : polygon) + { + wpoly.push_back(std::move(*reinterpret_cast>*>(&path.getPoints()))); + for (auto& point : wpoly.back()) + { + point.x /= 4; + point.y /= 4; + } + wagyu.add_ring(wpoly.back()); + } + } + + map_mpoly sln; + + wagyu.execute(mapbox::geometry::wagyu::clip_type_union, sln, mapbox::geometry::wagyu::fill_type_even_odd, mapbox::geometry::wagyu::fill_type_even_odd); + + Shape polys; + + for (auto& poly : sln) + { + for (auto& ring : poly) + { + ring.pop_back(); + for (auto& point : ring) + { + point.x *= 4; + point.y *= 4; + } + polys.emplace_back(std::move(*reinterpret_cast(&ring))); + } + } + polys = polys.unionPolygons(); + polys.removeColinearEdges(); + + return polys; +} + +void Shape::simplify(ClipperLib::PolyFillType fill_type) +{ + if (empty()) + { + return; + } + + // This is the actual content from clipper.cpp::SimplifyPolygons, but rewritten here in order + // to avoid having to put all the polygons in a transitory list + ClipperLib::Clipper clipper; + ClipperLib::Paths ret; + clipper.StrictlySimple(true); + addPaths(clipper, ClipperLib::ptSubject); + clipper.Execute(ClipperLib::ctUnion, ret, fill_type, fill_type); + + resize(ret.size()); + + for (size_t i = 0; i < ret.size(); i++) + { + Polygon& polygon = getLines()[i]; + polygon.setExplicitelyClosed(clipper_explicitely_closed_); // Required for polygon newly created by resize() + polygon.setPoints(std::move(ret[i])); + } +} + +void Shape::ensureManifold() +{ + std::vector duplicate_locations; + std::unordered_set poly_locations; + for (const Polygon& poly : (*this)) + { + for (const Point2LL& p : poly) + { + if (poly_locations.find(p) != poly_locations.end()) + { + duplicate_locations.push_back(p); + } + poly_locations.emplace(p); + } + } + Shape removal_dots; + for (const Point2LL& p : duplicate_locations) + { + Polygon& dot = removal_dots.newLine(); + dot.push_back(p + Point2LL(0, 5)); + dot.push_back(p + Point2LL(5, 0)); + dot.push_back(p + Point2LL(0, -5)); + dot.push_back(p + Point2LL(-5, 0)); + } + if (! removal_dots.empty()) + { + *this = difference(removal_dots); + } +} + +void Shape::applyMatrix(const PointMatrix& matrix) +{ + for (Polygon& polygon : *this) + { + polygon.applyMatrix(matrix); + } +} + +void Shape::applyMatrix(const Point3Matrix& matrix) +{ + for (Polygon& polygon : *this) + { + polygon.applyMatrix(matrix); + } +} + +#ifdef BUILD_TESTS +[[maybe_unused]] Shape Shape::fromWkt(const std::string& wkt) +{ + typedef boost::geometry::model::d2::point_xy point_type; + typedef boost::geometry::model::polygon polygon_type; + + polygon_type poly; + boost::geometry::read_wkt(wkt, poly); + + Shape ret; + + Polygon outer; + for (const auto& point : poly.outer()) + { + outer.emplace_back(point.x(), point.y()); + } + ret.push_back(outer); + + for (const auto& hole : poly.inners()) + { + Polygon inner; + for (const auto& point : hole) + { + inner.emplace_back(point.x(), point.y()); + } + ret.push_back(inner); + } + + return ret; +} + +[[maybe_unused]] void Shape::writeWkt(std::ostream& stream) const +{ + stream << "POLYGON ("; + const auto paths_str = getLines() + | ranges::views::transform( + [](const Polygon& path) + { + const auto line_string = ranges::views::concat(path, path | ranges::views::take(1)) + | ranges::views::transform( + [](const Point2LL& point) + { + return fmt::format("{} {}", point.X, point.Y); + }) + | ranges::views::join(ranges::views::c_str(", ")) | ranges::to(); + return "(" + line_string + ")"; + }) + | ranges::views::join(ranges::views::c_str(", ")) | ranges::to(); + stream << paths_str; + stream << ")"; +} +#endif + +template OpenLinesSet Shape::intersection(const OpenLinesSet& polylines, bool restitch, const coord_t max_stitch_distance) const; +template OpenLinesSet Shape::intersection(const ClosedLinesSet& polylines, bool restitch, const coord_t max_stitch_distance) const; +template OpenLinesSet Shape::intersection(const LinesSet& polylines, bool restitch, const coord_t max_stitch_distance) const; + +} // namespace cura diff --git a/src/geometry/SingleShape.cpp b/src/geometry/SingleShape.cpp new file mode 100644 index 0000000000..19b0702dc3 --- /dev/null +++ b/src/geometry/SingleShape.cpp @@ -0,0 +1,43 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "geometry/SingleShape.h" + +#include "geometry/Polygon.h" + +namespace cura +{ + +bool SingleShape::inside(const Point2LL& p, bool border_result) const +{ + if (size() < 1) + { + return false; + } + + if (! (*this)[0].inside(p, border_result)) + { + return false; + } + + for (unsigned int n = 1; n < size(); n++) + { + if ((*this)[n].inside(p, border_result)) + { + return false; + } + } + return true; +} + +Polygon& SingleShape::outerPolygon() +{ + return front(); +} + +const Polygon& SingleShape::outerPolygon() const +{ + return front(); +} + +} // namespace cura diff --git a/src/infill.cpp b/src/infill.cpp index c085307622..a2d931d262 100644 --- a/src/infill.cpp +++ b/src/infill.cpp @@ -5,12 +5,15 @@ #include //For std::sort. #include +#include #include #include #include #include "WallToolPaths.h" +#include "geometry/OpenPolyline.h" +#include "geometry/PointMatrix.h" #include "infill/GyroidInfill.h" #include "infill/ImageBasedDensityProvider.h" #include "infill/LightningGenerator.h" @@ -21,8 +24,8 @@ #include "infill/UniformDensityProvider.h" #include "plugins/slots.h" #include "sliceDataStorage.h" +#include "utils/OpenPolylineStitcher.h" #include "utils/PolygonConnector.h" -#include "utils/PolylineStitcher.h" #include "utils/Simplify.h" #include "utils/UnionFind.h" #include "utils/linearAlg2D.h" @@ -52,9 +55,9 @@ static inline int computeScanSegmentIdx(int x, int line_width) namespace cura { -Polygons Infill::generateWallToolPaths( +Shape Infill::generateWallToolPaths( std::vector& toolpaths, - Polygons& outer_contour, + Shape& outer_contour, const size_t wall_line_count, const coord_t line_width, const coord_t infill_overlap, @@ -65,7 +68,7 @@ Polygons Infill::generateWallToolPaths( outer_contour = outer_contour.offset(infill_overlap); scripta::log("infill_outer_contour", outer_contour, section_type, layer_idx, scripta::CellVDI{ "infill_overlap", infill_overlap }); - Polygons inner_contour; + Shape inner_contour; if (wall_line_count > 0) { constexpr coord_t wall_0_inset = 0; // Don't apply any outer wall inset for these. That's just for the outer wall. @@ -82,15 +85,15 @@ Polygons Infill::generateWallToolPaths( void Infill::generate( std::vector& toolpaths, - Polygons& result_polygons, - Polygons& result_lines, + Shape& result_polygons, + OpenLinesSet& result_lines, const Settings& settings, int layer_idx, SectionType section_type, const std::shared_ptr& cross_fill_provider, const std::shared_ptr& lightning_trees, const SliceMeshStorage* mesh, - const Polygons& prevent_small_exposed_to_air) + const Shape& prevent_small_exposed_to_air) { if (outer_contour_.empty()) { @@ -109,7 +112,7 @@ void Infill::generate( const auto too_small_length = INT2MM(static_cast(infill_line_width_) / 2.0); // Split the infill region in a narrow region and the normal region. - Polygons small_infill = inner_contour_; + Shape small_infill = inner_contour_; inner_contour_ = inner_contour_.offset(-small_area_width_ / 2); inner_contour_.removeSmallAreas(too_small_length * too_small_length, true); inner_contour_ = inner_contour_.offset(small_area_width_ / 2); @@ -125,15 +128,15 @@ void Infill::generate( if (small_infill_part.offset(-infill_line_width_ / 2).offset(infill_line_width_ / 2).area() < infill_line_width_ * infill_line_width_ * 10 && ! inner_contour_.intersection(small_infill_part.offset(infill_line_width_ / 4)).empty()) { - inner_contour_.add(small_infill_part); + inner_contour_.push_back(small_infill_part); } else { // the part must still be printed, so re-add it - small_infill.add(small_infill_part); + small_infill.push_back(small_infill_part); } } - inner_contour_.unionPolygons(); + inner_contour_ = inner_contour_.unionPolygons(); // Fill narrow area with walls. const size_t narrow_wall_count = small_area_width_ / infill_line_width_ + 1; @@ -178,27 +181,27 @@ void Infill::generate( { zig_zaggify_ = false; } - Polygons generated_result_polygons; - Polygons generated_result_lines; + Shape generated_result_polygons; + OpenLinesSet generated_result_lines; _generate(toolpaths, generated_result_polygons, generated_result_lines, settings, cross_fill_provider, lightning_trees, mesh); zig_zaggify_ = zig_zaggify_real; multiplyInfill(generated_result_polygons, generated_result_lines); - result_polygons.add(generated_result_polygons); - result_lines.add(generated_result_lines); + result_polygons.push_back(generated_result_polygons); + result_lines.push_back(generated_result_lines); } else { //_generate may clear() the generated_result_lines, but this is an output variable that may contain data before we start. - // So make sure we provide it with a Polygons that is safe to clear and only add stuff to result_lines. - Polygons generated_result_polygons; - Polygons generated_result_lines; + // So make sure we provide it with a Shape that is safe to clear and only add stuff to result_lines. + Shape generated_result_polygons; + OpenLinesSet generated_result_lines; _generate(toolpaths, generated_result_polygons, generated_result_lines, settings, cross_fill_provider, lightning_trees, mesh); - result_polygons.add(generated_result_polygons); - result_lines.add(generated_result_lines); + result_polygons.push_back(generated_result_polygons); + result_lines.push_back(generated_result_lines); } scripta::log("infill_result_polygons_0", result_polygons, section_type, layer_idx); scripta::log("infill_result_lines_0", result_lines, section_type, layer_idx); @@ -219,7 +222,7 @@ void Infill::generate( auto it = std::remove_if( result_polygons.begin(), result_polygons.end(), - [snap_distance](PolygonRef poly) + [snap_distance](const Polygon& poly) { return poly.shorterThan(snap_distance); }); @@ -228,7 +231,7 @@ void Infill::generate( PolygonConnector connector(infill_line_width_); connector.add(result_polygons); connector.add(toolpaths); - Polygons connected_polygons; + Shape connected_polygons; std::vector connected_paths; connector.connect(connected_polygons, connected_paths); result_polygons = connected_polygons; @@ -250,8 +253,8 @@ void Infill::generate( void Infill::_generate( std::vector& toolpaths, - Polygons& result_polygons, - Polygons& result_lines, + Shape& result_polygons, + OpenLinesSet& result_lines, const Settings& settings, const std::shared_ptr& cross_fill_provider, const std::shared_ptr& lightning_trees, @@ -323,8 +326,8 @@ void Infill::_generate( mesh ? mesh->settings.get("infill_pattern") : settings.get("infill_pattern"), mesh ? mesh->settings : settings); toolpaths.insert(toolpaths.end(), toolpaths_.begin(), toolpaths_.end()); - result_polygons.add(generated_result_polygons_); - result_lines.add(generated_result_lines_); + result_polygons.push_back(generated_result_polygons_); + result_lines.push_back(generated_result_lines_); #endif break; } @@ -348,14 +351,14 @@ void Infill::_generate( && (zig_zaggify_ || pattern_ == EFillMethod::CROSS || pattern_ == EFillMethod::CROSS_3D || pattern_ == EFillMethod::CUBICSUBDIV || pattern_ == EFillMethod::GYROID || pattern_ == EFillMethod::ZIG_ZAG)) { // don't stich for non-zig-zagged line infill types - Polygons stitched_lines; - PolylineStitcher::stitch(result_lines, stitched_lines, result_polygons, infill_line_width_); - result_lines = stitched_lines; + OpenLinesSet stitched_lines; + OpenPolylineStitcher::stitch(result_lines, stitched_lines, result_polygons, infill_line_width_); + result_lines = std::move(stitched_lines); } result_lines = simplifier.polyline(result_lines); } -void Infill::multiplyInfill(Polygons& result_polygons, Polygons& result_lines) +void Infill::multiplyInfill(Shape& result_polygons, OpenLinesSet& result_lines) { if (pattern_ == EFillMethod::CONCENTRIC) { @@ -366,13 +369,13 @@ void Infill::multiplyInfill(Polygons& result_polygons, Polygons& result_lines) coord_t offset = (odd_multiplier) ? infill_line_width_ : infill_line_width_ / 2; // Get the first offset these are mirrored from the original center line - Polygons result; - Polygons first_offset; + Shape result; + Shape first_offset; { - const Polygons first_offset_lines = result_lines.offsetPolyLine(offset); // make lines on both sides of the input lines - const Polygons first_offset_polygons_inward = result_polygons.offset(-offset); // make lines on the inside of the input polygons - const Polygons first_offset_polygons_outward = result_polygons.offset(offset); // make lines on the other side of the input polygons - const Polygons first_offset_polygons = first_offset_polygons_outward.difference(first_offset_polygons_inward); + const Shape first_offset_lines = result_lines.offset(offset); // make lines on both sides of the input lines + const Shape first_offset_polygons_inward = result_polygons.offset(-offset); // make lines on the inside of the input polygons + const Shape first_offset_polygons_outward = result_polygons.offset(offset); // make lines on the other side of the input polygons + const Shape first_offset_polygons = first_offset_polygons_outward.difference(first_offset_polygons_inward); first_offset = first_offset_lines.unionPolygons( first_offset_polygons); // usually we only have either lines or polygons, but this code also handles an infill pattern which generates both if (zig_zaggify_) @@ -380,20 +383,20 @@ void Infill::multiplyInfill(Polygons& result_polygons, Polygons& result_lines) first_offset = inner_contour_.difference(first_offset); } } - result.add(first_offset); + result.push_back(first_offset); // Create the additional offsets from the first offsets, generated earlier, the direction of these offsets is // depended on whether these lines should be connected or not. if (infill_multiplier_ > 3) { - Polygons reference_polygons = first_offset; - const size_t multiplier = static_cast(infill_multiplier_ / 2); + Shape reference_polygons = first_offset; + const size_t multiplier = infill_multiplier_ / 2; const int extra_offset = mirror_offset_ ? -infill_line_width_ : infill_line_width_; for (size_t infill_line = 1; infill_line < multiplier; ++infill_line) { - Polygons extra_polys = reference_polygons.offset(extra_offset); - result.add(extra_polys); + Shape extra_polys = reference_polygons.offset(extra_offset); + result.push_back(extra_polys); reference_polygons = std::move(extra_polys); } } @@ -408,45 +411,37 @@ void Infill::multiplyInfill(Polygons& result_polygons, Polygons& result_lines) result_polygons.clear(); result_lines.clear(); } - result_polygons.add(result); + result_polygons.push_back(result); if (! zig_zaggify_) { - for (PolygonRef poly : result_polygons) - { // make polygons into polylines - if (poly.empty()) - { - continue; - } - poly.add(poly[0]); - } - Polygons polylines = inner_contour_.intersectionPolyLines(result_polygons); + OpenLinesSet polylines = inner_contour_.intersection(static_cast>(result_polygons)); result_polygons.clear(); - PolylineStitcher::stitch(polylines, result_lines, result_polygons, infill_line_width_); + OpenPolylineStitcher::stitch(polylines, result_lines, result_polygons, infill_line_width_); } } -void Infill::generateGyroidInfill(Polygons& result_lines, Polygons& result_polygons) +void Infill::generateGyroidInfill(OpenLinesSet& result_lines, Shape& result_polygons) { - Polygons line_segments; + OpenLinesSet line_segments; GyroidInfill::generateTotalGyroidInfill(line_segments, zig_zaggify_, line_distance_, inner_contour_, z_); - PolylineStitcher::stitch(line_segments, result_lines, result_polygons, infill_line_width_); + OpenPolylineStitcher::stitch(line_segments, result_lines, result_polygons, infill_line_width_); } -void Infill::generateLightningInfill(const std::shared_ptr& trees, Polygons& result_lines) +void Infill::generateLightningInfill(const std::shared_ptr& trees, OpenLinesSet& result_lines) { // Don't need to support areas smaller than line width, as they are always within radius: if (std::abs(inner_contour_.area()) < infill_line_width_ || ! trees) { return; } - result_lines.add(trees->convertToLines(inner_contour_, infill_line_width_)); + result_lines.push_back(trees->convertToLines(inner_contour_, infill_line_width_)); } void Infill::generateConcentricInfill(std::vector& toolpaths, const Settings& settings) { const coord_t min_area = infill_line_width_ * infill_line_width_; - Polygons current_inset = inner_contour_; + Shape current_inset = inner_contour_; Simplify simplifier(settings); while (true) { @@ -470,13 +465,13 @@ void Infill::generateConcentricInfill(std::vector& toolpaths } } -void Infill::generateGridInfill(Polygons& result) +void Infill::generateGridInfill(OpenLinesSet& result) { generateLineInfill(result, line_distance_, fill_angle_, 0); generateLineInfill(result, line_distance_, fill_angle_ + 90, 0); } -void Infill::generateCubicInfill(Polygons& result) +void Infill::generateCubicInfill(OpenLinesSet& result) { const coord_t shift = one_over_sqrt_2 * z_; generateLineInfill(result, line_distance_, fill_angle_, shift); @@ -484,19 +479,19 @@ void Infill::generateCubicInfill(Polygons& result) generateLineInfill(result, line_distance_, fill_angle_ + 240, shift); } -void Infill::generateTetrahedralInfill(Polygons& result) +void Infill::generateTetrahedralInfill(OpenLinesSet& result) { generateHalfTetrahedralInfill(0.0, 0, result); generateHalfTetrahedralInfill(0.0, 90, result); } -void Infill::generateQuarterCubicInfill(Polygons& result) +void Infill::generateQuarterCubicInfill(OpenLinesSet& result) { generateHalfTetrahedralInfill(0.0, 0, result); generateHalfTetrahedralInfill(0.5, 90, result); } -void Infill::generateHalfTetrahedralInfill(double pattern_z_shift, int angle_shift, Polygons& result) +void Infill::generateHalfTetrahedralInfill(double pattern_z_shift, int angle_shift, OpenLinesSet& result) { const coord_t period = line_distance_ * 2; coord_t shift = coord_t(one_over_sqrt_2 * (z_ + pattern_z_shift * period * 2)) % period; @@ -507,29 +502,29 @@ void Infill::generateHalfTetrahedralInfill(double pattern_z_shift, int angle_shi generateLineInfill(result, period, fill_angle_ + angle_shift, -shift); } -void Infill::generateTriangleInfill(Polygons& result) +void Infill::generateTriangleInfill(OpenLinesSet& result) { generateLineInfill(result, line_distance_, fill_angle_, 0); generateLineInfill(result, line_distance_, fill_angle_ + 60, 0); generateLineInfill(result, line_distance_, fill_angle_ + 120, 0); } -void Infill::generateTrihexagonInfill(Polygons& result) +void Infill::generateTrihexagonInfill(OpenLinesSet& result) { generateLineInfill(result, line_distance_, fill_angle_, 0); generateLineInfill(result, line_distance_, fill_angle_ + 60, 0); generateLineInfill(result, line_distance_, fill_angle_ + 120, line_distance_ / 2); } -void Infill::generateCubicSubDivInfill(Polygons& result, const SliceMeshStorage& mesh) +void Infill::generateCubicSubDivInfill(OpenLinesSet& result, const SliceMeshStorage& mesh) { - Polygons uncropped; + OpenLinesSet uncropped; mesh.base_subdiv_cube->generateSubdivisionLines(z_, uncropped); constexpr bool restitch = false; // cubic subdivision lines are always single line segments - not polylines consisting of multiple segments. - result = outer_contour_.offset(infill_overlap_).intersectionPolyLines(uncropped, restitch); + result = outer_contour_.offset(infill_overlap_).intersection(uncropped, restitch); } -void Infill::generateCrossInfill(const SierpinskiFillProvider& cross_fill_provider, Polygons& result_polygons, Polygons& result_lines) +void Infill::generateCrossInfill(const SierpinskiFillProvider& cross_fill_provider, Shape& result_polygons, OpenLinesSet& result_lines) { Polygon cross_pattern_polygon = cross_fill_provider.generate(pattern_, z_, infill_line_width_, pocket_size_); @@ -540,24 +535,22 @@ void Infill::generateCrossInfill(const SierpinskiFillProvider& cross_fill_provid if (zig_zaggify_) { - Polygons cross_pattern_polygons; - cross_pattern_polygons.add(cross_pattern_polygon); - result_polygons.add(inner_contour_.intersection(cross_pattern_polygons)); + Shape cross_pattern_polygons; + cross_pattern_polygons.push_back(cross_pattern_polygon); + result_polygons.push_back(inner_contour_.intersection(cross_pattern_polygons)); } else { // make the polyline closed in order to handle cross_pattern_polygon as a polyline, rather than a closed polygon - cross_pattern_polygon.add(cross_pattern_polygon[0]); - - Polygons cross_pattern_polylines; - cross_pattern_polylines.add(cross_pattern_polygon); - Polygons poly_lines = inner_contour_.intersectionPolyLines(cross_pattern_polylines); - PolylineStitcher::stitch(poly_lines, result_lines, result_polygons, infill_line_width_); + OpenLinesSet cross_pattern_polylines; + cross_pattern_polylines.push_back(cross_pattern_polygon.toPseudoOpenPolyline()); + OpenLinesSet poly_lines = inner_contour_.intersection(cross_pattern_polylines); + OpenPolylineStitcher::stitch(poly_lines, result_lines, result_polygons, infill_line_width_); } } void Infill::addLineInfill( - Polygons& result, + OpenLinesSet& result, const PointMatrix& rotation_matrix, const int scanline_min_idx, const int line_distance, @@ -582,7 +575,7 @@ void Infill::addLineInfill( { // segment is too short to create infill continue; } - result.addLine(rotation_matrix.unapply(Point2LL(x, crossings[crossing_idx])), rotation_matrix.unapply(Point2LL(x, crossings[crossing_idx + 1]))); + result.addSegment(rotation_matrix.unapply(Point2LL(x, crossings[crossing_idx])), rotation_matrix.unapply(Point2LL(x, crossings[crossing_idx + 1]))); } scanline_idx += 1; } @@ -598,7 +591,7 @@ coord_t Infill::getShiftOffsetFromInfillOriginAndRotation(const double& infill_r return 0; } -void Infill::generateLineInfill(Polygons& result, int line_distance, const double& infill_rotation, coord_t shift) +void Infill::generateLineInfill(OpenLinesSet& result, int line_distance, const double& infill_rotation, coord_t shift) { shift += getShiftOffsetFromInfillOriginAndRotation(infill_rotation); PointMatrix rotation_matrix(infill_rotation); @@ -608,7 +601,7 @@ void Infill::generateLineInfill(Polygons& result, int line_distance, const doubl } -void Infill::generateZigZagInfill(Polygons& result, const coord_t line_distance, const double& infill_rotation) +void Infill::generateZigZagInfill(OpenLinesSet& result, const coord_t line_distance, const double& infill_rotation) { const coord_t shift = getShiftOffsetFromInfillOriginAndRotation(infill_rotation); @@ -641,7 +634,7 @@ void Infill::generateZigZagInfill(Polygons& result, const coord_t line_distance, * while I also call a boundary segment leaving from an even scanline toward the right as belonging to an even scansegment. */ void Infill::generateLinearBasedInfill( - Polygons& result, + OpenLinesSet& result, const int line_distance, const PointMatrix& rotation_matrix, ZigzagConnectorProcessor& zigzag_connector_processor, @@ -653,7 +646,7 @@ void Infill::generateLinearBasedInfill( return; } - Polygons outline = inner_contour_; // Make a copy. We'll be rotating this outline to make intersections always horizontal, for better performance. + Shape outline = inner_contour_; // Make a copy. We'll be rotating this outline to make intersections always horizontal, for better performance. outline.applyMatrix(rotation_matrix); coord_t shift = extra_shift + this->shift_; @@ -704,7 +697,7 @@ void Infill::generateLinearBasedInfill( for (size_t poly_idx = 0; poly_idx < outline.size(); poly_idx++) { - PolygonRef poly = outline[poly_idx]; + const Polygon& poly = outline[poly_idx]; if (connect_lines_) { crossings_on_line_[poly_idx].resize(poly.size()); // One for each line in this polygon. @@ -746,12 +739,12 @@ void Infill::generateLinearBasedInfill( for (int scanline_idx = scanline_idx0; scanline_idx != scanline_idx1 + direction; scanline_idx += direction) { - int x = scanline_idx * line_distance + shift; - int y = p1.Y + (p0.Y - p1.Y) * (x - p1.X) / (p0.X - p1.X); + const int x = scanline_idx * line_distance + shift; + const int y = p1.Y + (p0.Y - p1.Y) * (x - p1.X) / (p0.X - p1.X); assert(scanline_idx - scanline_min_idx >= 0 && scanline_idx - scanline_min_idx < int(cut_list.size()) && "reading infill cutlist index out of bounds!"); cut_list[scanline_idx - scanline_min_idx].push_back(y); Point2LL scanline_linesegment_intersection(x, y); - zigzag_connector_processor.registerScanlineSegmentIntersection(scanline_linesegment_intersection, scanline_idx); + zigzag_connector_processor.registerScanlineSegmentIntersection(scanline_linesegment_intersection, scanline_idx, line_distance / 4); crossings_per_scanline[scanline_idx - min_scanline_index].emplace_back(scanline_linesegment_intersection, poly_idx, point_idx); } zigzag_connector_processor.registerVertex(p1); @@ -854,7 +847,7 @@ void Infill::resolveIntersection(const coord_t at_distance, const Point2LL& inte } } -void Infill::connectLines(Polygons& result_lines) +void Infill::connectLines(OpenLinesSet& result_lines) { UnionFind connected_lines; // Keeps track of which lines are connected to which. for (const std::vector>& crossings_on_polygon : crossings_on_line_) @@ -874,7 +867,7 @@ void Infill::connectLines(Polygons& result_lines) const auto half_line_distance_squared = (line_distance_ * line_distance_) / 4; for (size_t polygon_index = 0; polygon_index < inner_contour_.size(); polygon_index++) { - ConstPolygonRef inner_contour_polygon = inner_contour_[polygon_index]; + const Polygon& inner_contour_polygon = inner_contour_[polygon_index]; if (inner_contour_polygon.empty()) { continue; @@ -1016,7 +1009,7 @@ void Infill::connectLines(Polygons& result_lines) } // Now go along the linked list of infill lines and output the infill lines to the actual result. - PolygonRef result_line = result_lines.newPoly(); + OpenPolyline& result_line = result_lines.newLine(); InfillLineSegment* old_line = current_infill_line; if (current_infill_line->previous_) { @@ -1058,21 +1051,21 @@ void Infill::InfillLineSegment::swapDirection() std::swap(next_, previous_); } -void Infill::InfillLineSegment::appendTo(PolygonRef& result_polyline, const bool include_start) +void Infill::InfillLineSegment::appendTo(OpenPolyline& result_polyline, const bool include_start) { if (include_start) { - result_polyline.add(altered_start_); + result_polyline.push_back(altered_start_); } if (start_bend_.has_value()) { - result_polyline.add(start_bend_.value()); + result_polyline.push_back(start_bend_.value()); } if (end_bend_.has_value()) { - result_polyline.add(end_bend_.value()); + result_polyline.push_back(end_bend_.value()); } - result_polyline.add(altered_end_); + result_polyline.push_back(altered_end_); } } // namespace cura diff --git a/src/infill/GyroidInfill.cpp b/src/infill/GyroidInfill.cpp index 6a95df7d54..c085840a21 100644 --- a/src/infill/GyroidInfill.cpp +++ b/src/infill/GyroidInfill.cpp @@ -1,11 +1,15 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #include "infill/GyroidInfill.h" +#include + +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" #include "utils/AABB.h" #include "utils/linearAlg2D.h" -#include "utils/polygon.h" namespace cura { @@ -18,7 +22,7 @@ GyroidInfill::~GyroidInfill() { } -void GyroidInfill::generateTotalGyroidInfill(Polygons& result_lines, bool zig_zaggify, coord_t line_distance, const Polygons& in_outline, coord_t z) +void GyroidInfill::generateTotalGyroidInfill(OpenLinesSet& result_lines, bool zig_zaggify, coord_t line_distance, const Shape& in_outline, coord_t z) { // generate infill based on the gyroid equation: sin_x * cos_y + sin_y * cos_z + sin_z * cos_x = 0 // kudos to the author of the Slic3r implementation equation code, the equation code here is based on that @@ -39,7 +43,7 @@ void GyroidInfill::generateTotalGyroidInfill(Polygons& result_lines, bool zig_za const double sin_z = std::sin(z_rads); std::vector odd_line_coords; std::vector even_line_coords; - Polygons result; + OpenLinesSet result; std::vector chains[2]; // [start_points[], end_points[]] std::vector connected_to[2]; // [chain_indices[], chain_indices[]] std::vector line_numbers; // which row/column line a chain is part of @@ -80,19 +84,19 @@ void GyroidInfill::generateTotalGyroidInfill(Polygons& result_lines, bool zig_za if (last_inside && current_inside) { // line doesn't hit the boundary, add the whole line - result.addLine(last, current); + result.addSegment(last, current); } else if (last_inside != current_inside) { // line hits the boundary, add the part that's inside the boundary - Polygons line; - line.addLine(last, current); + OpenLinesSet line; + line.addSegment(last, current); constexpr bool restitch = false; // only a single line doesn't need stitching - line = in_outline.intersectionPolyLines(line, restitch); + line = in_outline.intersection(line, restitch); if (line.size() > 0) { // some of the line is inside the boundary - result.addLine(line[0][0], line[0][1]); + result.addSegment(line[0][0], line[0][1]); if (zig_zaggify) { chain_end[chain_end_index] = line[0][(line[0][0] != last && line[0][0] != current) ? 0 : 1]; @@ -172,19 +176,19 @@ void GyroidInfill::generateTotalGyroidInfill(Polygons& result_lines, bool zig_za if (last_inside && current_inside) { // line doesn't hit the boundary, add the whole line - result.addLine(last, current); + result.addSegment(last, current); } else if (last_inside != current_inside) { // line hits the boundary, add the part that's inside the boundary - Polygons line; - line.addLine(last, current); + OpenLinesSet line; + line.addSegment(last, current); constexpr bool restitch = false; // only a single line doesn't need stitching - line = in_outline.intersectionPolyLines(line, restitch); + line = in_outline.intersection(line, restitch); if (line.size() > 0) { // some of the line is inside the boundary - result.addLine(line[0][0], line[0][1]); + result.addSegment(line[0][0], line[0][1]); if (zig_zaggify) { chain_end[chain_end_index] = line[0][(line[0][0] != last && line[0][0] != current) ? 0 : 1]; @@ -238,7 +242,7 @@ void GyroidInfill::generateTotalGyroidInfill(Polygons& result_lines, bool zig_za int chain_ends_remaining = chains[0].size() * 2; - for (ConstPolygonRef outline_poly : in_outline) + for (const Polygon& outline_poly : in_outline) { std::vector connector_points; // the points that make up a connector line @@ -340,7 +344,7 @@ void GyroidInfill::generateTotalGyroidInfill(Polygons& result_lines, bool zig_za if (chain_index != connector_start_chain_index && connected_to[(point_index + 1) % 2][chain_index] != connector_start_chain_index) { - result.add(connector_points); + result.push_back(OpenPolyline{ connector_points }); drawing = false; connector_points.clear(); // remember the connection @@ -389,9 +393,9 @@ void GyroidInfill::generateTotalGyroidInfill(Polygons& result_lines, bool zig_za { // output the connector line segments from the last chain to the first point in the outline connector_points.push_back(outline_poly[0]); - result.add(connector_points); + result.push_back(OpenPolyline{ connector_points }); // output the connector line segments from the first point in the outline to the first chain - result.add(path_to_first_chain); + result.push_back(OpenPolyline{ path_to_first_chain }); } if (chain_ends_remaining < 1) diff --git a/src/infill/LightningDistanceField.cpp b/src/infill/LightningDistanceField.cpp index 1afea87a24..bda9d7e27c 100644 --- a/src/infill/LightningDistanceField.cpp +++ b/src/infill/LightningDistanceField.cpp @@ -10,7 +10,7 @@ namespace cura constexpr coord_t radius_per_cell_size = 6; // The cell-size should be small compared to the radius, but not so small as to be inefficient. -LightningDistanceField::LightningDistanceField(const coord_t& radius, const Polygons& current_outline, const Polygons& current_overhang) +LightningDistanceField::LightningDistanceField(const coord_t& radius, const Shape& current_outline, const Shape& current_overhang) : cell_size_(radius / radius_per_cell_size) , grid_(cell_size_) , supporting_radius_(radius) @@ -20,7 +20,7 @@ LightningDistanceField::LightningDistanceField(const coord_t& radius, const Poly std::vector regular_dots = PolygonUtils::spreadDotsArea(current_overhang, cell_size_); for (const auto& p : regular_dots) { - const ClosestPolygonPoint cpp = PolygonUtils::findClosest(p, current_outline); + const ClosestPointPolygon cpp = PolygonUtils::findClosest(p, current_outline); const coord_t dist_to_boundary = vSize(p - cpp.p()); unsupported_points_.emplace_back(p, dist_to_boundary); } diff --git a/src/infill/LightningGenerator.cpp b/src/infill/LightningGenerator.cpp index a5a7749655..bf952de1d0 100644 --- a/src/infill/LightningGenerator.cpp +++ b/src/infill/LightningGenerator.cpp @@ -48,19 +48,19 @@ void LightningGenerator::generateInitialInternalOverhangs(const SliceMeshStorage const auto infill_line_width = mesh.settings.get("infill_line_width"); const coord_t infill_wall_offset = -infill_wall_line_count * infill_line_width; - Polygons infill_area_above; + Shape infill_area_above; // Iterate from top to bottom, to subtract the overhang areas above from the overhang areas on the layer below, to get only overhang in the top layer where it is overhanging. for (int layer_nr = mesh.layers.size() - 1; layer_nr >= 0; layer_nr--) { const SliceLayer& current_layer = mesh.layers[layer_nr]; - Polygons infill_area_here; + Shape infill_area_here; for (auto& part : current_layer.parts) { - infill_area_here.add(part.getOwnInfillArea().offset(infill_wall_offset)); + infill_area_here.push_back(part.getOwnInfillArea().offset(infill_wall_offset)); } // Remove the part of the infill area that is already supported by the walls. - Polygons overhang = infill_area_here.offset(-wall_supporting_radius).difference(infill_area_above); + Shape overhang = infill_area_here.offset(-wall_supporting_radius).difference(infill_area_above); overhang_per_layer[layer_nr] = overhang; infill_area_above = std::move(infill_area_here); @@ -80,15 +80,15 @@ void LightningGenerator::generateTrees(const SliceMeshStorage& mesh) const auto infill_line_width = mesh.settings.get("infill_line_width"); const coord_t infill_wall_offset = -infill_wall_line_count * infill_line_width; - std::vector infill_outlines; - infill_outlines.insert(infill_outlines.end(), mesh.layers.size(), Polygons()); + std::vector infill_outlines; + infill_outlines.insert(infill_outlines.end(), mesh.layers.size(), Shape()); // For-each layer from top to bottom: for (int layer_id = mesh.layers.size() - 1; layer_id >= 0; layer_id--) { for (const auto& part : mesh.layers[layer_id].parts) { - infill_outlines[layer_id].add(part.getOwnInfillArea().offset(infill_wall_offset)); + infill_outlines[layer_id].push_back(part.getOwnInfillArea().offset(infill_wall_offset)); } } @@ -100,7 +100,7 @@ void LightningGenerator::generateTrees(const SliceMeshStorage& mesh) for (int layer_id = top_layer_id; layer_id >= 0; layer_id--) { LightningLayer& current_lightning_layer = lightning_layers[layer_id]; - Polygons& current_outlines = infill_outlines[layer_id]; + Shape& current_outlines = infill_outlines[layer_id]; const auto& outlines_locator = *outlines_locator_ptr; // register all trees propagated from the previous layer as to-be-reconnected @@ -115,7 +115,7 @@ void LightningGenerator::generateTrees(const SliceMeshStorage& mesh) { return; } - const Polygons& below_outlines = infill_outlines[layer_id - 1]; + const Shape& below_outlines = infill_outlines[layer_id - 1]; outlines_locator_ptr = PolygonUtils::createLocToLineGrid(below_outlines, locator_cell_size); const auto& below_outlines_locator = *outlines_locator_ptr; diff --git a/src/infill/LightningLayer.cpp b/src/infill/LightningLayer.cpp index 6cefaff8be..9ecb401c41 100644 --- a/src/infill/LightningLayer.cpp +++ b/src/infill/LightningLayer.cpp @@ -5,6 +5,7 @@ #include // advance +#include "geometry/OpenPolyline.h" #include "infill/LightningDistanceField.h" #include "infill/LightningTreeNode.h" #include "sliceDataStorage.h" @@ -45,8 +46,8 @@ void LightningLayer::fillLocator(SparseLightningTreeNodeGrid& tree_node_locator) } void LightningLayer::generateNewTrees( - const Polygons& current_overhang, - const Polygons& current_outlines, + const Shape& current_overhang, + const Shape& current_outlines, const LocToLineGrid& outlines_locator, const coord_t supporting_radius, const coord_t wall_supporting_radius) @@ -80,14 +81,14 @@ void LightningLayer::generateNewTrees( GroundingLocation LightningLayer::getBestGroundingLocation( const Point2LL& unsupported_location, - const Polygons& current_outlines, + const Shape& current_outlines, const LocToLineGrid& outline_locator, const coord_t supporting_radius, const coord_t wall_supporting_radius, const SparseLightningTreeNodeGrid& tree_node_locator, const LightningTreeNodeSPtr& exclude_tree) { - ClosestPolygonPoint cpp = PolygonUtils::findClosest(unsupported_location, current_outlines); + ClosestPointPolygon cpp = PolygonUtils::findClosest(unsupported_location, current_outlines); Point2LL node_location = cpp.p(); const coord_t within_dist = vSize(node_location - unsupported_location); @@ -120,7 +121,7 @@ GroundingLocation LightningLayer::getBestGroundingLocation( } else { - return GroundingLocation{ sub_tree, std::optional() }; + return GroundingLocation{ sub_tree, std::optional() }; } } @@ -143,7 +144,7 @@ bool LightningLayer::attach(const Point2LL& unsupported_location, const Groundin void LightningLayer::reconnectRoots( std::vector& to_be_reconnected_tree_roots, - const Polygons& current_outlines, + const Shape& current_outlines, const LocToLineGrid& outline_locator, const coord_t supporting_radius, const coord_t wall_supporting_radius) @@ -217,19 +218,19 @@ void LightningLayer::reconnectRoots( } // Returns 'added someting'. -Polygons LightningLayer::convertToLines(const Polygons& limit_to_outline, const coord_t line_width) const +OpenLinesSet LightningLayer::convertToLines(const Shape& limit_to_outline, const coord_t line_width) const { - Polygons result_lines; + OpenLinesSet result_lines; if (tree_roots.empty()) { return result_lines; } - for (const auto& tree : tree_roots) + for (const LightningTreeNodeSPtr& tree : tree_roots) { tree->convertToPolylines(result_lines, line_width); } - result_lines = limit_to_outline.intersectionPolyLines(result_lines); + result_lines = limit_to_outline.intersection(result_lines); return result_lines; } diff --git a/src/infill/LightningTreeNode.cpp b/src/infill/LightningTreeNode.cpp index 573cffb80a..76ef91dbe1 100644 --- a/src/infill/LightningTreeNode.cpp +++ b/src/infill/LightningTreeNode.cpp @@ -3,6 +3,7 @@ #include "infill/LightningTreeNode.h" +#include "geometry/OpenPolyline.h" #include "utils/linearAlg2D.h" using namespace cura; @@ -64,7 +65,7 @@ LightningTreeNodeSPtr LightningTreeNode::addChild(LightningTreeNodeSPtr& new_chi void LightningTreeNode::propagateToNextLayer( std::vector& next_trees, - const Polygons& next_outlines, + const Shape& next_outlines, const LocToLineGrid& outline_locator, const coord_t prune_distance, const coord_t smooth_magnitude, @@ -169,7 +170,7 @@ LightningTreeNodeSPtr LightningTreeNode::closestNode(const Point2LL& loc) return result; } -bool LightningTreeNode::realign(const Polygons& outlines, const LocToLineGrid& outline_locator, std::vector& rerooted_parts) +bool LightningTreeNode::realign(const Shape& outlines, const LocToLineGrid& outline_locator, std::vector& rerooted_parts) { if (outlines.empty()) { @@ -354,43 +355,43 @@ const std::optional& LightningTreeNode::getLastGroundingLocation() con return last_grounding_location_; } -void LightningTreeNode::convertToPolylines(Polygons& output, const coord_t line_width) const +void LightningTreeNode::convertToPolylines(OpenLinesSet& output, const coord_t line_width) const { - Polygons result; - result.newPoly(); + OpenLinesSet result; + result.emplace_back(); convertToPolylines(0, result); removeJunctionOverlap(result, line_width); - output.add(result); + output.push_back(result); } -void LightningTreeNode::convertToPolylines(size_t long_line_idx, Polygons& output) const +void LightningTreeNode::convertToPolylines(size_t long_line_idx, OpenLinesSet& output) const { if (children_.empty()) { - output[long_line_idx].add(p_); + output[long_line_idx].push_back(p_); return; } size_t first_child_idx = rand() % children_.size(); children_[first_child_idx]->convertToPolylines(long_line_idx, output); - output[long_line_idx].add(p_); + output[long_line_idx].push_back(p_); for (size_t idx_offset = 1; idx_offset < children_.size(); idx_offset++) { size_t child_idx = (first_child_idx + idx_offset) % children_.size(); const LightningTreeNode& child = *children_[child_idx]; - output.newPoly(); + output.emplace_back(); size_t child_line_idx = output.size() - 1; child.convertToPolylines(child_line_idx, output); - output[child_line_idx].add(p_); + output[child_line_idx].push_back(p_); } } -void LightningTreeNode::removeJunctionOverlap(Polygons& result_lines, const coord_t line_width) const +void LightningTreeNode::removeJunctionOverlap(OpenLinesSet& result_lines, const coord_t line_width) const { const coord_t reduction = line_width / 2; // TODO make configurable? for (auto poly_it = result_lines.begin(); poly_it != result_lines.end();) { - PolygonRef polyline = *poly_it; + OpenPolyline& polyline = *poly_it; if (polyline.size() <= 1) { polyline = std::move(result_lines.back()); diff --git a/src/infill/NoZigZagConnectorProcessor.cpp b/src/infill/NoZigZagConnectorProcessor.cpp index 1ffa6532f4..25cb0fa48e 100644 --- a/src/infill/NoZigZagConnectorProcessor.cpp +++ b/src/infill/NoZigZagConnectorProcessor.cpp @@ -14,7 +14,7 @@ void NoZigZagConnectorProcessor::registerVertex(const Point2LL&) // No need to add anything. } -void NoZigZagConnectorProcessor::registerScanlineSegmentIntersection(const Point2LL&, int) +void NoZigZagConnectorProcessor::registerScanlineSegmentIntersection(const Point2LL&, int, coord_t) { // No need to add anything. } diff --git a/src/infill/SierpinskiFill.cpp b/src/infill/SierpinskiFill.cpp index bdb4962228..8c5663b1b1 100644 --- a/src/infill/SierpinskiFill.cpp +++ b/src/infill/SierpinskiFill.cpp @@ -10,11 +10,11 @@ #include +#include "geometry/Polygon.h" #include "infill/ImageBasedDensityProvider.h" #include "infill/UniformDensityProvider.h" #include "utils/AABB3D.h" #include "utils/SVG.h" -#include "utils/polygon.h" namespace cura { @@ -718,10 +718,10 @@ Polygon SierpinskiFill::generateCross() const edge_middle -= triangle.a_; break; } - ret.add(edge_middle / 2); + ret.push_back(edge_middle / 2); } - double realized_length = INT2MM(ret.polygonLength()); + double realized_length = INT2MM(ret.length()); double requested_length = root_.requested_length_; double error = (realized_length - requested_length) / requested_length; spdlog::debug("realized_length: {}, requested_length: {} :: {}% error", realized_length, requested_length, 0.01 * static_cast(10000 * error)); @@ -759,13 +759,13 @@ Polygon SierpinskiFill::generateCross(coord_t z, coord_t min_dist_to_side, coord sides. The steeper overhang is then only in the corner, which is deemed acceptable since the corners are never too sharp. */ const coord_t period = vSize(triangle.straight_corner_ - triangle.a_); - ret.add(get_edge_crossing_location(period, triangle.getFromEdge())); + ret.push_back(get_edge_crossing_location(period, triangle.getFromEdge())); last_triangle = ▵ } assert(last_triangle); const coord_t period = vSize(last_triangle->straight_corner_ - last_triangle->a_); - ret.add(get_edge_crossing_location(period, last_triangle->getToEdge())); + ret.push_back(get_edge_crossing_location(period, last_triangle->getToEdge())); if (pocket_size > 10) { @@ -793,12 +793,12 @@ Polygon SierpinskiFill::generateCross(coord_t z, coord_t min_dist_to_side, coord { coord_t pocket_rounding = std::min(std::min(pocket_size_side, vSize(v0) / 3), vSize(v1) / 3); // a third so that if a line segment is shortened on both sides the middle remains - pocketed.add(p1 + normal(v0, pocket_rounding)); - pocketed.add(p1 + normal(v1, pocket_rounding)); + pocketed.push_back(p1 + normal(v0, pocket_rounding)); + pocketed.push_back(p1 + normal(v1, pocket_rounding)); } else { - pocketed.add(p1); + pocketed.push_back(p1); } p0 = p1; } diff --git a/src/infill/SierpinskiFillProvider.cpp b/src/infill/SierpinskiFillProvider.cpp index 0123c42b04..769029aa11 100644 --- a/src/infill/SierpinskiFillProvider.cpp +++ b/src/infill/SierpinskiFillProvider.cpp @@ -5,11 +5,11 @@ #include +#include "geometry/Polygon.h" #include "infill/ImageBasedDensityProvider.h" #include "infill/UniformDensityProvider.h" #include "utils/AABB3D.h" #include "utils/math.h" -#include "utils/polygon.h" namespace cura { diff --git a/src/infill/SubDivCube.cpp b/src/infill/SubDivCube.cpp index f30affc602..a3490a9add 100644 --- a/src/infill/SubDivCube.cpp +++ b/src/infill/SubDivCube.cpp @@ -5,6 +5,9 @@ #include +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" #include "settings/types/Angle.h" //For the infill angle. #include "sliceDataStorage.h" #include "utils/math.h" @@ -83,27 +86,27 @@ void SubDivCube::precomputeOctree(SliceMeshStorage& mesh, const Point2LL& infill mesh.base_subdiv_cube = std::make_shared(mesh, center, curr_recursion_depth - 1); } -void SubDivCube::generateSubdivisionLines(const coord_t z, Polygons& result) +void SubDivCube::generateSubdivisionLines(const coord_t z, OpenLinesSet& result) { if (cube_properties_per_recursion_step_.empty()) // Infill is set to 0%. { return; } - Polygons directional_line_groups[3]; + OpenLinesSet directional_line_groups[3]; generateSubdivisionLines(z, directional_line_groups); for (int dir_idx = 0; dir_idx < 3; dir_idx++) { - Polygons& line_group = directional_line_groups[dir_idx]; + OpenLinesSet& line_group = directional_line_groups[dir_idx]; for (unsigned int line_idx = 0; line_idx < line_group.size(); line_idx++) { - result.addLine(line_group[line_idx][0], line_group[line_idx][1]); + result.addSegment(line_group[line_idx][0], line_group[line_idx][1]); } } } -void SubDivCube::generateSubdivisionLines(const coord_t z, Polygons (&directional_line_groups)[3]) +void SubDivCube::generateSubdivisionLines(const coord_t z, OpenLinesSet (&directional_line_groups)[3]) { CubeProperties cube_properties = cube_properties_per_recursion_step_[depth_]; @@ -228,15 +231,15 @@ coord_t SubDivCube::distanceFromPointToMesh(SliceMeshStorage& mesh, const LayerI return 2; *distance2 = 0; } - Polygons collide; + Shape collide; for (const SliceLayerPart& part : mesh.layers[layer_nr].parts) { - collide.add(part.infill_area); + collide.push_back(part.infill_area); } Point2LL centerpoint = location; bool inside = collide.inside(centerpoint); - ClosestPolygonPoint border_point = PolygonUtils::moveInside2(collide, centerpoint); + ClosestPointPolygon border_point = PolygonUtils::moveInside2(collide, centerpoint); Point2LL diff = border_point.location_ - location; *distance2 = vSize2(diff); if (inside) @@ -261,7 +264,7 @@ void SubDivCube::rotatePoint120(Point2LL& target) target.X = x; } -void SubDivCube::addLineAndCombine(Polygons& group, Point2LL from, Point2LL to) +void SubDivCube::addLineAndCombine(OpenLinesSet& group, Point2LL from, Point2LL to) { int epsilon = 10; // the smallest distance of two points which are viewed as coincident (dist > 0 due to rounding errors) for (unsigned int idx = 0; idx < group.size(); idx++) @@ -269,19 +272,19 @@ void SubDivCube::addLineAndCombine(Polygons& group, Point2LL from, Point2LL to) if (std::abs(from.X - group[idx][1].X) < epsilon && std::abs(from.Y - group[idx][1].Y) < epsilon) { from = group[idx][0]; - group.remove(idx); + group.removeAt(idx); idx--; continue; } if (std::abs(to.X - group[idx][0].X) < epsilon && std::abs(to.Y - group[idx][0].Y) < epsilon) { to = group[idx][1]; - group.remove(idx); + group.removeAt(idx); idx--; continue; } } - group.addLine(from, to); + group.addSegment(from, to); } } // namespace cura diff --git a/src/infill/ZigzagConnectorProcessor.cpp b/src/infill/ZigzagConnectorProcessor.cpp index 7f75b608ff..5cc94e9845 100644 --- a/src/infill/ZigzagConnectorProcessor.cpp +++ b/src/infill/ZigzagConnectorProcessor.cpp @@ -5,6 +5,11 @@ #include +#include "geometry/OpenPolyline.h" +#include "geometry/PointMatrix.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" + using namespace cura; @@ -20,7 +25,6 @@ void ZigzagConnectorProcessor::registerVertex(const Point2LL& vertex) } } - bool ZigzagConnectorProcessor::shouldAddCurrentConnector(int start_scanline_idx, int end_scanline_idx) const { int direction = end_scanline_idx - start_scanline_idx; @@ -83,8 +87,26 @@ bool ZigzagConnectorProcessor::shouldAddCurrentConnector(int start_scanline_idx, return should_add; } +bool ZigzagConnectorProcessor::handleConnectorTooCloseToSegment(const coord_t scanline_x, const coord_t min_distance_to_scanline) +{ + if (current_connector_.empty()) + { + return false; + } + else + { + return std::find_if( + current_connector_.begin(), + current_connector_.end(), + [scanline_x, min_distance_to_scanline](const Point2LL& point) + { + return std::abs(point.X - scanline_x) >= min_distance_to_scanline; + }) + == current_connector_.end(); + } +} -void ZigzagConnectorProcessor::registerScanlineSegmentIntersection(const Point2LL& intersection, int scanline_index) +void ZigzagConnectorProcessor::registerScanlineSegmentIntersection(const Point2LL& intersection, int scanline_index, coord_t min_distance_to_scanline) { if (is_first_connector_) { @@ -97,7 +119,7 @@ void ZigzagConnectorProcessor::registerScanlineSegmentIntersection(const Point2L else { // add the current connector if needed - if (shouldAddCurrentConnector(last_connector_index_, scanline_index)) + if (shouldAddCurrentConnector(last_connector_index_, scanline_index) && ! handleConnectorTooCloseToSegment(intersection.X, min_distance_to_scanline)) { const bool is_this_endpiece = scanline_index == last_connector_index_; current_connector_.push_back(intersection); @@ -143,10 +165,28 @@ void ZigzagConnectorProcessor::addZagConnector(std::vector& points, bo { return; } - Polygon polyline(points); + OpenPolyline polyline(points); if (is_endpiece && ! connected_endpieces_) { polyline.pop_back(); } addPolyline(polyline); } + +void cura::ZigzagConnectorProcessor::reset() +{ + is_first_connector_ = true; + first_connector_end_scanline_index_ = 0; + last_connector_index_ = 0; + first_connector_.clear(); + current_connector_.clear(); +} + +void cura::ZigzagConnectorProcessor::addPolyline(const OpenPolyline& polyline) +{ + result_.emplace_back(polyline); + for (Point2LL& p : result_.back()) + { + p = rotation_matrix_.unapply(p); + } +} diff --git a/src/layerPart.cpp b/src/layerPart.cpp index db08a9a196..e1754b73fe 100644 --- a/src/layerPart.cpp +++ b/src/layerPart.cpp @@ -3,18 +3,19 @@ #include "layerPart.h" +#include "geometry/OpenPolyline.h" #include "progress/Progress.h" #include "settings/EnumSettings.h" //For ESurfaceMode. #include "settings/Settings.h" #include "sliceDataStorage.h" #include "slicer.h" -#include "utils/PolylineStitcher.h" +#include "utils/OpenPolylineStitcher.h" #include "utils/Simplify.h" //Simplifying the layers after creating them. #include "utils/ThreadPool.h" /* The layer-part creation step is the first step in creating actual useful data for 3D printing. -It takes the result of the Slice step, which is an unordered list of polygons, and makes groups of polygons, +It takes the result of the Slice step, which is an unordered list of polygons_, and makes groups of polygons_, each of these groups is called a "part", which sometimes are also known as "islands". These parts represent isolated areas in the 2D layer with possible holes. @@ -29,39 +30,39 @@ namespace cura void createLayerWithParts(const Settings& settings, SliceLayer& storageLayer, SlicerLayer* layer) { - PolylineStitcher::stitch(layer->openPolylines, storageLayer.openPolyLines, layer->polygons, settings.get("wall_line_width_0")); + OpenPolylineStitcher::stitch(layer->open_polylines_, storageLayer.open_polylines, layer->polygons_, settings.get("wall_line_width_0")); - storageLayer.openPolyLines = Simplify(settings).polyline(storageLayer.openPolyLines); + storageLayer.open_polylines = Simplify(settings).polyline(storageLayer.open_polylines); const bool union_all_remove_holes = settings.get("meshfix_union_all_remove_holes"); if (union_all_remove_holes) { - for (unsigned int i = 0; i < layer->polygons.size(); i++) + for (unsigned int i = 0; i < layer->polygons_.size(); i++) { - if (layer->polygons[i].orientation()) - layer->polygons[i].reverse(); + if (layer->polygons_[i].orientation()) + layer->polygons_[i].reverse(); } } - std::vector result; + std::vector result; const bool union_layers = settings.get("meshfix_union_all"); const ESurfaceMode surface_only = settings.get("magic_mesh_surface_mode"); if (surface_only == ESurfaceMode::SURFACE && ! union_layers) { // Don't do anything with overlapping areas; no union nor xor - result.reserve(layer->polygons.size()); - for (const PolygonRef poly : layer->polygons) + result.reserve(layer->polygons_.size()); + for (const Polygon& poly : layer->polygons_) { if (poly.empty()) { continue; } result.emplace_back(); - result.back().add(poly); + result.back().push_back(poly); } } else { - result = layer->polygons.splitIntoParts(union_layers || union_all_remove_holes); + result = layer->polygons_.splitIntoParts(union_layers || union_all_remove_holes); } for (auto& part : result) @@ -98,7 +99,7 @@ void createLayerParts(SliceMeshStorage& mesh, Slicer* slicer) for (LayerIndex layer_nr = total_layers - 1; layer_nr >= 0; layer_nr--) { SliceLayer& layer_storage = mesh.layers[layer_nr]; - if (layer_storage.parts.size() > 0 || (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL && layer_storage.openPolyLines.size() > 0)) + if (layer_storage.parts.size() > 0 || (mesh.settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL && layer_storage.open_polylines.size() > 0)) { mesh.layer_nr_max_filled_layer = layer_nr; // last set by the highest non-empty layer break; diff --git a/src/mesh.cpp b/src/mesh.cpp index 740dc442c8..97c16800fd 100644 --- a/src/mesh.cpp +++ b/src/mesh.cpp @@ -3,6 +3,8 @@ #include "mesh.h" +#include + #include #include "utils/Point3D.h" diff --git a/src/multiVolumes.cpp b/src/multiVolumes.cpp index 6852469919..68c7231522 100644 --- a/src/multiVolumes.cpp +++ b/src/multiVolumes.cpp @@ -7,10 +7,12 @@ #include "Application.h" #include "Slice.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" #include "settings/EnumSettings.h" #include "settings/types/LayerIndex.h" #include "slicer.h" -#include "utils/PolylineStitcher.h" +#include "utils/OpenPolylineStitcher.h" namespace cura { @@ -53,11 +55,11 @@ void carveMultipleVolumes(std::vector& volumes) SlicerLayer& layer2 = volume_2.layers[layerNr]; if (alternate_carve_order && layerNr % 2 == 0 && volume_1.mesh->settings_.get("infill_mesh_order") == volume_2.mesh->settings_.get("infill_mesh_order")) { - layer2.polygons = layer2.polygons.difference(layer1.polygons); + layer2.polygons_ = layer2.polygons_.difference(layer1.polygons_); } else { - layer1.polygons = layer1.polygons.difference(layer2.polygons); + layer1.polygons_ = layer1.polygons_.difference(layer2.polygons_); } } } @@ -88,7 +90,7 @@ void generateMultipleVolumesOverlap(std::vector& volumes) aabb.expandXY(overlap); // expand to account for the case where two models and their bounding boxes are adjacent along the X or Y-direction for (LayerIndex layer_nr = 0; layer_nr < volume->layers.size(); layer_nr++) { - Polygons all_other_volumes; + Shape all_other_volumes; for (Slicer* other_volume : volumes) { if (other_volume->mesh->settings_.get("infill_mesh") || other_volume->mesh->settings_.get("anti_overhang_mesh") @@ -97,11 +99,11 @@ void generateMultipleVolumesOverlap(std::vector& volumes) continue; } SlicerLayer& other_volume_layer = other_volume->layers[layer_nr]; - all_other_volumes = all_other_volumes.unionPolygons(other_volume_layer.polygons.offset(offset_to_merge_other_merged_volumes), fill_type); + all_other_volumes = all_other_volumes.unionPolygons(other_volume_layer.polygons_.offset(offset_to_merge_other_merged_volumes), fill_type); } SlicerLayer& volume_layer = volume->layers[layer_nr]; - volume_layer.polygons = volume_layer.polygons.unionPolygons(all_other_volumes.intersection(volume_layer.polygons.offset(overlap / 2)), fill_type); + volume_layer.polygons_ = volume_layer.polygons_.unionPolygons(all_other_volumes.intersection(volume_layer.polygons_.offset(overlap / 2)), fill_type); } } } @@ -118,28 +120,29 @@ void MultiVolumes::carveCuttingMeshes(std::vector& volumes, const std:: Slicer& cutting_mesh_volume = *volumes[carving_mesh_idx]; for (LayerIndex layer_nr = 0; layer_nr < cutting_mesh_volume.layers.size(); layer_nr++) { - Polygons& cutting_mesh_polygons = cutting_mesh_volume.layers[layer_nr].polygons; - Polygons& cutting_mesh_polylines = cutting_mesh_volume.layers[layer_nr].openPolylines; - Polygons cutting_mesh_area_recomputed; - Polygons* cutting_mesh_area; + Shape& cutting_mesh_polygons = cutting_mesh_volume.layers[layer_nr].polygons_; + OpenLinesSet& cutting_mesh_polylines = cutting_mesh_volume.layers[layer_nr].open_polylines_; + Shape cutting_mesh_area_recomputed; + Shape* cutting_mesh_area; coord_t surface_line_width = cutting_mesh.settings_.get("wall_line_width_0"); { // compute cutting_mesh_area if (cutting_mesh.settings_.get("magic_mesh_surface_mode") == ESurfaceMode::BOTH) { - cutting_mesh_area_recomputed = cutting_mesh_polygons.unionPolygons(cutting_mesh_polylines.offsetPolyLine(surface_line_width / 2)); + cutting_mesh_area_recomputed = cutting_mesh_polygons.unionPolygons(cutting_mesh_polylines.offset(surface_line_width / 2)); cutting_mesh_area = &cutting_mesh_area_recomputed; } else if (cutting_mesh.settings_.get("magic_mesh_surface_mode") == ESurfaceMode::SURFACE) { // break up polygons into polylines // they have to be polylines, because they might break up further when doing the cutting - for (PolygonRef poly : cutting_mesh_polygons) + for (Polygon& poly : cutting_mesh_polygons) { - poly.add(poly[0]); + poly.push_back(poly.front()); + cutting_mesh_polylines.emplace_back(poly.getPoints()); } - cutting_mesh_polylines.add(cutting_mesh_polygons); + cutting_mesh_polygons.clear(); - cutting_mesh_area_recomputed = cutting_mesh_polylines.offsetPolyLine(surface_line_width / 2); + cutting_mesh_area_recomputed = cutting_mesh_polylines.offset(surface_line_width / 2); cutting_mesh_area = &cutting_mesh_area_recomputed; } else @@ -148,8 +151,8 @@ void MultiVolumes::carveCuttingMeshes(std::vector& volumes, const std:: } } - Polygons new_outlines; - Polygons new_polylines; + Shape new_outlines; + OpenLinesSet new_polylines; for (unsigned int carved_mesh_idx = 0; carved_mesh_idx < volumes.size(); carved_mesh_idx++) { const Mesh& carved_mesh = meshes[carved_mesh_idx]; @@ -159,13 +162,13 @@ void MultiVolumes::carveCuttingMeshes(std::vector& volumes, const std:: continue; } Slicer& carved_volume = *volumes[carved_mesh_idx]; - Polygons& carved_mesh_layer = carved_volume.layers[layer_nr].polygons; + Shape& carved_mesh_layer = carved_volume.layers[layer_nr].polygons_; - Polygons intersection = cutting_mesh_polygons.intersection(carved_mesh_layer); - new_outlines.add(intersection); + Shape intersection = cutting_mesh_polygons.intersection(carved_mesh_layer); + new_outlines.push_back(intersection); if (cutting_mesh.settings_.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL) // niet te geleuven { - new_polylines.add(carved_mesh_layer.intersectionPolyLines(cutting_mesh_polylines)); + new_polylines.push_back(carved_mesh_layer.intersection(cutting_mesh_polylines)); } carved_mesh_layer = carved_mesh_layer.difference(*cutting_mesh_area); @@ -174,7 +177,7 @@ void MultiVolumes::carveCuttingMeshes(std::vector& volumes, const std:: if (cutting_mesh.settings_.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL) { cutting_mesh_polylines.clear(); - PolylineStitcher::stitch(new_polylines, cutting_mesh_polylines, cutting_mesh_polygons, surface_line_width); + OpenPolylineStitcher::stitch(new_polylines, cutting_mesh_polylines, cutting_mesh_polygons, surface_line_width); } } } diff --git a/src/pathPlanning/Comb.cpp b/src/pathPlanning/Comb.cpp index e8fbcb5919..a5659a29e6 100644 --- a/src/pathPlanning/Comb.cpp +++ b/src/pathPlanning/Comb.cpp @@ -29,7 +29,7 @@ LocToLineGrid& Comb::getOutsideLocToLine(const ExtruderTrain& train) return *outside_loc_to_line_[train.extruder_nr_]; } -Polygons& Comb::getBoundaryOutside(const ExtruderTrain& train) +Shape& Comb::getBoundaryOutside(const ExtruderTrain& train) { if (boundary_outside_[train.extruder_nr_].empty()) { @@ -39,7 +39,7 @@ Polygons& Comb::getBoundaryOutside(const ExtruderTrain& train) return boundary_outside_[train.extruder_nr_]; } -Polygons& Comb::getModelBoundary(const ExtruderTrain& train) +Shape& Comb::getModelBoundary(const ExtruderTrain& train) { if (model_boundary_[train.extruder_nr_].empty()) { @@ -61,8 +61,8 @@ LocToLineGrid& Comb::getModelBoundaryLocToLine(const ExtruderTrain& train) Comb::Comb( const SliceDataStorage& storage, const LayerIndex layer_nr, - const Polygons& comb_boundary_inside_minimum, - const Polygons& comb_boundary_inside_optimal, + const Shape& comb_boundary_inside_minimum, + const Shape& comb_boundary_inside_optimal, coord_t comb_boundary_offset, coord_t travel_avoid_distance, coord_t move_inside_distance) @@ -119,7 +119,7 @@ bool Comb::calc( // normal combing within part using optimal comb boundary if (start_inside && end_inside && start_part_idx == end_part_idx) { - PolygonsPart part = parts_view_inside_optimal_.assemblePart(start_part_idx); + SingleShape part = parts_view_inside_optimal_.assemblePart(start_part_idx); comb_paths.emplace_back(); const bool combing_succeeded = LinePolygonsCrossings::comb( part, @@ -155,7 +155,7 @@ bool Comb::calc( // normal combing within part using minimum comb boundary if (start_inside_min && end_inside_min && start_part_idx_min == end_part_idx_min) { - PolygonsPart part = parts_view_inside_minimum_.assemblePart(start_part_idx_min); + SingleShape part = parts_view_inside_minimum_.assemblePart(start_part_idx_min); comb_paths.emplace_back(); comb_result = LinePolygonsCrossings::comb( @@ -374,7 +374,7 @@ bool Comb::calc( } // Try to move comb_path_input points inside by the amount of `move_inside_distance` and see if the points are still in boundary_inside_optimal, add result in comb_path_output -void Comb::moveCombPathInside(Polygons& boundary_inside, Polygons& boundary_inside_optimal, CombPath& comb_path_input, CombPath& comb_path_output) +void Comb::moveCombPathInside(Shape& boundary_inside, Shape& boundary_inside_optimal, CombPath& comb_path_input, CombPath& comb_path_output) { const coord_t dist = move_inside_distance_; const coord_t dist2 = dist * dist; @@ -409,7 +409,7 @@ Comb::Crossing::Crossing( const bool dest_is_inside, const unsigned int dest_part_idx, const unsigned int dest_part_boundary_crossing_poly_idx, - const Polygons& boundary_inside, + const Shape& boundary_inside, const LocToLineGrid& inside_loc_to_line) : dest_is_inside_(dest_is_inside) , boundary_inside_(boundary_inside) @@ -419,16 +419,16 @@ Comb::Crossing::Crossing( { if (dest_is_inside) { - dest_crossing_poly_.emplace(boundary_inside[dest_part_boundary_crossing_poly_idx]); // initialize with most obvious poly, cause mostly a combing move will move outside the - // part, rather than inside a hole in the part + dest_crossing_poly_ = &(boundary_inside[dest_part_boundary_crossing_poly_idx]); // initialize with most obvious poly, cause mostly a combing move will move outside the + // part, rather than inside a hole in the part } } -bool Comb::moveInside(Polygons& boundary_inside, bool is_inside, LocToLineGrid* inside_loc_to_line, Point2LL& dest_point, size_t& inside_poly) +bool Comb::moveInside(Shape& boundary_inside, bool is_inside, LocToLineGrid* inside_loc_to_line, Point2LL& dest_point, size_t& inside_poly) { if (is_inside) { - ClosestPolygonPoint cpp + ClosestPointPolygon cpp = PolygonUtils::ensureInsideOrOutside(boundary_inside, dest_point, offset_extra_start_end_, max_moveInside_distance2_, &boundary_inside, inside_loc_to_line); if (! cpp.isValid()) { @@ -456,7 +456,7 @@ void Comb::Crossing::findCrossingInOrMid(const PartsView& partsView_inside, cons }); dest_part_ = partsView_inside.assemblePart(dest_part_idx_); - ClosestPolygonPoint boundary_crossing_point; + ClosestPointPolygon boundary_crossing_point; { // set [result] to a point on the destination part closest to close_to (but also a bit close to _dest_point) std::unordered_set dest_part_poly_indices; for (unsigned int poly_idx : partsView_inside[dest_part_idx_]) @@ -476,7 +476,7 @@ void Comb::Crossing::findCrossingInOrMid(const PartsView& partsView_inside, cons if (dist2_score_here < dist2_score) { dist2_score = dist2_score_here; - boundary_crossing_point = ClosestPolygonPoint(closest_here, boundary_segment.point_idx_, boundary_segment.getPolygon(), boundary_segment.poly_idx_); + boundary_crossing_point = ClosestPointPolygon(closest_here, boundary_segment.point_idx_, &boundary_segment.getPolygon(), boundary_segment.poly_idx_); } return true; }; @@ -489,7 +489,7 @@ void Comb::Crossing::findCrossingInOrMid(const PartsView& partsView_inside, cons result = dest_point_; } - ClosestPolygonPoint crossing_1_in_cp = PolygonUtils::ensureInsideOrOutside( + ClosestPointPolygon crossing_1_in_cp = PolygonUtils::ensureInsideOrOutside( dest_part_, result, boundary_crossing_point, @@ -513,7 +513,7 @@ void Comb::Crossing::findCrossingInOrMid(const PartsView& partsView_inside, cons } } -bool Comb::Crossing::findOutside(const ExtruderTrain& train, const Polygons& outside, const Point2LL close_to, const bool fail_on_unavoidable_obstacles, Comb& comber) +bool Comb::Crossing::findOutside(const ExtruderTrain& train, const Shape& outside, const Point2LL close_to, const bool fail_on_unavoidable_obstacles, Comb& comber) { out_ = in_or_mid_; if (dest_is_inside_ || outside.inside(in_or_mid_, true)) // start in_between @@ -524,7 +524,7 @@ bool Comb::Crossing::findOutside(const ExtruderTrain& train, const Polygons& out { return vSize2((candidate - preferred_crossing_1_out) / 2); }); - std::optional crossing_1_out_cpp = PolygonUtils::findClose(in_or_mid_, outside, comber.getOutsideLocToLine(train), close_to_penalty_function); + std::optional crossing_1_out_cpp = PolygonUtils::findClose(in_or_mid_, outside, comber.getOutsideLocToLine(train), close_to_penalty_function); if (crossing_1_out_cpp) { out_ = PolygonUtils::moveOutside(*crossing_1_out_cpp, comber.offset_dist_to_get_from_on_the_polygon_to_outside_); @@ -539,7 +539,7 @@ bool Comb::Crossing::findOutside(const ExtruderTrain& train, const Polygons& out { // if move is too far over in_between // find crossing closer by assert(dest_crossing_poly_ && "destination crossing poly should have been instantiated!"); - std::shared_ptr> best = findBestCrossing(train, outside, **dest_crossing_poly_, dest_point_, close_to, comber); + std::shared_ptr> best = findBestCrossing(train, outside, **dest_crossing_poly_, dest_point_, close_to, comber); if (best) { in_or_mid_ = PolygonUtils::moveInside(best->first, comber.offset_dist_to_get_from_on_the_polygon_to_outside_); @@ -554,21 +554,21 @@ bool Comb::Crossing::findOutside(const ExtruderTrain& train, const Polygons& out } -std::shared_ptr> Comb::Crossing::findBestCrossing( +std::shared_ptr> Comb::Crossing::findBestCrossing( const ExtruderTrain& train, - const Polygons& outside, - ConstPolygonRef from, + const Shape& outside, + const Polygon& from, const Point2LL estimated_start, const Point2LL estimated_end, Comb& comber) { - ClosestPolygonPoint* best_in = nullptr; - ClosestPolygonPoint* best_out = nullptr; + ClosestPointPolygon* best_in = nullptr; + ClosestPointPolygon* best_out = nullptr; coord_t best_detour_score = std::numeric_limits::max(); coord_t best_crossing_dist2; - std::vector> crossing_out_candidates = PolygonUtils::findClose(from, outside, comber.getOutsideLocToLine(train)); + std::vector> crossing_out_candidates = PolygonUtils::findClose(from, outside, comber.getOutsideLocToLine(train)); bool seen_close_enough_connection = false; - for (std::pair& crossing_candidate : crossing_out_candidates) + for (std::pair& crossing_candidate : crossing_out_candidates) { const coord_t crossing_dist2 = vSize2(crossing_candidate.first.location_ - crossing_candidate.second.location_); if (crossing_dist2 > comber.max_crossing_dist2_ * 2) @@ -603,7 +603,7 @@ std::shared_ptr> Comb::Cross } if (best_detour_score == std::numeric_limits::max()) { // i.e. if best_in == nullptr or if best_out == nullptr - return std::shared_ptr>(); + return std::shared_ptr>(); } if (best_crossing_dist2 > comber.max_crossing_dist2_) { // find closer point on line segments, rather than moving between vertices of the polygons only @@ -611,10 +611,10 @@ std::shared_ptr> Comb::Cross best_crossing_dist2 = vSize2(best_in->location_ - best_out->location_); if (best_crossing_dist2 > comber.max_crossing_dist2_) { - return std::shared_ptr>(); + return std::shared_ptr>(); } } - return std::make_shared>(*best_in, *best_out); + return std::make_shared>(*best_in, *best_out); } } // namespace cura diff --git a/src/pathPlanning/LinePolygonsCrossings.cpp b/src/pathPlanning/LinePolygonsCrossings.cpp index 1391dffb7a..a2867386ea 100644 --- a/src/pathPlanning/LinePolygonsCrossings.cpp +++ b/src/pathPlanning/LinePolygonsCrossings.cpp @@ -22,7 +22,7 @@ bool LinePolygonsCrossings::calcScanlineCrossings(bool fail_on_unavoidable_obsta { for (unsigned int poly_idx = 0; poly_idx < boundary_.size(); poly_idx++) { - ConstPolygonRef poly = boundary_[poly_idx]; + const Polygon& poly = boundary_[poly_idx]; Point2LL p0 = transformation_matrix_.apply(poly[poly.size() - 1]); for (unsigned int point_idx = 0; point_idx < poly.size(); point_idx++) { @@ -74,7 +74,7 @@ bool LinePolygonsCrossings::lineSegmentCollidesWithBoundary() transformed_start_point_ = transformation_matrix_.apply(start_point_); transformed_end_point_ = transformation_matrix_.apply(end_point_); - for (ConstPolygonRef poly : boundary_) + for (const Polygon& poly : boundary_) { Point2LL p0 = transformation_matrix_.apply(poly.back()); for (Point2LL p1_ : poly) @@ -152,7 +152,7 @@ void LinePolygonsCrossings::generateBasicCombingPath(const Crossing& min, const { // minimise the path length by measuring the length of both paths around the polygon so we can determine the shorter path - ConstPolygonRef poly = boundary_[min.poly_idx_]; + const Polygon& poly = boundary_[min.poly_idx_]; combPath.push_back(transformation_matrix_.unapply(Point2LL(min.x_ - std::abs(dist_to_move_boundary_point_outside_), transformed_start_point_.Y))); // follow the path in the same direction as the winding order of the boundary polygon diff --git a/src/path_ordering.cpp b/src/path_ordering.cpp new file mode 100644 index 0000000000..8d7cd91ad1 --- /dev/null +++ b/src/path_ordering.cpp @@ -0,0 +1,53 @@ +// Copyright (c) 2023 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "path_ordering.h" //The definitions we're implementing here. + +#include "WallToolPaths.h" +#include "geometry/OpenPolyline.h" +#include "sliceDataStorage.h" //For SliceLayerPart. + +namespace cura +{ + +template +const PointsSet& PathOrdering::getVertexData() +{ + return *vertices_; +} + +template<> +const PointsSet& PathOrdering::getVertexData() +{ + return vertices_->outline.outerPolygon(); +} + +template<> +const PointsSet& PathOrdering::getVertexData() +{ + return vertices_->outline.outerPolygon(); +} + +template<> +const PointsSet& PathOrdering::getVertexData() +{ + return vertices_->outline_.outerPolygon(); +} +template<> +const PointsSet& PathOrdering::getVertexData() +{ + if (! cached_vertices_) + { + cached_vertices_ = vertices_->toPolygon(); + } + return *cached_vertices_; +} + +template const PointsSet& PathOrdering::getVertexData(); +template const PointsSet& PathOrdering::getVertexData(); +template const PointsSet& PathOrdering::getVertexData(); +template const PointsSet& PathOrdering::getVertexData(); +template const PointsSet& PathOrdering::getVertexData(); +template const PointsSet& PathOrdering::getVertexData(); + +} // namespace cura diff --git a/src/plugins/converters.cpp b/src/plugins/converters.cpp index fe6cbe9c0b..f53d2cbc7c 100644 --- a/src/plugins/converters.cpp +++ b/src/plugins/converters.cpp @@ -11,11 +11,12 @@ #include "GCodePathConfig.h" #include "WallToolPaths.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" #include "pathPlanning/GCodePath.h" #include "pathPlanning/SpeedDerivatives.h" #include "settings/Settings.h" #include "settings/types/LayerIndex.h" -#include "utils/polygon.h" namespace cura::plugins { @@ -102,7 +103,7 @@ simplify_request::value_type simplify_request::operator()( auto* msg_polygon = msg_polygons->add_polygons(); auto* msg_outline = msg_polygon->mutable_outline(); - for (const auto& point : ranges::front(polygons.paths)) + for (const auto& point : ranges::front(polygons)) { auto* msg_outline_path = msg_outline->add_path(); msg_outline_path->set_x(point.X); @@ -110,7 +111,7 @@ simplify_request::value_type simplify_request::operator()( } auto* msg_holes = msg_polygon->mutable_holes(); - for (const auto& polygon : polygons.paths | ranges::views::drop(1)) + for (const auto& polygon : polygons | ranges::views::drop(1)) { auto* msg_hole = msg_holes->Add(); for (const auto& point : polygon) @@ -136,18 +137,18 @@ simplify_response::native_value_type Polygon o{}; for (const auto& point : paths.outline().path()) { - o.add(Point2LL{ point.x(), point.y() }); + o.push_back(Point2LL{ point.x(), point.y() }); } - poly.add(o); + poly.push_back(o); for (const auto& hole : paths.holes()) { Polygon h{}; for (const auto& point : hole.path()) { - h.add(Point2LL{ point.x(), point.y() }); + h.push_back(Point2LL{ point.x(), point.y() }); } - poly.add(h); + poly.push_back(h); } } return poly; @@ -186,7 +187,7 @@ infill_generate_request::value_type auto* msg_polygon = msg_polygons->add_polygons(); auto* msg_outline = msg_polygon->mutable_outline(); - for (const auto& point : ranges::front(inner_contour.paths)) + for (const auto& point : ranges::front(inner_contour)) { auto* msg_outline_path = msg_outline->add_path(); msg_outline_path->set_x(point.X); @@ -194,7 +195,7 @@ infill_generate_request::value_type } auto* msg_holes = msg_polygon->mutable_holes(); - for (const auto& polygon : inner_contour.paths | ranges::views::drop(1)) + for (const auto& polygon : inner_contour | ranges::views::drop(1)) { auto* msg_hole = msg_holes->Add(); for (const auto& point : polygon) @@ -211,8 +212,8 @@ infill_generate_request::value_type infill_generate_response::native_value_type infill_generate_response::operator()(const infill_generate_response::value_type& message) const { VariableWidthLines toolpaths; - Polygons result_polygons; - Polygons result_lines; + Shape result_polygons; + OpenLinesSet result_lines; for (auto& tool_path : message.tool_paths().tool_paths()) { @@ -232,14 +233,14 @@ infill_generate_response::native_value_type infill_generate_response::operator() for (auto& polygon_msg : message.polygons().polygons()) { - Polygons polygon{}; + Shape polygon{}; Polygon outline{}; for (auto& path_msg : polygon_msg.outline().path()) { - outline.add(Point2LL{ path_msg.x(), path_msg.y() }); + outline.push_back(Point2LL{ path_msg.x(), path_msg.y() }); } - polygon.add(outline); + polygon.push_back(outline); for (auto& hole_msg : polygon_msg.holes()) @@ -247,20 +248,20 @@ infill_generate_response::native_value_type infill_generate_response::operator() Polygon hole{}; for (auto& path_msg : hole_msg.path()) { - hole.add(Point2LL{ path_msg.x(), path_msg.y() }); + hole.emplace_back(path_msg.x(), path_msg.y()); } - polygon.add(hole); + polygon.push_back(hole); } - result_polygons.add(polygon); + result_polygons.push_back(polygon); } for (auto& polygon : message.poly_lines().paths()) { - Polygon poly_line; + OpenPolyline poly_line; for (auto& p : polygon.path()) { - poly_line.emplace_back(Point2LL{ p.x(), p.y() }); + poly_line.emplace_back(p.x(), p.y()); } result_lines.emplace_back(poly_line); } @@ -486,4 +487,4 @@ gcode_paths_modify_response::native_value_type } } // namespace cura::plugins -#endif // ENABLE_PLUGINS \ No newline at end of file +#endif // ENABLE_PLUGINS diff --git a/src/raft.cpp b/src/raft.cpp index 4b27f0911f..8bc54ec41e 100644 --- a/src/raft.cpp +++ b/src/raft.cpp @@ -20,7 +20,7 @@ namespace cura void Raft::generate(SliceDataStorage& storage) { assert( - storage.raftBaseOutline.size() == 0 && storage.raftInterfaceOutline.size() == 0 && storage.raftSurfaceOutline.size() == 0 + storage.raft_base_outline.size() == 0 && storage.raft_interface_outline.size() == 0 && storage.raft_surface_outline.size() == 0 && "Raft polygon isn't generated yet, so should be empty!"); const Settings& settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings.get("raft_base_extruder_nr").settings_; constexpr bool include_support = true; @@ -29,35 +29,35 @@ void Raft::generate(SliceDataStorage& storage) const auto raft_interface_margin = settings.get("raft_interface_margin"); const auto raft_surface_margin = settings.get("raft_surface_margin"); - storage.raftBaseOutline = storage.raftSurfaceOutline = storage.raftInterfaceOutline = storage.getLayerOutlines(0, include_support, dont_include_prime_tower); - storage.raftBaseOutline = storage.raftBaseOutline.offset(raft_base_margin, ClipperLib::jtRound); - storage.raftInterfaceOutline = storage.raftInterfaceOutline.offset(raft_interface_margin, ClipperLib::jtRound); - storage.raftSurfaceOutline = storage.raftSurfaceOutline.offset(raft_surface_margin, ClipperLib::jtRound); + storage.raft_base_outline = storage.raft_surface_outline = storage.raft_interface_outline = storage.getLayerOutlines(0, include_support, dont_include_prime_tower); + storage.raft_base_outline = storage.raft_base_outline.offset(raft_base_margin, ClipperLib::jtRound); + storage.raft_interface_outline = storage.raft_interface_outline.offset(raft_interface_margin, ClipperLib::jtRound); + storage.raft_surface_outline = storage.raft_surface_outline.offset(raft_surface_margin, ClipperLib::jtRound); const coord_t shield_line_width_layer0 = settings.get("skirt_brim_line_width"); const coord_t max_raft_distance = std::max(std::max(raft_base_margin, raft_interface_margin), raft_surface_margin); if (storage.draft_protection_shield.size() > 0) { - Polygons draft_shield_raft + Shape draft_shield_raft = storage.draft_protection_shield .offset(shield_line_width_layer0) // start half a line width outside shield .difference(storage.draft_protection_shield.offset(-max_raft_distance - shield_line_width_layer0 / 2, ClipperLib::jtRound)); // end distance inside shield - storage.raftBaseOutline = storage.raftBaseOutline.unionPolygons(draft_shield_raft); - storage.raftSurfaceOutline = storage.raftSurfaceOutline.unionPolygons(draft_shield_raft); - storage.raftInterfaceOutline = storage.raftInterfaceOutline.unionPolygons(draft_shield_raft); + storage.raft_base_outline = storage.raft_base_outline.unionPolygons(draft_shield_raft); + storage.raft_surface_outline = storage.raft_surface_outline.unionPolygons(draft_shield_raft); + storage.raft_interface_outline = storage.raft_interface_outline.unionPolygons(draft_shield_raft); } - if (storage.oozeShield.size() > 0 && storage.oozeShield[0].size() > 0) + if (storage.ooze_shield.size() > 0 && storage.ooze_shield[0].size() > 0) { - const Polygons& ooze_shield = storage.oozeShield[0]; - Polygons ooze_shield_raft = ooze_shield - .offset(shield_line_width_layer0) // start half a line width outside shield - .difference(ooze_shield.offset(-max_raft_distance - shield_line_width_layer0 / 2, ClipperLib::jtRound)); // end distance inside shield - storage.raftBaseOutline = storage.raftBaseOutline.unionPolygons(ooze_shield_raft); - storage.raftSurfaceOutline = storage.raftSurfaceOutline.unionPolygons(ooze_shield_raft); - storage.raftInterfaceOutline = storage.raftInterfaceOutline.unionPolygons(ooze_shield_raft); + const Shape& ooze_shield = storage.ooze_shield[0]; + Shape ooze_shield_raft = ooze_shield + .offset(shield_line_width_layer0) // start half a line width outside shield + .difference(ooze_shield.offset(-max_raft_distance - shield_line_width_layer0 / 2, ClipperLib::jtRound)); // end distance inside shield + storage.raft_base_outline = storage.raft_base_outline.unionPolygons(ooze_shield_raft); + storage.raft_surface_outline = storage.raft_surface_outline.unionPolygons(ooze_shield_raft); + storage.raft_interface_outline = storage.raft_interface_outline.unionPolygons(ooze_shield_raft); } - const auto remove_inside_corners = [](Polygons& outline, bool remove_inside_corners, coord_t smoothing, coord_t line_width) + const auto remove_inside_corners = [](Shape& outline, bool remove_inside_corners, coord_t smoothing, coord_t line_width) { if (remove_inside_corners) { @@ -80,7 +80,7 @@ void Raft::generate(SliceDataStorage& storage) for (auto& part : outline_parts) { part.makeConvex(); - outline.add(part); + outline.push_back(part); } outline = outline.unionPolygons(); @@ -114,14 +114,14 @@ void Raft::generate(SliceDataStorage& storage) } }; const auto nominal_raft_line_width = settings.get("skirt_brim_line_width"); - remove_inside_corners(storage.raftBaseOutline, settings.get("raft_base_remove_inside_corners"), settings.get("raft_base_smoothing"), nominal_raft_line_width); + remove_inside_corners(storage.raft_base_outline, settings.get("raft_base_remove_inside_corners"), settings.get("raft_base_smoothing"), nominal_raft_line_width); remove_inside_corners( - storage.raftInterfaceOutline, + storage.raft_interface_outline, settings.get("raft_interface_remove_inside_corners"), settings.get("raft_interface_smoothing"), nominal_raft_line_width); remove_inside_corners( - storage.raftSurfaceOutline, + storage.raft_surface_outline, settings.get("raft_surface_remove_inside_corners"), settings.get("raft_surface_smoothing"), nominal_raft_line_width); diff --git a/src/settings/AdaptiveLayerHeights.cpp b/src/settings/AdaptiveLayerHeights.cpp index 14b37b8eaa..14e5e743c0 100644 --- a/src/settings/AdaptiveLayerHeights.cpp +++ b/src/settings/AdaptiveLayerHeights.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "settings/AdaptiveLayerHeights.h" @@ -6,6 +6,7 @@ #include #include #include +#include #include "Application.h" #include "Slice.h" diff --git a/src/settings/Settings.cpp b/src/settings/Settings.cpp index d946ec4496..136606cde2 100644 --- a/src/settings/Settings.cpp +++ b/src/settings/Settings.cpp @@ -1,13 +1,14 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "settings/Settings.h" #include +#include #include +#include #include // regex parsing for temp flow graph #include // ostringstream -#include #include //Parsing strings (stod, stoul). #include @@ -18,6 +19,8 @@ #include "BeadingStrategy/BeadingStrategyFactory.h" #include "ExtruderTrain.h" #include "Slice.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" #include "settings/EnumSettings.h" #include "settings/FlowTempGraph.h" #include "settings/types/Angle.h" @@ -27,7 +30,6 @@ #include "settings/types/Temperature.h" //For temperature settings. #include "settings/types/Velocity.h" //For velocity settings. #include "utils/Matrix4x3D.h" -#include "utils/polygon.h" #include "utils/string.h" //For Escaped. #include "utils/types/string_switch.h" //For string switch. @@ -257,11 +259,11 @@ FlowTempGraph Settings::get(const std::string& key) const } template<> -Polygons Settings::get(const std::string& key) const +Shape Settings::get(const std::string& key) const { std::string value_string = get(key); - Polygons result; + Shape result; if (value_string.empty()) { return result; // Empty at this point. @@ -287,8 +289,7 @@ Polygons Settings::get(const std::string& key) const { std::string polygon_str = *polygon_match_iter++; - result.emplace_back(); - PolygonRef poly = result.back(); + Polygon& poly = result.newLine(); std::regex point2D_regex(R"(\[([^,\[]*),([^,\]]*)\])"); // matches to a list of exactly two things diff --git a/src/skin.cpp b/src/skin.cpp index 842a6e576f..c0dcadd3d8 100644 --- a/src/skin.cpp +++ b/src/skin.cpp @@ -14,6 +14,7 @@ #include "settings/types/Angle.h" //For the infill support angle. #include "settings/types/Ratio.h" #include "sliceDataStorage.h" +#include "utils/Simplify.h" #include "utils/math.h" #include "utils/polygonUtils.h" @@ -55,9 +56,9 @@ SkinInfillAreaComputation::SkinInfillAreaComputation(const LayerIndex& layer_nr, * * this function may only read/write the skin and infill from the *current* layer. */ -Polygons SkinInfillAreaComputation::getOutlineOnLayer(const SliceLayerPart& part_here, const LayerIndex layer2_nr) +Shape SkinInfillAreaComputation::getOutlineOnLayer(const SliceLayerPart& part_here, const LayerIndex layer2_nr) { - Polygons result; + Shape result; if (layer2_nr >= static_cast(mesh_.layers.size())) { return result; @@ -67,7 +68,7 @@ Polygons SkinInfillAreaComputation::getOutlineOnLayer(const SliceLayerPart& part { if (part_here.boundaryBox.hit(part2.boundaryBox)) { - result.add(part2.outline); + result.push_back(part2.outline); } } return result; @@ -126,15 +127,15 @@ void SkinInfillAreaComputation::generateSkinAndInfillAreas() void SkinInfillAreaComputation::generateSkinAndInfillAreas(SliceLayerPart& part) { // Make a copy of the outline which we later intersect and union with the resized skins to ensure the resized skin isn't too large or removed completely. - Polygons top_skin; + Shape top_skin; if (top_layer_count_ > 0) { - top_skin = Polygons(part.inner_area); + top_skin = Shape(part.inner_area); } - Polygons bottom_skin; + Shape bottom_skin; if (bottom_layer_count_ > 0 || layer_nr_ < LayerIndex(initial_bottom_layer_count_)) { - bottom_skin = Polygons(part.inner_area); + bottom_skin = Shape(part.inner_area); } calculateBottomSkin(part, bottom_skin); @@ -143,7 +144,7 @@ void SkinInfillAreaComputation::generateSkinAndInfillAreas(SliceLayerPart& part) applySkinExpansion(part.inner_area, top_skin, bottom_skin); // Now combine the resized top skin and bottom skin. - Polygons skin = top_skin.unionPolygons(bottom_skin); + Shape skin = top_skin.unionPolygons(bottom_skin); skin.removeSmallAreas(MIN_AREA_SIZE); // Create infill area irrespective if the infill is to be generated or not(would be used for bridging). @@ -154,7 +155,7 @@ void SkinInfillAreaComputation::generateSkinAndInfillAreas(SliceLayerPart& part) generateInfill(part); } - for (const PolygonsPart& skin_area_part : skin.splitIntoParts()) + for (const SingleShape& skin_area_part : skin.splitIntoParts()) { if (skin_area_part.empty()) { @@ -171,7 +172,7 @@ void SkinInfillAreaComputation::generateSkinAndInfillAreas(SliceLayerPart& part) * * this function may only read/write the skin and infill from the *current* layer. */ -void SkinInfillAreaComputation::calculateBottomSkin(const SliceLayerPart& part, Polygons& downskin) +void SkinInfillAreaComputation::calculateBottomSkin(const SliceLayerPart& part, Shape& downskin) { if (bottom_layer_count_ == 0 && initial_bottom_layer_count_ == 0) { @@ -182,7 +183,7 @@ void SkinInfillAreaComputation::calculateBottomSkin(const SliceLayerPart& part, return; // don't subtract anything form the downskin } LayerIndex bottom_check_start_layer_idx{ std::max(LayerIndex{ 0 }, LayerIndex{ layer_nr_ - bottom_layer_count_ }) }; - Polygons not_air = getOutlineOnLayer(part, bottom_check_start_layer_idx); + Shape not_air = getOutlineOnLayer(part, bottom_check_start_layer_idx); if (! no_small_gaps_heuristic_) { for (int downskin_layer_nr = bottom_check_start_layer_idx + 1; downskin_layer_nr < layer_nr_; downskin_layer_nr++) @@ -198,7 +199,7 @@ void SkinInfillAreaComputation::calculateBottomSkin(const SliceLayerPart& part, downskin = downskin.difference(not_air); // skin overlaps with the walls } -void SkinInfillAreaComputation::calculateTopSkin(const SliceLayerPart& part, Polygons& upskin) +void SkinInfillAreaComputation::calculateTopSkin(const SliceLayerPart& part, Shape& upskin) { if (layer_nr_ > LayerIndex(mesh_.layers.size()) - top_layer_count_ || top_layer_count_ <= 0) { @@ -207,7 +208,7 @@ void SkinInfillAreaComputation::calculateTopSkin(const SliceLayerPart& part, Pol return; } - Polygons not_air = getOutlineOnLayer(part, layer_nr_ + top_layer_count_); + Shape not_air = getOutlineOnLayer(part, layer_nr_ + top_layer_count_); if (! no_small_gaps_heuristic_) { for (int upskin_layer_nr = layer_nr_ + 1; upskin_layer_nr < layer_nr_ + top_layer_count_; upskin_layer_nr++) @@ -231,7 +232,7 @@ void SkinInfillAreaComputation::calculateTopSkin(const SliceLayerPart& part, Pol * * this function may only read/write the skin and infill from the *current* layer. */ -void SkinInfillAreaComputation::applySkinExpansion(const Polygons& original_outline, Polygons& upskin, Polygons& downskin) +void SkinInfillAreaComputation::applySkinExpansion(const Shape& original_outline, Shape& upskin, Shape& downskin) { const coord_t min_width = mesh_.settings.get("min_skin_width_for_expansion") / 2; @@ -242,13 +243,13 @@ void SkinInfillAreaComputation::applySkinExpansion(const Polygons& original_outl // The expansion is only applied to that opened shape. if (bottom_skin_expand_distance_ != 0) { - const Polygons expanded = downskin.offset(-min_width).offset(min_width + bottom_skin_expand_distance_); + const Shape expanded = downskin.offset(-min_width).offset(min_width + bottom_skin_expand_distance_); // And then re-joined with the original part that was not offset, to retain parts smaller than min_width. downskin = downskin.unionPolygons(expanded); } if (top_skin_expand_distance_ != 0) { - const Polygons expanded = upskin.offset(-min_width).offset(min_width + top_skin_expand_distance_); + const Shape expanded = upskin.offset(-min_width).offset(min_width + top_skin_expand_distance_); upskin = upskin.unionPolygons(expanded); } } @@ -330,7 +331,7 @@ void SkinInfillAreaComputation::generateRoofingFillAndSkinFill(SliceLayerPart& p const size_t roofing_layer_count = std::min(mesh_.settings.get("roofing_layer_count"), mesh_.settings.get("top_layers")); const coord_t skin_overlap = mesh_.settings.get("skin_overlap_mm"); - Polygons filled_area_above = generateFilledAreaAbove(part, roofing_layer_count); + Shape filled_area_above = generateFilledAreaAbove(part, roofing_layer_count); skin_part.roofing_fill = skin_part.outline.difference(filled_area_above); skin_part.skin_fill = skin_part.outline.intersection(filled_area_above); @@ -348,14 +349,14 @@ void SkinInfillAreaComputation::generateRoofingFillAndSkinFill(SliceLayerPart& p * * this function may only read the skin and infill from the *current* layer. */ -Polygons SkinInfillAreaComputation::generateFilledAreaAbove(SliceLayerPart& part, size_t roofing_layer_count) +Shape SkinInfillAreaComputation::generateFilledAreaAbove(SliceLayerPart& part, size_t roofing_layer_count) { - Polygons filled_area_above = getOutlineOnLayer(part, layer_nr_ + roofing_layer_count); + Shape filled_area_above = getOutlineOnLayer(part, layer_nr_ + roofing_layer_count); if (! no_small_gaps_heuristic_) { for (int layer_nr_above = layer_nr_ + 1; layer_nr_above < layer_nr_ + roofing_layer_count; layer_nr_above++) { - Polygons outlines_above = getOutlineOnLayer(part, layer_nr_above); + Shape outlines_above = getOutlineOnLayer(part, layer_nr_above); filled_area_above = filled_area_above.intersection(outlines_above); } } @@ -367,7 +368,7 @@ Polygons SkinInfillAreaComputation::generateFilledAreaAbove(SliceLayerPart& part // has air below (fixes https://github.com/Ultimaker/Cura/issues/2656) // set air_below to the skin area for the current layer that has air below it - Polygons air_below = getOutlineOnLayer(part, layer_nr_).difference(getOutlineOnLayer(part, layer_nr_ - 1)); + Shape air_below = getOutlineOnLayer(part, layer_nr_).difference(getOutlineOnLayer(part, layer_nr_ - 1)); if (! air_below.empty()) { @@ -384,21 +385,21 @@ Polygons SkinInfillAreaComputation::generateFilledAreaAbove(SliceLayerPart& part * * this function may only read the skin and infill from the *current* layer. */ -Polygons SkinInfillAreaComputation::generateFilledAreaBelow(SliceLayerPart& part, size_t flooring_layer_count) +Shape SkinInfillAreaComputation::generateFilledAreaBelow(SliceLayerPart& part, size_t flooring_layer_count) { if (layer_nr_ < flooring_layer_count) { return {}; } const int lowest_flooring_layer = layer_nr_ - flooring_layer_count; - Polygons filled_area_below = getOutlineOnLayer(part, lowest_flooring_layer); + Shape filled_area_below = getOutlineOnLayer(part, lowest_flooring_layer); if (! no_small_gaps_heuristic_) { const int next_lowest_flooring_layer = lowest_flooring_layer + 1; for (int layer_nr_below = next_lowest_flooring_layer; layer_nr_below < layer_nr_; layer_nr_below++) { - Polygons outlines_below = getOutlineOnLayer(part, layer_nr_below); + Shape outlines_below = getOutlineOnLayer(part, layer_nr_below); filled_area_below = filled_area_below.intersection(outlines_below); } } @@ -417,27 +418,27 @@ void SkinInfillAreaComputation::generateInfillSupport(SliceMeshStorage& mesh) SliceLayer& layer = mesh.layers[layer_idx]; SliceLayer& layer_above = mesh.layers[layer_idx + 1]; - Polygons inside_above; - Polygons infill_above; + Shape inside_above; + Shape infill_above; for (SliceLayerPart& part_above : layer_above.parts) { - inside_above.add(part_above.infill_area); - infill_above.add(part_above.getOwnInfillArea()); + inside_above.push_back(part_above.infill_area); + infill_above.push_back(part_above.getOwnInfillArea()); } for (SliceLayerPart& part : layer.parts) { - const Polygons& infill_area = part.infill_area; + const Shape& infill_area = part.infill_area; if (infill_area.empty()) { continue; } - const Polygons unsupported = infill_area.offset(-max_dist_from_lower_layer); - const Polygons basic_overhang = unsupported.difference(inside_above); - const Polygons overhang_extented = basic_overhang.offset(max_dist_from_lower_layer + 50); // +50 for easier joining with support from layer above - const Polygons full_overhang = overhang_extented.difference(inside_above); - const Polygons infill_support = infill_above.unionPolygons(full_overhang); + const Shape unsupported = infill_area.offset(-max_dist_from_lower_layer); + const Shape basic_overhang = unsupported.difference(inside_above); + const Shape overhang_extented = basic_overhang.offset(max_dist_from_lower_layer + 50); // +50 for easier joining with support from layer above + const Shape full_overhang = overhang_extented.difference(inside_above); + const Shape infill_support = infill_above.unionPolygons(full_overhang); part.infill_area_own = infill_support.intersection(part.getOwnInfillArea()); } @@ -465,9 +466,12 @@ void SkinInfillAreaComputation::generateGradualInfill(SliceMeshStorage& mesh) const LayerIndex mesh_min_layer = mesh.settings.get("initial_bottom_layers"); const LayerIndex mesh_max_layer = mesh.layers.size() - 1 - mesh.settings.get("top_layers"); + const Simplify simplifier(mesh.settings.get("infill_extruder_nr").settings_); + const auto infill_wall_count = mesh.settings.get("infill_wall_line_count"); const auto infill_wall_width = mesh.settings.get("infill_line_width"); const auto infill_overlap = mesh.settings.get("infill_overlap_mm"); + const auto is_connected = mesh.settings.get("zig_zaggify_infill") || mesh.settings.get("infill_pattern") == EFillMethod::ZIG_ZAG; for (LayerIndex layer_idx = 0; layer_idx < static_cast(mesh.layers.size()); layer_idx++) { // loop also over layers which don't contain infill cause of bottom_ and top_layer to initialize their infill_area_per_combine_per_density SliceLayer& layer = mesh.layers[layer_idx]; @@ -476,7 +480,7 @@ void SkinInfillAreaComputation::generateGradualInfill(SliceMeshStorage& mesh) { assert((part.infill_area_per_combine_per_density.empty() && "infill_area_per_combine_per_density is supposed to be uninitialized")); - const Polygons& infill_area = Infill::generateWallToolPaths( + const Shape& infill_area = Infill::generateWallToolPaths( part.infill_wall_toolpaths, part.getOwnInfillArea(), infill_wall_count, @@ -493,7 +497,8 @@ void SkinInfillAreaComputation::generateGradualInfill(SliceMeshStorage& mesh) // note: no need to copy part.infill_area, cause it's the empty vector anyway continue; } - Polygons less_dense_infill = infill_area; // one step less dense with each infill_step + Shape less_dense_infill = infill_area; // one step less dense with each infill_step + Shape sum_more_dense; // NOTE: Only used for zig-zag or connected fills. for (size_t infill_step = 0; infill_step < max_infill_steps; infill_step++) { LayerIndex min_layer = layer_idx + infill_step * gradual_infill_step_layer_count + static_cast(layer_skip_count); @@ -507,14 +512,14 @@ void SkinInfillAreaComputation::generateGradualInfill(SliceMeshStorage& mesh) break; } const SliceLayer& upper_layer = mesh.layers[static_cast(upper_layer_idx)]; - Polygons relevent_upper_polygons; + Shape relevent_upper_polygons; for (const SliceLayerPart& upper_layer_part : upper_layer.parts) { if (! upper_layer_part.boundaryBox.hit(part.boundaryBox)) { continue; } - relevent_upper_polygons.add(upper_layer_part.getOwnInfillArea()); + relevent_upper_polygons.push_back(upper_layer_part.getOwnInfillArea()); } less_dense_infill = less_dense_infill.intersection(relevent_upper_polygons); } @@ -524,13 +529,18 @@ void SkinInfillAreaComputation::generateGradualInfill(SliceMeshStorage& mesh) } // add new infill_area_per_combine for the current density part.infill_area_per_combine_per_density.emplace_back(); - std::vector& infill_area_per_combine_current_density = part.infill_area_per_combine_per_density.back(); - const Polygons more_dense_infill = infill_area.difference(less_dense_infill); - infill_area_per_combine_current_density.push_back(more_dense_infill); + std::vector& infill_area_per_combine_current_density = part.infill_area_per_combine_per_density.back(); + const Shape more_dense_infill = infill_area.difference(less_dense_infill); + infill_area_per_combine_current_density.push_back( + simplifier.polygon(more_dense_infill.difference(sum_more_dense).offset(-infill_wall_width).offset(infill_wall_width))); + if (is_connected) + { + sum_more_dense = sum_more_dense.unionPolygons(more_dense_infill); + } } part.infill_area_per_combine_per_density.emplace_back(); - std::vector& infill_area_per_combine_current_density = part.infill_area_per_combine_per_density.back(); - infill_area_per_combine_current_density.push_back(infill_area); + std::vector& infill_area_per_combine_current_density = part.infill_area_per_combine_per_density.back(); + infill_area_per_combine_current_density.push_back(simplifier.polygon(infill_area.difference(sum_more_dense).offset(-infill_wall_width).offset(infill_wall_width))); part.infill_area_own = std::nullopt; // clear infill_area_own, it's not needed any more. assert(! part.infill_area_per_combine_per_density.empty() && "infill_area_per_combine_per_density is now initialized"); } @@ -585,14 +595,14 @@ void SkinInfillAreaComputation::combineInfillLayers(SliceMeshStorage& mesh) { for (unsigned int density_idx = 0; density_idx < part.infill_area_per_combine_per_density.size(); density_idx++) { // go over each density of gradual infill (these density areas overlap!) - std::vector& infill_area_per_combine = part.infill_area_per_combine_per_density[density_idx]; - Polygons result; + std::vector& infill_area_per_combine = part.infill_area_per_combine_per_density[density_idx]; + Shape result; for (SliceLayerPart& lower_layer_part : lower_layer->parts) { if (part.boundaryBox.hit(lower_layer_part.boundaryBox)) { - Polygons intersection = infill_area_per_combine[combine_count_here - 1].intersection(lower_layer_part.infill_area).offset(-200).offset(200); - result.add(intersection); // add area to be thickened + Shape intersection = infill_area_per_combine[combine_count_here - 1].intersection(lower_layer_part.infill_area).offset(-200).offset(200); + result.push_back(intersection); // add area to be thickened infill_area_per_combine[combine_count_here - 1] = infill_area_per_combine[combine_count_here - 1].difference(intersection); // remove thickened area from less thick layer here unsigned int max_lower_density_idx = density_idx; @@ -612,7 +622,7 @@ void SkinInfillAreaComputation::combineInfillLayers(SliceMeshStorage& mesh) lower_density_idx <= max_lower_density_idx && lower_density_idx < lower_layer_part.infill_area_per_combine_per_density.size(); lower_density_idx++) { - std::vector& lower_infill_area_per_combine = lower_layer_part.infill_area_per_combine_per_density[lower_density_idx]; + std::vector& lower_infill_area_per_combine = lower_layer_part.infill_area_per_combine_per_density[lower_density_idx]; lower_infill_area_per_combine[0] = lower_infill_area_per_combine[0].difference(intersection); // remove thickened area from lower (single thickness) layer } @@ -637,10 +647,10 @@ void SkinInfillAreaComputation::generateTopAndBottomMostSkinFill(SliceLayerPart& { for (SkinPart& skin_part : part.skin_parts) { - Polygons filled_area_above = generateFilledAreaAbove(part, 1); + Shape filled_area_above = generateFilledAreaAbove(part, 1); skin_part.top_most_surface_fill = skin_part.outline.difference(filled_area_above); - Polygons filled_area_below = generateFilledAreaBelow(part, 1); + Shape filled_area_below = generateFilledAreaBelow(part, 1); skin_part.bottom_most_surface_fill = skin_part.skin_fill.difference(filled_area_below); } } diff --git a/src/sliceDataStorage.cpp b/src/sliceDataStorage.cpp index b369dbb082..2447ce356f 100644 --- a/src/sliceDataStorage.cpp +++ b/src/sliceDataStorage.cpp @@ -1,14 +1,17 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "sliceDataStorage.h" +#include + #include #include "Application.h" //To get settings. #include "ExtruderTrain.h" #include "FffProcessor.h" //To create a mesh group with if none is provided. #include "Slice.h" +#include "geometry/OpenPolyline.h" #include "infill/DensityProvider.h" // for destructor #include "infill/LightningGenerator.h" #include "infill/SierpinskiFillProvider.h" @@ -16,7 +19,6 @@ #include "raft.h" #include "utils/math.h" //For PI. - namespace cura { @@ -32,12 +34,12 @@ SupportStorage::~SupportStorage() supportLayers.clear(); } -Polygons& SliceLayerPart::getOwnInfillArea() +Shape& SliceLayerPart::getOwnInfillArea() { - return const_cast(const_cast(this)->getOwnInfillArea()); + return const_cast(const_cast(this)->getOwnInfillArea()); } -const Polygons& SliceLayerPart::getOwnInfillArea() const +const Shape& SliceLayerPart::getOwnInfillArea() const { if (infill_area_own) { @@ -68,24 +70,24 @@ SliceLayer::~SliceLayer() { } -Polygons SliceLayer::getOutlines(bool external_polys_only) const +Shape SliceLayer::getOutlines(bool external_polys_only) const { - Polygons ret; + Shape ret; getOutlines(ret, external_polys_only); return ret; } -void SliceLayer::getOutlines(Polygons& result, bool external_polys_only) const +void SliceLayer::getOutlines(Shape& result, bool external_polys_only) const { for (const SliceLayerPart& part : parts) { if (external_polys_only) { - result.add(part.outline.outerPolygon()); + result.push_back(part.outline.outerPolygon()); } else { - result.add(part.print_outline); + result.push_back(part.print_outline); } } } @@ -172,7 +174,7 @@ bool SliceMeshStorage::getExtruderIsUsed(const size_t extruder_nr, const LayerIn } } if (settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL && settings.get("wall_0_extruder_nr").extruder_nr_ == extruder_nr - && layer.openPolyLines.size() > 0) + && layer.open_polylines.size() > 0) { return true; } @@ -266,7 +268,7 @@ SliceDataStorage::SliceDataStorage() machine_size.include(machine_max); } -Polygons SliceDataStorage::getLayerOutlines( +Shape SliceDataStorage::getLayerOutlines( const LayerIndex layer_nr, const bool include_support, const bool include_prime_tower, @@ -283,37 +285,37 @@ Polygons SliceDataStorage::getLayerOutlines( case Raft::LayerType::RaftInterface: case Raft::LayerType::RaftSurface: { - const Polygons* raftOutline; + const Shape* raftOutline; bool use_current_extruder_for_raft = extruder_nr == -1; switch (layer_type) { case Raft::LayerType::RaftBase: - raftOutline = &raftBaseOutline; + raftOutline = &raft_base_outline; use_current_extruder_for_raft |= extruder_nr == int(mesh_group_settings.get("raft_base_extruder_nr").extruder_nr_); break; case Raft::LayerType::RaftInterface: - raftOutline = &raftInterfaceOutline; + raftOutline = &raft_interface_outline; use_current_extruder_for_raft |= extruder_nr == int(mesh_group_settings.get("raft_interface_extruder_nr").extruder_nr_); break; case Raft::LayerType::RaftSurface: - raftOutline = &raftSurfaceOutline; + raftOutline = &raft_surface_outline; use_current_extruder_for_raft |= extruder_nr == int(mesh_group_settings.get("raft_surface_extruder_nr").extruder_nr_); break; default: assert(false && "unreachable due to outer switch statement"); - return Polygons(); + return Shape(); } if (include_support && use_current_extruder_for_raft) { if (external_polys_only) { - std::vector parts = raftOutline->splitIntoParts(); - Polygons result; - for (PolygonsPart& part : parts) + std::vector parts = raftOutline->splitIntoParts(); + Shape result; + for (SingleShape& part : parts) { - result.add(part.outerPolygon()); + result.push_back(part.outerPolygon()); } return result; } @@ -324,14 +326,14 @@ Polygons SliceDataStorage::getLayerOutlines( } else { - return Polygons(); + return Shape(); } break; } case Raft::LayerType::Airgap: case Raft::LayerType::Model: { - Polygons total; + Shape total; if (include_models && layer_nr >= 0) { for (const std::shared_ptr& mesh : meshes) @@ -345,7 +347,7 @@ Polygons SliceDataStorage::getLayerOutlines( layer.getOutlines(total, external_polys_only); if (mesh->settings.get("magic_mesh_surface_mode") != ESurfaceMode::NORMAL) { - total = total.unionPolygons(layer.openPolyLines.offsetPolyLine(MM2INT(0.1))); + total = total.unionPolygons(layer.open_polylines.offset(MM2INT(0.1))); } } } @@ -356,25 +358,21 @@ Polygons SliceDataStorage::getLayerOutlines( { for (const SupportInfillPart& support_infill_part : support_layer.support_infill_parts) { - total.add(support_infill_part.outline_); + total.push_back(support_infill_part.outline_); } - total.add(support_layer.support_bottom); - total.add(support_layer.support_roof); + total.push_back(support_layer.support_bottom); + total.push_back(support_layer.support_roof); } } - int prime_tower_outer_extruder_nr = primeTower.extruder_order_[0]; - if (include_prime_tower && (extruder_nr == -1 || extruder_nr == prime_tower_outer_extruder_nr)) + if (include_prime_tower && primeTower.enabled_ && (extruder_nr == -1 || (! primeTower.extruder_order_.empty() && extruder_nr == primeTower.extruder_order_[0]))) { - if (primeTower.enabled_) - { - total.add(primeTower.getOuterPoly(layer_nr)); - } + total.push_back(primeTower.getOuterPoly(layer_nr)); } return total; } default: assert(false && "unreachable as switch statement is exhaustive"); - return Polygons(); + return Shape(); } } @@ -568,13 +566,13 @@ bool SliceDataStorage::getExtruderPrimeBlobEnabled(const size_t extruder_nr) con return train.settings_.get("prime_blob_enable"); } -Polygons SliceDataStorage::getMachineBorder(int checking_extruder_nr) const +Shape SliceDataStorage::getMachineBorder(int checking_extruder_nr) const { const Settings& mesh_group_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; - Polygons border; + Shape border; border.emplace_back(); - PolygonRef outline = border.back(); + Polygon& outline = border.back(); switch (mesh_group_settings.get("machine_shape")) { case BuildPlateShape::ELLIPTIC: @@ -596,14 +594,14 @@ Polygons SliceDataStorage::getMachineBorder(int checking_extruder_nr) const break; } - Polygons disallowed_areas = mesh_group_settings.get("machine_disallowed_areas"); + Shape disallowed_areas = mesh_group_settings.get("machine_disallowed_areas"); disallowed_areas = disallowed_areas.unionPolygons(); // union overlapping disallowed areas // The disallowed areas are expressed in buildplate-centered coordinates, but the models // may be expressed in front-left-centered coordinantes, so in this case we need to translate them if (! mesh_group_settings.get("machine_center_is_zero")) { - for (PolygonRef poly : disallowed_areas) + for (Polygon& poly : disallowed_areas) { for (Point2LL& p : poly) { @@ -633,12 +631,12 @@ Polygons SliceDataStorage::getMachineBorder(int checking_extruder_nr) const } Point2LL translation(extruder_settings.get("machine_nozzle_offset_x"), extruder_settings.get("machine_nozzle_offset_y")); prime_pos -= translation; - Polygons prime_polygons; + Shape prime_polygons; prime_polygons.emplace_back(PolygonUtils::makeCircle(prime_pos, prime_clearance, std::numbers::pi / 32)); disallowed_areas = disallowed_areas.unionPolygons(prime_polygons); } - Polygons disallowed_all_extruders; + Shape disallowed_all_extruders; bool first = true; for (size_t extruder_nr = 0; extruder_nr < extruder_is_used.size(); extruder_nr++) { @@ -648,7 +646,7 @@ Polygons SliceDataStorage::getMachineBorder(int checking_extruder_nr) const } Settings& extruder_settings = Application::getInstance().current_slice_->scene.extruders[extruder_nr].settings_; Point2LL translation(extruder_settings.get("machine_nozzle_offset_x"), extruder_settings.get("machine_nozzle_offset_y")); - Polygons extruder_border = disallowed_areas; + Shape extruder_border = disallowed_areas; extruder_border.translate(translation); if (first) { @@ -660,9 +658,9 @@ Polygons SliceDataStorage::getMachineBorder(int checking_extruder_nr) const disallowed_all_extruders = disallowed_all_extruders.unionPolygons(extruder_border); } } - disallowed_all_extruders.processEvenOdd(ClipperLib::pftNonZero); // prevent overlapping disallowed areas from XORing + disallowed_all_extruders = disallowed_all_extruders.processEvenOdd(ClipperLib::pftNonZero); // prevent overlapping disallowed areas from XORing - Polygons border_all_extruders = border; // each extruders border areas must be limited to the global border, which is the union of all extruders borders + Shape border_all_extruders = border; // each extruders border areas must be limited to the global border, which is the union of all extruders borders if (mesh_group_settings.has("nozzle_offsetting_for_disallowed_areas") && mesh_group_settings.get("nozzle_offsetting_for_disallowed_areas")) { for (size_t extruder_nr = 0; extruder_nr < extruder_is_used.size(); extruder_nr++) @@ -682,7 +680,7 @@ Polygons SliceDataStorage::getMachineBorder(int checking_extruder_nr) const } Settings& other_extruder_settings = Application::getInstance().current_slice_->scene.extruders[other_extruder_nr].settings_; Point2LL other_translation(other_extruder_settings.get("machine_nozzle_offset_x"), other_extruder_settings.get("machine_nozzle_offset_y")); - Polygons translated_border = border; + Shape translated_border = border; translated_border.translate(translation - other_translation); border_all_extruders = border_all_extruders.intersection(translated_border); } @@ -694,7 +692,7 @@ Polygons SliceDataStorage::getMachineBorder(int checking_extruder_nr) const } -void SupportLayer::excludeAreasFromSupportInfillAreas(const Polygons& exclude_polygons, const AABB& exclude_polygons_boundary_box) +void SupportLayer::excludeAreasFromSupportInfillAreas(const Shape& exclude_polygons, const AABB& exclude_polygons_boundary_box) { // record the indexes that need to be removed and do that after std::list to_remove_part_indices; // LIFO for removing @@ -710,7 +708,7 @@ void SupportLayer::excludeAreasFromSupportInfillAreas(const Polygons& exclude_po continue; } - Polygons result_polygons = support_infill_part.outline_.difference(exclude_polygons); + Shape result_polygons = support_infill_part.outline_.difference(exclude_polygons); // if no smaller parts get generated, this mean this part should be removed. if (result_polygons.empty()) @@ -719,7 +717,7 @@ void SupportLayer::excludeAreasFromSupportInfillAreas(const Polygons& exclude_po continue; } - std::vector smaller_support_islands = result_polygons.splitIntoParts(); + std::vector smaller_support_islands = result_polygons.splitIntoParts(); if (smaller_support_islands.empty()) { // extra safety guard in case result_polygons consists of too small polygons which are automatically removed in splitIntoParts @@ -734,7 +732,7 @@ void SupportLayer::excludeAreasFromSupportInfillAreas(const Polygons& exclude_po for (size_t support_island_idx = 1; support_island_idx < smaller_support_islands.size(); ++support_island_idx) { - const PolygonsPart& smaller_island = smaller_support_islands[support_island_idx]; + const SingleShape& smaller_island = smaller_support_islands[support_island_idx]; support_infill_parts.emplace_back(smaller_island, support_infill_part.support_line_width_, support_infill_part.inset_count_to_generate_); } } @@ -758,7 +756,7 @@ void SupportLayer::excludeAreasFromSupportInfillAreas(const Polygons& exclude_po void SupportLayer::fillInfillParts( const LayerIndex layer_nr, - const std::vector& support_fill_per_layer, + const std::vector& support_fill_per_layer, const coord_t infill_layer_height, const std::vector>& meshes, const coord_t support_line_width, @@ -768,14 +766,14 @@ void SupportLayer::fillInfillParts( const coord_t custom_line_distance /*has default 0*/) { // Find the model exactly z-distance above the support layer. - Polygons overhang_z_dist_above; + Shape overhang_z_dist_above; for (const auto& mesh : meshes) { const coord_t mesh_z_distance_top = mesh->settings.get("support_top_distance"); const size_t overhang_layer_nr = layer_nr + (mesh_z_distance_top / infill_layer_height) + 1; if (overhang_layer_nr < mesh->overhang_areas.size()) { - overhang_z_dist_above.add(mesh->overhang_areas[overhang_layer_nr]); + overhang_z_dist_above.push_back(mesh->overhang_areas[overhang_layer_nr]); } } overhang_z_dist_above = overhang_z_dist_above.unionPolygons(); @@ -786,7 +784,7 @@ void SupportLayer::fillInfillParts( bool use_fractional_config = true; for (auto& support_areas : all_support_areas_in_layer) { - for (const PolygonsPart& island_outline : support_areas.splitIntoParts(unionAll)) + for (const SingleShape& island_outline : support_areas.splitIntoParts(unionAll)) { support_infill_parts.emplace_back(island_outline, support_line_width, use_fractional_config, wall_line_count, custom_line_distance); } diff --git a/src/slicer.cpp b/src/slicer.cpp index 5c7dcfe11e..f66e2d40ec 100644 --- a/src/slicer.cpp +++ b/src/slicer.cpp @@ -1,17 +1,19 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "slicer.h" #include // remove_if +#include #include -#include #include #include #include "Application.h" #include "Slice.h" +#include "geometry/OpenPolyline.h" +#include "geometry/SingleShape.h" // Needed in order to call splitIntoParts() #include "plugins/slots.h" #include "raft.h" #include "settings/AdaptiveLayerHeights.h" @@ -30,49 +32,49 @@ constexpr int largest_neglected_gap_first_phase = MM2INT(0.01); //!< distance be constexpr int largest_neglected_gap_second_phase = MM2INT(0.02); //!< distance between two line segments regarded as connected constexpr int max_stitch1 = MM2INT(10.0); //!< maximal distance stitched between open polylines to form polygons -void SlicerLayer::makeBasicPolygonLoops(Polygons& open_polylines) +void SlicerLayer::makeBasicPolygonLoops(OpenLinesSet& open_polylines) { - for (size_t start_segment_idx = 0; start_segment_idx < segments.size(); start_segment_idx++) + for (size_t start_segment_idx = 0; start_segment_idx < segments_.size(); start_segment_idx++) { - if (! segments[start_segment_idx].addedToPolygon) + if (! segments_[start_segment_idx].addedToPolygon) { makeBasicPolygonLoop(open_polylines, start_segment_idx); } } // Clear the segmentList to save memory, it is no longer needed after this point. - segments.clear(); + segments_.clear(); } -void SlicerLayer::makeBasicPolygonLoop(Polygons& open_polylines, const size_t start_segment_idx) +void SlicerLayer::makeBasicPolygonLoop(OpenLinesSet& open_polylines, const size_t start_segment_idx) { - Polygon poly; - poly.add(segments[start_segment_idx].start); + Polygon poly(true); + poly.push_back(segments_[start_segment_idx].start); for (int segment_idx = start_segment_idx; segment_idx != -1;) { - SlicerSegment& segment = segments[segment_idx]; - poly.add(segment.end); + SlicerSegment& segment = segments_[segment_idx]; + poly.push_back(segment.end); segment.addedToPolygon = true; segment_idx = getNextSegmentIdx(segment, start_segment_idx); if (segment_idx == static_cast(start_segment_idx)) { // polyon is closed - polygons.add(poly); + polygons_.push_back(std::move(poly)); return; } } // polygon couldn't be closed - open_polylines.add(poly); + open_polylines.emplace_back(std::move(poly.getPoints())); } int SlicerLayer::tryFaceNextSegmentIdx(const SlicerSegment& segment, const int face_idx, const size_t start_segment_idx) const { - decltype(face_idx_to_segment_idx.begin()) it; - auto it_end = face_idx_to_segment_idx.end(); - it = face_idx_to_segment_idx.find(face_idx); + decltype(face_idx_to_segment_idx_.begin()) it; + auto it_end = face_idx_to_segment_idx_.end(); + it = face_idx_to_segment_idx_.find(face_idx); if (it != it_end) { const int segment_idx = (*it).second; - Point2LL p1 = segments[segment_idx].start; + Point2LL p1 = segments_[segment_idx].start; Point2LL diff = segment.end - p1; if (shorterThen(diff, largest_neglected_gap_first_phase)) { @@ -80,7 +82,7 @@ int SlicerLayer::tryFaceNextSegmentIdx(const SlicerSegment& segment, const int f { return start_segment_idx; } - if (segments[segment_idx].addedToPolygon) + if (segments_[segment_idx].addedToPolygon) { return -1; } @@ -128,7 +130,7 @@ int SlicerLayer::getNextSegmentIdx(const SlicerSegment& segment, const size_t st return next_segment_idx; } -void SlicerLayer::connectOpenPolylines(Polygons& open_polylines) +void SlicerLayer::connectOpenPolylines(OpenLinesSet& open_polylines) { constexpr bool allow_reverse = false; // Search a bit fewer cells but at cost of covering more area. @@ -137,7 +139,7 @@ void SlicerLayer::connectOpenPolylines(Polygons& open_polylines) connectOpenPolylinesImpl(open_polylines, largest_neglected_gap_second_phase, cell_size, allow_reverse); } -void SlicerLayer::stitch(Polygons& open_polylines) +void SlicerLayer::stitch(OpenLinesSet& open_polylines) { bool allow_reverse = true; connectOpenPolylinesImpl(open_polylines, max_stitch1, max_stitch1, allow_reverse); @@ -189,7 +191,8 @@ bool SlicerLayer::PossibleStitch::operator<(const PossibleStitch& other) const return false; } -std::priority_queue SlicerLayer::findPossibleStitches(const Polygons& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse) const +std::priority_queue + SlicerLayer::findPossibleStitches(const OpenLinesSet& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse) const { std::priority_queue stitch_queue; @@ -224,7 +227,7 @@ std::priority_queue SlicerLayer::findPossibleStitch // insert the starts of the polylines). for (unsigned int polyline_0_idx = 0; polyline_0_idx < open_polylines.size(); polyline_0_idx++) { - ConstPolygonRef polyline_0 = open_polylines[polyline_0_idx]; + const OpenPolyline& polyline_0 = open_polylines[polyline_0_idx]; if (polyline_0.size() < 1) continue; @@ -240,7 +243,7 @@ std::priority_queue SlicerLayer::findPossibleStitch { for (unsigned int polyline_0_idx = 0; polyline_0_idx < open_polylines.size(); polyline_0_idx++) { - ConstPolygonRef polyline_0 = open_polylines[polyline_0_idx]; + const OpenPolyline& polyline_0 = open_polylines[polyline_0_idx]; if (polyline_0.size() < 1) continue; @@ -255,7 +258,7 @@ std::priority_queue SlicerLayer::findPossibleStitch // search for nearby end points for (unsigned int polyline_1_idx = 0; polyline_1_idx < open_polylines.size(); polyline_1_idx++) { - ConstPolygonRef polyline_1 = open_polylines[polyline_1_idx]; + const OpenPolyline& polyline_1 = open_polylines[polyline_1_idx]; if (polyline_1.size() < 1) continue; @@ -335,7 +338,7 @@ std::priority_queue SlicerLayer::findPossibleStitch return stitch_queue; } -void SlicerLayer::planPolylineStitch(const Polygons& open_polylines, Terminus& terminus_0, Terminus& terminus_1, bool reverse[2]) const +void SlicerLayer::planPolylineStitch(const OpenLinesSet& open_polylines, Terminus& terminus_0, Terminus& terminus_1, bool reverse[2]) const { size_t polyline_0_idx = terminus_0.getPolylineIdx(); size_t polyline_1_idx = terminus_1.getPolylineIdx(); @@ -384,7 +387,7 @@ void SlicerLayer::planPolylineStitch(const Polygons& open_polylines, Terminus& t } } -void SlicerLayer::joinPolylines(PolygonRef& polyline_0, PolygonRef& polyline_1, const bool reverse[2]) const +void SlicerLayer::joinPolylines(OpenPolyline& polyline_0, OpenPolyline& polyline_1, const bool reverse[2]) { if (reverse[0]) { @@ -399,13 +402,13 @@ void SlicerLayer::joinPolylines(PolygonRef& polyline_0, PolygonRef& polyline_1, { // reverse polyline_1 by adding in reverse order for (int poly_idx = polyline_1.size() - 1; poly_idx >= 0; poly_idx--) - polyline_0.add(polyline_1[poly_idx]); + polyline_0.push_back(polyline_1[poly_idx]); } else { // append polyline_1 onto polyline_0 for (Point2LL& p : polyline_1) - polyline_0.add(p); + polyline_0.push_back(p); } polyline_1.clear(); } @@ -451,7 +454,7 @@ void SlicerLayer::TerminusTrackingMap::updateMap( } } -void SlicerLayer::connectOpenPolylinesImpl(Polygons& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse) +void SlicerLayer::connectOpenPolylinesImpl(OpenLinesSet& open_polylines, coord_t max_dist, coord_t cell_size, bool allow_reverse) { // below code closes smallest gaps first @@ -491,9 +494,8 @@ void SlicerLayer::connectOpenPolylinesImpl(Polygons& open_polylines, coord_t max if (completed_poly) { // finished polygon - PolygonRef polyline_0 = open_polylines[best_polyline_0_idx]; - polygons.add(polyline_0); - polyline_0.clear(); + OpenPolyline& polyline_0 = open_polylines[best_polyline_0_idx]; + polygons_.push_back(Polygon(std::move(polyline_0.getPoints()), true)); // Will also clear the polyline Terminus cur_terms[2] = { { best_polyline_0_idx, false }, { best_polyline_0_idx, true } }; for (size_t idx = 0U; idx != 2U; ++idx) { @@ -511,8 +513,8 @@ void SlicerLayer::connectOpenPolylinesImpl(Polygons& open_polylines, coord_t max // need to reread since planPolylineStitch can swap terminus_0/1 best_polyline_0_idx = terminus_0.getPolylineIdx(); best_polyline_1_idx = terminus_1.getPolylineIdx(); - PolygonRef polyline_0 = open_polylines[best_polyline_0_idx]; - PolygonRef polyline_1 = open_polylines[best_polyline_1_idx]; + OpenPolyline& polyline_0 = open_polylines[best_polyline_0_idx]; + OpenPolyline& polyline_1 = open_polylines[best_polyline_1_idx]; // join polylines according to plan joinPolylines(polyline_0, polyline_1, reverse); @@ -534,7 +536,7 @@ void SlicerLayer::connectOpenPolylinesImpl(Polygons& open_polylines, coord_t max } } -void SlicerLayer::stitch_extensive(Polygons& open_polylines) +void SlicerLayer::stitch_extensive(OpenLinesSet& open_polylines) { // For extensive stitching find 2 open polygons that are touching 2 closed polygons. // Then find the shortest path over this polygon that can be used to connect the open polygons, @@ -549,7 +551,7 @@ void SlicerLayer::stitch_extensive(Polygons& open_polylines) for (unsigned int polyline_1_idx = 0; polyline_1_idx < open_polylines.size(); polyline_1_idx++) { - PolygonRef polyline_1 = open_polylines[polyline_1_idx]; + OpenPolyline& polyline_1 = open_polylines[polyline_1_idx]; if (polyline_1.size() < 1) continue; @@ -565,7 +567,7 @@ void SlicerLayer::stitch_extensive(Polygons& open_polylines) for (unsigned int polyline_2_idx = 0; polyline_2_idx < open_polylines.size(); polyline_2_idx++) { - PolygonRef polyline_2 = open_polylines[polyline_2_idx]; + OpenPolyline& polyline_2 = open_polylines[polyline_2_idx]; if (polyline_2.size() < 1 || polyline_1_idx == polyline_2_idx) continue; @@ -585,24 +587,24 @@ void SlicerLayer::stitch_extensive(Polygons& open_polylines) { if (best_result->pointIdxA == best_result->pointIdxB) { - polygons.add(open_polylines[best_polyline_1_idx]); + polygons_.push_back(Polygon(open_polylines[best_polyline_1_idx].getPoints(), true)); open_polylines[best_polyline_1_idx].clear(); } else if (best_result->AtoB) { - PolygonRef poly = polygons.newPoly(); - for (unsigned int j = best_result->pointIdxA; j != best_result->pointIdxB; j = (j + 1) % polygons[best_result->polygonIdx].size()) - poly.add(polygons[best_result->polygonIdx][j]); + Polygon& poly = polygons_.newLine(); + for (unsigned int j = best_result->pointIdxA; j != best_result->pointIdxB; j = (j + 1) % polygons_[best_result->polygonIdx].size()) + poly.push_back(polygons_[best_result->polygonIdx][j]); for (unsigned int j = open_polylines[best_polyline_1_idx].size() - 1; int(j) >= 0; j--) - poly.add(open_polylines[best_polyline_1_idx][j]); + poly.push_back(open_polylines[best_polyline_1_idx][j]); open_polylines[best_polyline_1_idx].clear(); } else { - unsigned int n = polygons.size(); - polygons.add(open_polylines[best_polyline_1_idx]); - for (unsigned int j = best_result->pointIdxB; j != best_result->pointIdxA; j = (j + 1) % polygons[best_result->polygonIdx].size()) - polygons[n].add(polygons[best_result->polygonIdx][j]); + unsigned int n = polygons_.size(); + polygons_.push_back(Polygon(open_polylines[best_polyline_1_idx].getPoints(), true)); + for (unsigned int j = best_result->pointIdxB; j != best_result->pointIdxA; j = (j + 1) % polygons_[best_result->polygonIdx].size()) + polygons_[n].push_back(polygons_[best_result->polygonIdx][j]); open_polylines[best_polyline_1_idx].clear(); } } @@ -611,26 +613,26 @@ void SlicerLayer::stitch_extensive(Polygons& open_polylines) if (best_result->pointIdxA == best_result->pointIdxB) { for (unsigned int n = 0; n < open_polylines[best_polyline_1_idx].size(); n++) - open_polylines[best_polyline_2_idx].add(open_polylines[best_polyline_1_idx][n]); + open_polylines[best_polyline_2_idx].push_back(open_polylines[best_polyline_1_idx][n]); open_polylines[best_polyline_1_idx].clear(); } else if (best_result->AtoB) { Polygon poly; - for (unsigned int n = best_result->pointIdxA; n != best_result->pointIdxB; n = (n + 1) % polygons[best_result->polygonIdx].size()) - poly.add(polygons[best_result->polygonIdx][n]); + for (unsigned int n = best_result->pointIdxA; n != best_result->pointIdxB; n = (n + 1) % polygons_[best_result->polygonIdx].size()) + poly.push_back(polygons_[best_result->polygonIdx][n]); for (unsigned int n = poly.size() - 1; int(n) >= 0; n--) - open_polylines[best_polyline_2_idx].add(poly[n]); + open_polylines[best_polyline_2_idx].push_back(poly[n]); for (unsigned int n = 0; n < open_polylines[best_polyline_1_idx].size(); n++) - open_polylines[best_polyline_2_idx].add(open_polylines[best_polyline_1_idx][n]); + open_polylines[best_polyline_2_idx].push_back(open_polylines[best_polyline_1_idx][n]); open_polylines[best_polyline_1_idx].clear(); } else { - for (unsigned int n = best_result->pointIdxB; n != best_result->pointIdxA; n = (n + 1) % polygons[best_result->polygonIdx].size()) - open_polylines[best_polyline_2_idx].add(polygons[best_result->polygonIdx][n]); + for (unsigned int n = best_result->pointIdxB; n != best_result->pointIdxA; n = (n + 1) % polygons_[best_result->polygonIdx].size()) + open_polylines[best_polyline_2_idx].push_back(polygons_[best_result->polygonIdx][n]); for (unsigned int n = open_polylines[best_polyline_1_idx].size() - 1; int(n) >= 0; n--) - open_polylines[best_polyline_2_idx].add(open_polylines[best_polyline_1_idx][n]); + open_polylines[best_polyline_2_idx].push_back(open_polylines[best_polyline_1_idx][n]); open_polylines[best_polyline_1_idx].clear(); } } @@ -665,21 +667,21 @@ std::optional SlicerLayer::findPolygonGapCloser(Point2LL ip0, P else { // Find out if we have should go from A to B or the other way around. - Point2LL p0 = polygons[ret.polygonIdx][ret.pointIdxA]; + Point2LL p0 = polygons_[ret.polygonIdx][ret.pointIdxA]; int64_t lenA = vSize(p0 - ip0); - for (unsigned int i = ret.pointIdxA; i != ret.pointIdxB; i = (i + 1) % polygons[ret.polygonIdx].size()) + for (unsigned int i = ret.pointIdxA; i != ret.pointIdxB; i = (i + 1) % polygons_[ret.polygonIdx].size()) { - Point2LL p1 = polygons[ret.polygonIdx][i]; + Point2LL p1 = polygons_[ret.polygonIdx][i]; lenA += vSize(p0 - p1); p0 = p1; } lenA += vSize(p0 - ip1); - p0 = polygons[ret.polygonIdx][ret.pointIdxB]; + p0 = polygons_[ret.polygonIdx][ret.pointIdxB]; int64_t lenB = vSize(p0 - ip1); - for (unsigned int i = ret.pointIdxB; i != ret.pointIdxA; i = (i + 1) % polygons[ret.polygonIdx].size()) + for (unsigned int i = ret.pointIdxB; i != ret.pointIdxA; i = (i + 1) % polygons_[ret.polygonIdx].size()) { - Point2LL p1 = polygons[ret.polygonIdx][i]; + Point2LL p1 = polygons_[ret.polygonIdx][i]; lenB += vSize(p0 - p1); p0 = p1; } @@ -701,12 +703,12 @@ std::optional SlicerLayer::findPolygonGapCloser(Point2LL ip0, P std::optional SlicerLayer::findPolygonPointClosestTo(Point2LL input) { - for (size_t n = 0; n < polygons.size(); n++) + for (size_t n = 0; n < polygons_.size(); n++) { - Point2LL p0 = polygons[n][polygons[n].size() - 1]; - for (size_t i = 0; i < polygons[n].size(); i++) + Point2LL p0 = polygons_[n][polygons_[n].size() - 1]; + for (size_t i = 0; i < polygons_[n].size(); i++) { - Point2LL p1 = polygons[n][i]; + Point2LL p1 = polygons_[n][i]; // Q = A + Normal( B - A ) * ((( B - A ) dot ( P - A )) / VSize( A - B )); Point2LL pDiff = p1 - p0; @@ -735,7 +737,7 @@ std::optional SlicerLayer::findPolygonPointClosestTo(Point2L void SlicerLayer::makePolygons(const Mesh* mesh) { - Polygons open_polylines; + OpenLinesSet open_polylines; makeBasicPolygonLoops(open_polylines); @@ -757,47 +759,43 @@ void SlicerLayer::makePolygons(const Mesh* mesh) if (mesh->settings_.get("meshfix_keep_open_polygons")) { - for (PolygonRef polyline : open_polylines) + for (const OpenPolyline& polyline : open_polylines) { - if (polyline.size() > 0) - polygons.add(polyline); + polygons_.push_back(Polygon(polyline.getPoints(), false), CheckNonEmptyParam::OnlyIfNotEmpty); } } - for (PolygonRef polyline : open_polylines) + for (const OpenPolyline& polyline : open_polylines) { - if (polyline.size() > 0) - { - openPolylines.add(polyline); - } + open_polylines_.push_back(std::move(polyline), CheckNonEmptyParam::OnlyIfNotEmpty); } // Remove all the tiny polygons, or polygons that are not closed. As they do not contribute to the actual print. const coord_t snap_distance = std::max(mesh->settings_.get("minimum_polygon_circumference"), static_cast(1)); - auto it = std::remove_if( - polygons.begin(), - polygons.end(), - [snap_distance](PolygonRef poly) + auto itPolygons = std::remove_if( + polygons_.begin(), + polygons_.end(), + [snap_distance](const Polygon& poly) { return poly.shorterThan(snap_distance); }); - polygons.erase(it, polygons.end()); + polygons_.erase(itPolygons, polygons_.end()); // Finally optimize all the polygons. Every point removed saves time in the long run. - polygons = Simplify(mesh->settings_).polygon(polygons); - polygons.removeDegenerateVerts(); // remove verts connected to overlapping line segments + polygons_ = Simplify(mesh->settings_).polygon(polygons_); + polygons_.removeDegenerateVerts(); // remove verts connected to overlapping line segments // Clean up polylines for Surface Mode printing - it = std::remove_if( - openPolylines.begin(), - openPolylines.end(), - [snap_distance](PolygonRef poly) + auto itPolylines = std::remove_if( + open_polylines_.begin(), + open_polylines_.end(), + [snap_distance](const OpenPolyline& line) { - return poly.shorterThan(snap_distance); + return line.shorterThan(snap_distance); }); - openPolylines.erase(it, openPolylines.end()); + open_polylines_.erase(itPolylines, open_polylines_.end()); - openPolylines.removeDegenerateVertsPolyline(); + open_polylines_.removeDegenerateVerts(); } Slicer::Slicer(Mesh* i_mesh, const coord_t thickness, const size_t slice_layer_count, bool use_variable_layer_heights, std::vector* adaptive_layers) @@ -841,8 +839,8 @@ void Slicer::buildSegments(const Mesh& mesh, const std::vector Slicer::buildLayersWithHeight( layers_res.resize(slice_layer_count); // set (and initialize compensation for) initial layer, depending on slicing mode - layers_res[0].z = slicing_tolerance == SlicingTolerance::INCLUSIVE ? 0 : std::max(0LL, initial_layer_thickness - thickness); + layers_res[0].z_ = slicing_tolerance == SlicingTolerance::INCLUSIVE ? 0 : std::max(0LL, initial_layer_thickness - thickness); coord_t adjusted_layer_offset = initial_layer_thickness; if (use_variable_layer_heights) { - layers_res[0].z = (*adaptive_layers)[0].z_position_; + layers_res[0].z_ = (*adaptive_layers)[0].z_position_; } else if (slicing_tolerance == SlicingTolerance::MIDDLE) { - layers_res[0].z = initial_layer_thickness / 2; + layers_res[0].z_ = initial_layer_thickness / 2; adjusted_layer_offset = initial_layer_thickness + (thickness / 2); } @@ -996,11 +994,11 @@ std::vector Slicer::buildLayersWithHeight( { if (use_variable_layer_heights) { - layers_res[layer_nr].z = (*adaptive_layers)[layer_nr].z_position_; + layers_res[layer_nr].z_ = (*adaptive_layers)[layer_nr].z_position_; } else { - layers_res[layer_nr].z = adjusted_layer_offset + (thickness * (layer_nr - 1)); + layers_res[layer_nr].z_ = adjusted_layer_offset + (thickness * (layer_nr - 1)); } } @@ -1021,15 +1019,15 @@ void Slicer::makePolygons(Mesh& mesh, SlicingTolerance slicing_tolerance, std::v case SlicingTolerance::INCLUSIVE: for (LayerIndex layer_nr = 0; layer_nr + 1 < layers.size(); layer_nr++) { - layers[layer_nr].polygons = layers[layer_nr].polygons.unionPolygons(layers[layer_nr + 1].polygons); + layers[layer_nr].polygons_ = layers[layer_nr].polygons_.unionPolygons(layers[layer_nr + 1].polygons_); } break; case SlicingTolerance::EXCLUSIVE: for (LayerIndex layer_nr = 0; layer_nr + 1 < layers.size(); layer_nr++) { - layers[layer_nr].polygons = layers[layer_nr].polygons.intersection(layers[layer_nr + 1].polygons); + layers[layer_nr].polygons_ = layers[layer_nr].polygons_.intersection(layers[layer_nr + 1].polygons_); } - layers.back().polygons.clear(); + layers.back().polygons_.clear(); break; case SlicingTolerance::MIDDLE: default: @@ -1038,7 +1036,7 @@ void Slicer::makePolygons(Mesh& mesh, SlicingTolerance slicing_tolerance, std::v } size_t layer_apply_initial_xy_offset = 0; - if (layers.size() > 0 && layers[0].polygons.size() == 0 && ! mesh.settings_.get("support_mesh") && ! mesh.settings_.get("anti_overhang_mesh") + if (layers.size() > 0 && layers[0].polygons_.size() == 0 && ! mesh.settings_.get("support_mesh") && ! mesh.settings_.get("anti_overhang_mesh") && ! mesh.settings_.get("cutting_mesh") && ! mesh.settings_.get("infill_mesh")) { layer_apply_initial_xy_offset = 1; @@ -1060,18 +1058,18 @@ void Slicer::makePolygons(Mesh& mesh, SlicingTolerance slicing_tolerance, std::v const auto xy_offset_local = (layer_nr <= layer_apply_initial_xy_offset) ? xy_offset_0 : xy_offset; if (xy_offset_local != 0) { - layers[layer_nr].polygons = layers[layer_nr].polygons.offset(xy_offset_local, ClipperLib::JoinType::jtRound); + layers[layer_nr].polygons_ = layers[layer_nr].polygons_.offset(xy_offset_local, ClipperLib::JoinType::jtRound); } if (xy_offset_hole != 0) { - const auto parts = layers[layer_nr].polygons.splitIntoParts(); - layers[layer_nr].polygons.clear(); + const auto parts = layers[layer_nr].polygons_.splitIntoParts(); + layers[layer_nr].polygons_.clear(); for (const auto& part : parts) { - Polygons holes; - Polygons outline; - for (ConstPolygonRef poly : part) + Shape holes; + Shape outline; + for (const Polygon& poly : part) { const auto area = poly.area(); const auto abs_area = std::abs(area); @@ -1080,25 +1078,25 @@ void Slicer::makePolygons(Mesh& mesh, SlicingTolerance slicing_tolerance, std::v { if (hole_offset_max_diameter == 0) { - holes.add(poly.offset(xy_offset_hole)); + holes.push_back(poly.offset(xy_offset_hole)); } else if (abs_area < max_hole_area) { const auto distance = static_cast(std::lerp(xy_offset_hole, 0, abs_area / max_hole_area)); - holes.add(poly.offset(distance)); + holes.push_back(poly.offset(distance)); } else { - holes.add(poly); + holes.push_back(poly); } } else { - outline.add(poly); + outline.push_back(poly); } } - layers[layer_nr].polygons.add(outline.difference(holes.unionPolygons())); + layers[layer_nr].polygons_.push_back(outline.difference(holes.unionPolygons())); } } }); diff --git a/src/support.cpp b/src/support.cpp index 7488c6832f..a432da3d58 100644 --- a/src/support.cpp +++ b/src/support.cpp @@ -35,6 +35,7 @@ #include "slicer.h" #include "utils/Simplify.h" #include "utils/ThreadPool.h" +#include "utils/linearAlg2D.h" #include "utils/math.h" namespace cura @@ -61,13 +62,13 @@ bool AreaSupport::handleSupportModifierMesh(SliceDataStorage& storage, const Set switch (modifier_type) { case ANTI_OVERHANG: - support_layer.anti_overhang.add(slicer_layer.polygons); + support_layer.anti_overhang.push_back(slicer_layer.polygons_); break; case SUPPORT_DROP_DOWN: - support_layer.support_mesh_drop_down.add(slicer_layer.polygons); + support_layer.support_mesh_drop_down.push_back(slicer_layer.polygons_); break; case SUPPORT_VANILLA: - support_layer.support_mesh.add(slicer_layer.polygons); + support_layer.support_mesh.push_back(slicer_layer.polygons_); break; } } @@ -75,10 +76,7 @@ bool AreaSupport::handleSupportModifierMesh(SliceDataStorage& storage, const Set } -void AreaSupport::splitGlobalSupportAreasIntoSupportInfillParts( - SliceDataStorage& storage, - const std::vector& global_support_areas_per_layer, - unsigned int total_layer_count) +void AreaSupport::splitGlobalSupportAreasIntoSupportInfillParts(SliceDataStorage& storage, const std::vector& global_support_areas_per_layer, unsigned int total_layer_count) { if (total_layer_count == 0) { @@ -105,7 +103,7 @@ void AreaSupport::splitGlobalSupportAreasIntoSupportInfillParts( wall_line_count_this_layer++; } - const Polygons& global_support_areas = global_support_areas_per_layer[layer_nr]; + const Shape& global_support_areas = global_support_areas_per_layer[layer_nr]; if (global_support_areas.size() == 0 || layer_nr < min_layer || layer_nr > max_layer) { // Initialize support_infill_parts empty @@ -187,6 +185,8 @@ void AreaSupport::generateGradualSupport(SliceDataStorage& storage) const size_t max_density_steps = infill_extruder.settings_.get("gradual_support_infill_steps"); const coord_t wall_width = infill_extruder.settings_.get("support_line_width"); + const bool is_connected = infill_extruder.settings_.get("zig_zaggify_infill") || infill_extruder.settings_.get("infill_pattern") == EFillMethod::ZIG_ZAG; + const Simplify simplifier(infill_extruder.settings_); // no early-out for this function; it needs to initialize the [infill_area_per_combine_per_density] double layer_skip_count{ 8.0 }; // skip every so many layers as to ignore small gaps in the model making computation more easy @@ -215,13 +215,13 @@ void AreaSupport::generateGradualSupport(SliceDataStorage& storage) { SupportInfillPart& support_infill_part = support_infill_parts[part_idx]; - Polygons original_area = support_infill_part.getInfillArea(); + Shape original_area = support_infill_part.getInfillArea(); if (original_area.empty()) { continue; } // NOTE: This both generates the walls _and_ returns the _actual_ infill area (the one _without_ walls) for use in the rest of the method. - const Polygons infill_area = Infill::generateWallToolPaths( + const Shape infill_area = Infill::generateWallToolPaths( support_infill_part.wall_toolpaths_, original_area, support_infill_part.inset_count_to_generate_, @@ -233,7 +233,8 @@ void AreaSupport::generateGradualSupport(SliceDataStorage& storage) const AABB& this_part_boundary_box = support_infill_part.outline_boundary_box_; // calculate density areas for this island - Polygons less_dense_support = infill_area; // one step less dense with each density_step + Shape less_dense_support = infill_area; // one step less dense with each density_step + Shape sum_more_dense; // NOTE: Only used for zig-zag or connected fills. for (unsigned int density_step = 0; density_step < max_density_steps; ++density_step) { LayerIndex actual_min_layer{ layer_nr + density_step * gradual_support_step_layer_count + static_cast(layer_skip_count) }; @@ -249,7 +250,7 @@ void AreaSupport::generateGradualSupport(SliceDataStorage& storage) // compute intersections with relevant upper parts const std::vector upper_infill_parts = storage.support.supportLayers[upper_layer_idx].support_infill_parts; - Polygons relevant_upper_polygons; + Shape relevant_upper_polygons; for (unsigned int upper_part_idx = 0; upper_part_idx < upper_infill_parts.size(); ++upper_part_idx) { if (support_infill_part.outline_.empty()) @@ -277,7 +278,7 @@ void AreaSupport::generateGradualSupport(SliceDataStorage& storage) // if (upper_part_boundary_box.hit(this_part_boundary_box)) { - relevant_upper_polygons.add(upper_infill_parts[upper_part_idx].outline_); + relevant_upper_polygons.push_back(upper_infill_parts[upper_part_idx].outline_); } } @@ -290,14 +291,18 @@ void AreaSupport::generateGradualSupport(SliceDataStorage& storage) // add new infill_area_per_combine_per_density for the current density support_infill_part.infill_area_per_combine_per_density_.emplace_back(); - std::vector& support_area_current_density = support_infill_part.infill_area_per_combine_per_density_.back(); - const Polygons more_dense_support = infill_area.difference(less_dense_support); - support_area_current_density.push_back(more_dense_support); + std::vector& support_area_current_density = support_infill_part.infill_area_per_combine_per_density_.back(); + const Shape more_dense_support = infill_area.difference(less_dense_support); + support_area_current_density.push_back(simplifier.polygon(more_dense_support.difference(sum_more_dense))); + if (is_connected) + { + sum_more_dense = sum_more_dense.unionPolygons(more_dense_support); + } } support_infill_part.infill_area_per_combine_per_density_.emplace_back(); - std::vector& support_area_current_density = support_infill_part.infill_area_per_combine_per_density_.back(); - support_area_current_density.push_back(infill_area); + std::vector& support_area_current_density = support_infill_part.infill_area_per_combine_per_density_.back(); + support_area_current_density.push_back(simplifier.polygon(infill_area.difference(sum_more_dense))); assert(support_infill_part.infill_area_per_combine_per_density_.size() != 0 && "support_infill_part.infill_area_per_combine_per_density should now be initialized"); #ifdef DEBUG @@ -365,8 +370,8 @@ void AreaSupport::combineSupportInfillLayers(SliceDataStorage& storage) } for (unsigned int density_idx = 0; density_idx < part.infill_area_per_combine_per_density_.size(); ++density_idx) { // go over each density of gradual infill (these density areas overlap!) - std::vector& infill_area_per_combine = part.infill_area_per_combine_per_density_[density_idx]; - Polygons result; + std::vector& infill_area_per_combine = part.infill_area_per_combine_per_density_[density_idx]; + Shape result; for (SupportInfillPart& lower_layer_part : lower_layer.support_infill_parts) { if (! part.outline_boundary_box_.hit(lower_layer_part.outline_boundary_box_)) @@ -374,13 +379,13 @@ void AreaSupport::combineSupportInfillLayers(SliceDataStorage& storage) continue; } - Polygons intersection = infill_area_per_combine[combine_count_here - 1].intersection(lower_layer_part.getInfillArea()).offset(-200).offset(200); + Shape intersection = infill_area_per_combine[combine_count_here - 1].intersection(lower_layer_part.getInfillArea()).offset(-200).offset(200); if (intersection.size() <= 0) { continue; } - result.add(intersection); // add area to be thickened + result.push_back(intersection); // add area to be thickened infill_area_per_combine[combine_count_here - 1] = infill_area_per_combine[combine_count_here - 1].difference(intersection); // remove thickened area from less thick layer here @@ -401,7 +406,7 @@ void AreaSupport::combineSupportInfillLayers(SliceDataStorage& storage) lower_density_idx <= max_lower_density_idx && lower_density_idx < lower_layer_part.infill_area_per_combine_per_density_.size(); lower_density_idx++) { - std::vector& lower_infill_area_per_combine = lower_layer_part.infill_area_per_combine_per_density_[lower_density_idx]; + std::vector& lower_infill_area_per_combine = lower_layer_part.infill_area_per_combine_per_density_[lower_density_idx]; lower_infill_area_per_combine[0] = lower_infill_area_per_combine[0].difference(intersection); // remove thickened area from lower (single thickness) layer } @@ -430,9 +435,9 @@ void AreaSupport::cleanup(SliceDataStorage& storage) } else { - for (const std::vector& infill_area_per_combine_this_density : part.infill_area_per_combine_per_density_) + for (const std::vector& infill_area_per_combine_this_density : part.infill_area_per_combine_per_density_) { - for (const Polygons& infill_area_this_combine_this_density : infill_area_per_combine_this_density) + for (const Shape& infill_area_this_combine_this_density : infill_area_per_combine_this_density) { // remove small areas which were introduced by rounding errors in comparing the same area on two consecutive layer if (! infill_area_this_combine_this_density.empty() && infill_area_this_combine_this_density.area() > support_line_width * support_line_width) @@ -457,9 +462,9 @@ void AreaSupport::cleanup(SliceDataStorage& storage) } } -Polygons AreaSupport::join(const SliceDataStorage& storage, const Polygons& supportLayer_up, Polygons& supportLayer_this) +Shape AreaSupport::join(const SliceDataStorage& storage, const Shape& supportLayer_up, Shape& supportLayer_this) { - Polygons joined; + Shape joined; const Settings& infill_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings.get("support_infill_extruder_nr").settings_; const AngleRadians conical_support_angle = infill_settings.get("support_conical_angle"); @@ -478,7 +483,7 @@ Polygons AreaSupport::join(const SliceDataStorage& storage, const Polygons& supp { const Settings& mesh_group_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; // Don't go outside the build volume. - Polygons machine_volume_border; + Shape machine_volume_border; switch (mesh_group_settings.get("machine_shape")) { case BuildPlateShape::ELLIPTIC: @@ -496,12 +501,12 @@ Polygons AreaSupport::join(const SliceDataStorage& storage, const Polygons& supp const coord_t y = machine_middle.y_ + sin(angle) * depth / 2; border_circle.emplace_back(x, y); } - machine_volume_border.add(border_circle); + machine_volume_border.push_back(border_circle); break; } case BuildPlateShape::RECTANGULAR: default: - machine_volume_border.add(storage.machine_size.flatten().toPolygon()); + machine_volume_border.push_back(storage.machine_size.flatten().toPolygon()); break; } coord_t adhesion_size = 0; // Make sure there is enough room for the platform adhesion around support. @@ -556,8 +561,8 @@ Polygons AreaSupport::join(const SliceDataStorage& storage, const Polygons& supp machine_volume_border = machine_volume_border.offset(-adhesion_size); const coord_t conical_smallest_breadth = infill_settings.get("support_conical_min_width"); - Polygons insetted = supportLayer_up.offset(-conical_smallest_breadth / 2); - Polygons small_parts = supportLayer_up.difference(insetted.offset(conical_smallest_breadth / 2 + 20)); + Shape insetted = supportLayer_up.offset(-conical_smallest_breadth / 2); + Shape small_parts = supportLayer_up.difference(insetted.offset(conical_smallest_breadth / 2 + 20)); joined = supportLayer_this.unionPolygons(supportLayer_up.offset(conical_support_offset)).unionPolygons(small_parts).intersection(machine_volume_border); } else @@ -603,7 +608,7 @@ void AreaSupport::generateOverhangAreas(SliceDataStorage& storage) void AreaSupport::generateSupportAreas(SliceDataStorage& storage) { - std::vector global_support_areas_per_layer; + std::vector global_support_areas_per_layer; global_support_areas_per_layer.resize(storage.print_layer_count); int max_layer_nr_support_mesh_filled; @@ -666,17 +671,17 @@ void AreaSupport::generateSupportAreas(SliceDataStorage& storage) support_meshes_handled = true; } } - std::vector mesh_support_areas_per_layer; - mesh_support_areas_per_layer.resize(storage.print_layer_count, Polygons()); + std::vector mesh_support_areas_per_layer; + mesh_support_areas_per_layer.resize(storage.print_layer_count, Shape()); generateSupportAreasForMesh(storage, *infill_settings, *roof_settings, *bottom_settings, mesh_idx, storage.print_layer_count, mesh_support_areas_per_layer); for (size_t layer_idx = 0; layer_idx < storage.print_layer_count; layer_idx++) { - global_support_areas_per_layer[layer_idx].add(mesh_support_areas_per_layer[layer_idx]); + global_support_areas_per_layer[layer_idx].push_back(mesh_support_areas_per_layer[layer_idx]); } } - for (Polygons& support_areas : global_support_areas_per_layer) + for (Shape& support_areas : global_support_areas_per_layer) { support_areas = support_areas.unionPolygons(); } @@ -802,7 +807,7 @@ void AreaSupport::generateOverhangAreasForMesh(SliceDataStorage& storage, SliceM storage.print_layer_count, [&](const size_t layer_idx) { - std::pair basic_and_full_overhang = computeBasicAndFullOverhang(storage, mesh, layer_idx); + std::pair basic_and_full_overhang = computeBasicAndFullOverhang(storage, mesh, layer_idx); mesh.overhang_areas[layer_idx] = basic_and_full_overhang.first; // Store the results. mesh.full_overhang_areas[layer_idx] = basic_and_full_overhang.second; scripta::log("support_basic_overhang_area", basic_and_full_overhang.first, SectionType::SUPPORT, layer_idx); @@ -810,7 +815,7 @@ void AreaSupport::generateOverhangAreasForMesh(SliceDataStorage& storage, SliceM }); } -Polygons AreaSupport::generateVaryingXYDisallowedArea(const SliceMeshStorage& storage, const LayerIndex layer_idx) +Shape AreaSupport::generateVaryingXYDisallowedArea(const SliceMeshStorage& storage, const LayerIndex layer_idx) { const auto& mesh_group_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; const Simplify simplify{ mesh_group_settings }; @@ -822,7 +827,7 @@ Polygons AreaSupport::generateVaryingXYDisallowedArea(const SliceMeshStorage& st constexpr auto close_dist = 20; - auto layer_current = simplify.polygon(storage.layers[layer_idx].getOutlines().offset(-close_dist).offset(close_dist)); + Shape layer_current = simplify.polygon(storage.layers[layer_idx].getOutlines().offset(-close_dist).offset(close_dist)); using point_pair_t = std::pair; using poly_point_key = std::tuple; @@ -835,7 +840,7 @@ Polygons AreaSupport::generateVaryingXYDisallowedArea(const SliceMeshStorage& st { double support_distance; double delta_z; - Polygons layer_delta; + Shape layer_delta; }; std::vector z_distances_layer_deltas; @@ -951,14 +956,14 @@ Polygons AreaSupport::generateVaryingXYDisallowedArea(const SliceMeshStorage& st } const auto smooth_dist = xy_distance / 2.0; - Polygons varying_xy_disallowed_areas = layer_current - // offset using the varying offset distances we calculated previously - .offset(varying_offsets) - // close operation to smooth the x/y disallowed area boundary. With varying xy distances we see some jumps in the boundary. - // As the x/y disallowed areas "cut in" to support the xy-disallowed area may propagate through the support area. If the - // x/y disallowed area is not smoothed boost has trouble generating a voronoi diagram. - .offset(smooth_dist) - .offset(-smooth_dist); + Shape varying_xy_disallowed_areas = layer_current + // offset using the varying offset distances we calculated previously + .offsetMulti(varying_offsets) + // close operation to smooth the x/y disallowed area boundary. With varying xy distances we see some jumps in the boundary. + // As the x/y disallowed areas "cut in" to support the xy-disallowed area may propagate through the support area. If the + // x/y disallowed area is not smoothed boost has trouble generating a voronoi diagram. + .offset(smooth_dist) + .offset(-smooth_dist); scripta::log("support_varying_xy_disallowed_areas", varying_xy_disallowed_areas, SectionType::SUPPORT, layer_idx); return varying_xy_disallowed_areas; } @@ -982,7 +987,7 @@ void AreaSupport::generateSupportAreasForMesh( const Settings& bottom_settings, const size_t mesh_idx, const size_t layer_count, - std::vector& support_areas) + std::vector& support_areas) { SliceMeshStorage& mesh = *storage.meshes[mesh_idx]; @@ -1010,11 +1015,11 @@ void AreaSupport::generateSupportAreasForMesh( } // Compute the areas that are disallowed by the X/Y distance. - std::vector xy_disallowed_per_layer; + std::vector xy_disallowed_per_layer; xy_disallowed_per_layer.resize(layer_count); - std::vector sloped_areas_per_layer; + std::vector sloped_areas_per_layer; sloped_areas_per_layer.resize(layer_count); - sloped_areas_per_layer[0] = Polygons(); + sloped_areas_per_layer[0] = Shape(); // simplified processing for bottom layer - just ensure support clears part by XY distance const coord_t xy_distance = infill_settings.get("support_xy_distance"); const coord_t xy_distance_overhang = infill_settings.get("support_xy_distance_overhang"); @@ -1037,7 +1042,7 @@ void AreaSupport::generateSupportAreasForMesh( layer_count, [&](const size_t layer_idx) { - const Polygons outlines = storage.getLayerOutlines(layer_idx, no_support, no_prime_tower); + const Shape outlines = storage.getLayerOutlines(layer_idx, no_support, no_prime_tower); // Build sloped areas. We need this for the stair-stepping later on. // Specifically, sloped areass are used in 'moveUpFromModel' to prevent a stair step happening over an area where there isn't a slope. @@ -1045,10 +1050,10 @@ void AreaSupport::generateSupportAreasForMesh( sloped_areas_per_layer[layer_idx] = // Take the outer areas of the previous layer, where the outer areas are (mostly) just _inside_ the shape. storage.getLayerOutlines(layer_idx - 1, no_support, no_prime_tower) - .tubeShape(sloped_area_detection_width, 10) + .createTubeShape(sloped_area_detection_width, 10) // Intersect those with the outer areas of the current layer, where the outer areas are (mostly) _outside_ the shape. // This will detect every slope (and some/most vertical walls) between those two layers. - .intersection(outlines.tubeShape(10, sloped_area_detection_width)) + .intersection(outlines.createTubeShape(10, sloped_area_detection_width)) // Do an opening operation so we're not stuck with tiny patches. // The later offset is extended with the line-width, so all patches are merged together if there's less than a line-width between them. .offset(-10) @@ -1068,8 +1073,8 @@ void AreaSupport::generateSupportAreasForMesh( // we also want to use the min XY distance when the support is resting on a sloped surface so we calculate the area of the // layer below that protrudes beyond the current layer's area and combine it with the current layer's overhang disallowed area - Polygons minimum_xy_disallowed_areas = mesh.layers[layer_idx].getOutlines().offset(xy_distance_overhang); - Polygons varying_xy_disallowed_areas = generateVaryingXYDisallowedArea(mesh, layer_idx); + Shape minimum_xy_disallowed_areas = mesh.layers[layer_idx].getOutlines().offset(xy_distance_overhang); + Shape varying_xy_disallowed_areas = generateVaryingXYDisallowedArea(mesh, layer_idx); xy_disallowed_per_layer[layer_idx] = minimum_xy_disallowed_areas.unionPolygons(varying_xy_disallowed_areas); scripta::log("support_xy_disallowed_areas", xy_disallowed_per_layer[layer_idx], SectionType::SUPPORT, layer_idx); } @@ -1080,8 +1085,8 @@ void AreaSupport::generateSupportAreasForMesh( } }); - std::vector tower_roofs; - Polygons stair_removal; // polygons to subtract from support because of stair-stepping + std::vector tower_roofs; + Shape stair_removal; // polygons to subtract from support because of stair-stepping const bool is_support_mesh_nondrop_place_holder = is_support_mesh_place_holder && ! mesh.settings.get("support_mesh_drop_down"); const bool is_support_mesh_drop_down_place_holder = is_support_mesh_place_holder && mesh.settings.get("support_mesh_drop_down"); @@ -1124,7 +1129,7 @@ void AreaSupport::generateSupportAreasForMesh( for (size_t layer_idx = layer_count - 1 - layer_z_distance_top; layer_idx != static_cast(-1); layer_idx--) { - Polygons layer_this = mesh.full_overhang_areas[layer_idx + layer_z_distance_top]; + Shape layer_this = mesh.full_overhang_areas[layer_idx + layer_z_distance_top]; if (extension_offset && ! is_support_mesh_place_holder) { @@ -1133,11 +1138,11 @@ void AreaSupport::generateSupportAreasForMesh( // model outline and the support is effectively calculating a voronoi. The offset is first applied to // the support and next to the model to ensure that the expanded support area is connected to the original // support area. Please note that the horizontal expansion is rounded down to an integer offset_per_step. - Polygons model_outline = storage.getLayerOutlines(layer_idx, no_support, no_prime_tower); + Shape model_outline = storage.getLayerOutlines(layer_idx, no_support, no_prime_tower); const coord_t offset_per_step = support_line_width / 2; // perform a small offset we don't enlarge small features of the support - Polygons horizontal_expansion = layer_this; + Shape horizontal_expansion = layer_this; for (coord_t offset_cumulative = 0; offset_cumulative <= extension_offset; offset_cumulative += offset_per_step) { horizontal_expansion = horizontal_expansion.offset(offset_per_step); @@ -1158,10 +1163,9 @@ void AreaSupport::generateSupportAreasForMesh( if (layer_idx + 1 < layer_count) { // join with support from layer up - const Polygons empty; - const Polygons* layer_above = (layer_idx < support_areas.size()) ? &support_areas[layer_idx + 1] : ∅ - const Polygons model_mesh_on_layer - = (layer_idx > 0) && ! is_support_mesh_nondrop_place_holder ? storage.getLayerOutlines(layer_idx, no_support, no_prime_tower) : empty; + const Shape empty; + const Shape* layer_above = (layer_idx < support_areas.size()) ? &support_areas[layer_idx + 1] : ∅ + const Shape model_mesh_on_layer = (layer_idx > 0) && ! is_support_mesh_nondrop_place_holder ? storage.getLayerOutlines(layer_idx, no_support, no_prime_tower) : empty; if (is_support_mesh_nondrop_place_holder) { layer_above = ∅ @@ -1173,7 +1177,7 @@ void AreaSupport::generateSupportAreasForMesh( // make towers for small support if (use_towers) { - for (PolygonsPart poly : layer_this.splitIntoParts()) + for (SingleShape poly : layer_this.splitIntoParts()) { const auto polygon_part = poly.difference(xy_disallowed_per_layer[layer_idx]).offset(-half_min_feature_width).offset(half_min_feature_width); @@ -1186,8 +1190,8 @@ void AreaSupport::generateSupportAreasForMesh( constexpr size_t tower_top_layer_count = 6; // number of layers after which to conclude that a tiny support area needs a tower if (layer_idx < layer_count - tower_top_layer_count && layer_idx >= tower_top_layer_count + bottom_empty_layer_count) { - Polygons tiny_tower_here; - tiny_tower_here.add(polygon_part); + Shape tiny_tower_here; + tiny_tower_here.push_back(polygon_part); tower_roofs.emplace_back(tiny_tower_here); } } @@ -1236,7 +1240,7 @@ void AreaSupport::generateSupportAreasForMesh( // do stuff for when support on buildplate only if (support_type == ESupportType::PLATFORM_ONLY) { - Polygons touching_buildplate = support_areas[0]; // TODO: not working for conical support! + Shape touching_buildplate = support_areas[0]; // TODO: not working for conical support! const AngleRadians conical_support_angle = infill_settings.get("support_conical_angle"); coord_t conical_support_offset; if (conical_support_angle > 0) @@ -1250,7 +1254,7 @@ void AreaSupport::generateSupportAreasForMesh( const bool conical_support = infill_settings.get("support_conical_enabled") && conical_support_angle != 0; for (LayerIndex layer_idx = 1; layer_idx < storage.support.supportLayers.size(); layer_idx++) { - const Polygons& layer = support_areas[layer_idx]; + const Shape& layer = support_areas[layer_idx]; if (conical_support) { // with conical support the next layer is allowed to be larger than the previous @@ -1301,13 +1305,13 @@ void AreaSupport::generateSupportAreasForMesh( // Procedure to remove floating support for (size_t layer_idx = 1; layer_idx < layer_count - 1; layer_idx++) { - Polygons& layer_this = support_areas[layer_idx]; + Shape& layer_this = support_areas[layer_idx]; if (! layer_this.empty()) { - Polygons& layer_below = support_areas[layer_idx - 1]; - Polygons& layer_above = support_areas[layer_idx + 1]; - Polygons surrounding_layer = layer_above.unionPolygons(layer_below); + Shape& layer_below = support_areas[layer_idx - 1]; + Shape& layer_above = support_areas[layer_idx + 1]; + Shape surrounding_layer = layer_above.unionPolygons(layer_below); layer_this = layer_this.intersection(surrounding_layer); } } @@ -1326,9 +1330,9 @@ void AreaSupport::generateSupportAreasForMesh( void AreaSupport::moveUpFromModel( const SliceDataStorage& storage, - Polygons& stair_removal, - Polygons& sloped_areas, - Polygons& support_areas, + Shape& stair_removal, + Shape& sloped_areas, + Shape& support_areas, const size_t layer_idx, const size_t bottom_empty_layer_count, const size_t bottom_stair_step_layer_count, @@ -1391,9 +1395,9 @@ void AreaSupport::moveUpFromModel( const size_t bottom_layer_nr = layer_idx - bottom_empty_layer_count; constexpr bool no_support = false; constexpr bool no_prime_tower = false; - const Polygons bottom_outline = storage.getLayerOutlines(bottom_layer_nr, no_support, no_prime_tower); + const Shape bottom_outline = storage.getLayerOutlines(bottom_layer_nr, no_support, no_prime_tower); - Polygons to_be_removed; + Shape to_be_removed; if (bottom_stair_step_layer_count <= 1) { to_be_removed = bottom_outline; @@ -1403,13 +1407,13 @@ void AreaSupport::moveUpFromModel( to_be_removed = stair_removal.unionPolygons(bottom_outline); if (layer_idx % bottom_stair_step_layer_count == 0) { // update stairs for next step - const Polygons supporting_bottom = storage.getLayerOutlines(bottom_layer_nr - 1, no_support, no_prime_tower); - const Polygons allowed_step_width = supporting_bottom.offset(support_bottom_stair_step_width).intersection(sloped_areas); + const Shape supporting_bottom = storage.getLayerOutlines(bottom_layer_nr - 1, no_support, no_prime_tower); + const Shape allowed_step_width = supporting_bottom.offset(support_bottom_stair_step_width).intersection(sloped_areas); const int64_t step_bottom_layer_nr = bottom_layer_nr - bottom_stair_step_layer_count + 1; if (step_bottom_layer_nr >= 0) { - const Polygons step_bottom_outline = storage.getLayerOutlines(step_bottom_layer_nr, no_support, no_prime_tower); + const Shape step_bottom_outline = storage.getLayerOutlines(step_bottom_layer_nr, no_support, no_prime_tower); stair_removal = step_bottom_outline.intersection(allowed_step_width); } else @@ -1433,9 +1437,9 @@ void AreaSupport::moveUpFromModel( * ^^^^^^^^^ overhang extensions * ^^^^^^^^^^^^^^ overhang */ -std::pair AreaSupport::computeBasicAndFullOverhang(const SliceDataStorage& storage, const SliceMeshStorage& mesh, const LayerIndex& layer_idx) +std::pair AreaSupport::computeBasicAndFullOverhang(const SliceDataStorage& storage, const SliceMeshStorage& mesh, const LayerIndex& layer_idx) { - const Polygons outlines = mesh.layers[layer_idx].getOutlines(); + const Shape outlines = mesh.layers[layer_idx].getOutlines(); constexpr bool no_support = false; constexpr bool no_prime_tower = false; @@ -1451,29 +1455,29 @@ std::pair AreaSupport::computeBasicAndFullOverhang(const Sli // To avoids generating support for textures on vertical surfaces, a moving average // is taken over smooth_height. The smooth_height is currently an educated guess // that we might want to expose to the frontend in the future. - Polygons outlines_below = storage.getLayerOutlines(layer_idx - 1, no_support, no_prime_tower).offset(max_dist_from_lower_layer); + Shape outlines_below = storage.getLayerOutlines(layer_idx - 1, no_support, no_prime_tower).offset(max_dist_from_lower_layer); for (int layer_idx_offset = 2; layer_idx - layer_idx_offset >= 0 && layer_idx_offset <= layers_below; layer_idx_offset++) { auto outlines_below_ = storage.getLayerOutlines(layer_idx - layer_idx_offset, no_support, no_prime_tower).offset(max_dist_from_lower_layer * layer_idx_offset); outlines_below = outlines_below.unionPolygons(outlines_below_); } - Polygons basic_overhang = outlines.difference(outlines_below); + Shape basic_overhang = outlines.difference(outlines_below); const SupportLayer& support_layer = storage.support.supportLayers[layer_idx]; if (! support_layer.anti_overhang.empty()) { // Merge anti overhang into one polygon, otherwise overlapping polygons // will create opposite effect. - Polygons merged_polygons = support_layer.anti_overhang.unionPolygons(); + Shape merged_polygons = support_layer.anti_overhang.unionPolygons(); basic_overhang = basic_overhang.difference(merged_polygons); } - Polygons overhang_extended = basic_overhang - // +0.1mm for easier joining with support from layer above - .offset(max_dist_from_lower_layer * layers_below + MM2INT(0.1)); - Polygons full_overhang = overhang_extended.intersection(outlines); + Shape overhang_extended = basic_overhang + // +0.1mm for easier joining with support from layer above + .offset(max_dist_from_lower_layer * layers_below + MM2INT(0.1)); + Shape full_overhang = overhang_extended.intersection(outlines); return std::make_pair(basic_overhang, full_overhang); } @@ -1509,7 +1513,7 @@ void AreaSupport::detectOverhangPoints(const SliceDataStorage& storage, SliceMes continue; } - const Polygons overhang = part.outline.difference(storage.support.supportLayers[layer_idx].anti_overhang); + const Shape overhang = part.outline.difference(storage.support.supportLayers[layer_idx].anti_overhang); if (! overhang.empty()) { scripta::log("support_overhangs", overhang, SectionType::SUPPORT, layer_idx); @@ -1521,10 +1525,10 @@ void AreaSupport::detectOverhangPoints(const SliceDataStorage& storage, SliceMes void AreaSupport::handleTowers( const Settings& settings, - const Polygons& xy_disallowed_area, - Polygons& supportLayer_this, - std::vector& tower_roofs, - std::vector>& overhang_points, + const Shape& xy_disallowed_area, + Shape& supportLayer_this, + std::vector& tower_roofs, + std::vector>& overhang_points, LayerIndex layer_idx, size_t layer_count) { @@ -1533,7 +1537,7 @@ void AreaSupport::handleTowers( { return; } - std::vector& overhang_points_here = overhang_points[layer_overhang_point]; // may be changed if an overhang point has a (smaller) overhang point directly below + std::vector& overhang_points_here = overhang_points[layer_overhang_point]; // may be changed if an overhang point has a (smaller) overhang point directly below // handle new tower rooftops if (overhang_points_here.size() > 0) { @@ -1541,16 +1545,16 @@ void AreaSupport::handleTowers( if (layer_overhang_point < static_cast(layer_count) && ! overhang_points[layer_overhang_point - 1].empty()) { const auto max_tower_supported_diameter = settings.get("support_tower_maximum_supported_diameter"); - std::vector& overhang_points_below = overhang_points[layer_overhang_point - 1]; - for (Polygons& poly_here : overhang_points_here) + std::vector& overhang_points_below = overhang_points[layer_overhang_point - 1]; + for (Shape& poly_here : overhang_points_here) { - for (const Polygons& poly_below : overhang_points_below) + for (const Shape& poly_below : overhang_points_below) { poly_here = poly_here.difference(poly_below.offset(max_tower_supported_diameter * 2)); } } } - for (Polygons& poly : overhang_points_here) + for (Shape& poly : overhang_points_here) { if (poly.size() > 0) { @@ -1577,18 +1581,18 @@ void AreaSupport::handleTowers( tower_roof_expansion_distance = layer_thickness / tan_tower_roof_angle; } - for (Polygons& tower_roof : tower_roofs - | ranges::views::filter( - [](const auto& poly) - { - return ! poly.empty(); - })) + for (Shape& tower_roof : tower_roofs + | ranges::views::filter( + [](const auto& poly) + { + return ! poly.empty(); + })) { supportLayer_this = supportLayer_this.unionPolygons(tower_roof); if (tower_roof.area() < tower_diameter * tower_diameter) { - Polygons model_outline = xy_disallowed_area; + Shape model_outline = xy_disallowed_area; // Rather than offsetting the tower with tower_roof_expansion_distance we do this step wise to achieve two things // - prevent support from folding around the model @@ -1624,13 +1628,13 @@ void AreaSupport::handleTowers( } } -void AreaSupport::handleWallStruts(const Settings& settings, Polygons& supportLayer_this) +void AreaSupport::handleWallStruts(const Settings& settings, Shape& supportLayer_this) { const coord_t max_tower_supported_diameter = settings.get("support_tower_maximum_supported_diameter"); const coord_t tower_diameter = settings.get("support_tower_diameter"); for (unsigned int p = 0; p < supportLayer_this.size(); p++) { - PolygonRef poly = supportLayer_this[p]; + const Polygon& poly = supportLayer_this[p]; if (poly.size() < 6) // might be a single wall { int best = -1; @@ -1657,19 +1661,19 @@ void AreaSupport::handleWallStruts(const Settings& settings, Polygons& supportLa if (width < max_tower_supported_diameter) { Point2LL mid = (poly[best] + poly[(best + 1) % poly.size()]) / 2; - Polygons struts; - PolygonRef strut = struts.newPoly(); - strut.add(mid + Point2LL(tower_diameter / 2, tower_diameter / 2)); - strut.add(mid + Point2LL(-tower_diameter / 2, tower_diameter / 2)); - strut.add(mid + Point2LL(-tower_diameter / 2, -tower_diameter / 2)); - strut.add(mid + Point2LL(tower_diameter / 2, -tower_diameter / 2)); + Shape struts; + Polygon& strut = struts.newLine(); + strut.push_back(mid + Point2LL(tower_diameter / 2, tower_diameter / 2)); + strut.push_back(mid + Point2LL(-tower_diameter / 2, tower_diameter / 2)); + strut.push_back(mid + Point2LL(-tower_diameter / 2, -tower_diameter / 2)); + strut.push_back(mid + Point2LL(tower_diameter / 2, -tower_diameter / 2)); supportLayer_this = supportLayer_this.unionPolygons(struts); } } } } -void AreaSupport::generateSupportBottom(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer) +void AreaSupport::generateSupportBottom(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer) { const Settings& mesh_group_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; const coord_t layer_height = mesh_group_settings.get("layer_height"); @@ -1688,19 +1692,19 @@ void AreaSupport::generateSupportBottom(SliceDataStorage& storage, const SliceMe for (LayerIndex layer_idx = support_layers.size() - 1; layer_idx >= static_cast(z_distance_bottom); --layer_idx) { const unsigned int bottom_layer_idx_below = std::max(0, int(layer_idx) - int(bottom_layer_count) - int(z_distance_bottom)); - Polygons mesh_outlines; + Shape mesh_outlines; for (auto layer_idx_below = bottom_layer_idx_below; layer_idx_below < layer_idx - z_distance_bottom + 1; layer_idx_below += 1) { - mesh_outlines.add(mesh.layers[layer_idx_below].getOutlines()); + mesh_outlines.push_back(mesh.layers[layer_idx_below].getOutlines()); } - Polygons bottoms; + Shape bottoms; generateSupportInterfaceLayer(global_support_areas_per_layer[layer_idx], mesh_outlines, bottom_line_width, bottom_outline_offset, minimum_bottom_area, bottoms); - support_layers[layer_idx].support_bottom.add(bottoms); + support_layers[layer_idx].support_bottom.push_back(bottoms); scripta::log("support_interface_bottoms", bottoms, SectionType::SUPPORT, layer_idx); } } -void AreaSupport::generateSupportRoof(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer) +void AreaSupport::generateSupportRoof(SliceDataStorage& storage, const SliceMeshStorage& mesh, std::vector& global_support_areas_per_layer) { const Settings& mesh_group_settings = Application::getInstance().current_slice_->scene.current_mesh_group->settings; const coord_t layer_height = mesh_group_settings.get("layer_height"); @@ -1721,17 +1725,17 @@ void AreaSupport::generateSupportRoof(SliceDataStorage& storage, const SliceMesh const LayerIndex top_layer_idx_above{ std::min(LayerIndex{ support_layers.size() - 1 }, LayerIndex{ layer_idx + roof_layer_count + z_distance_top }) }; // Maximum layer of the model that generates support roof. - Polygons mesh_outlines; + Shape mesh_outlines; for (auto layer_idx_above = top_layer_idx_above; layer_idx_above > layer_idx + z_distance_top - 1; layer_idx_above -= 1) { - mesh_outlines.add(mesh.layers[layer_idx_above].getOutlines()); + mesh_outlines.push_back(mesh.layers[layer_idx_above].getOutlines()); } - Polygons roofs; + Shape roofs; generateSupportInterfaceLayer(global_support_areas_per_layer[layer_idx], mesh_outlines, roof_line_width, roof_outline_offset, minimum_roof_area, roofs); - support_layers[layer_idx].support_roof.add(roofs); + support_layers[layer_idx].support_roof.push_back(roofs); if (layer_idx > 0 && layer_idx < support_layers.size() - 1) { - support_layers[layer_idx].support_fractional_roof.add(roofs.difference(support_layers[layer_idx + 1].support_roof)); + support_layers[layer_idx].support_fractional_roof.push_back(roofs.difference(support_layers[layer_idx + 1].support_roof)); } scripta::log("support_interface_roofs", roofs, SectionType::SUPPORT, layer_idx); } @@ -1746,7 +1750,7 @@ void AreaSupport::generateSupportRoof(SliceDataStorage& storage, const SliceMesh int lower = static_cast(layer_idx); int upper = std::min(static_cast(layer_idx + roof_layer_count + z_distance_top + 5), static_cast(global_support_areas_per_layer.size()) - 1); - for (Polygons& global_support : global_support_areas_per_layer | ranges::views::slice(lower, upper)) + for (Shape& global_support : global_support_areas_per_layer | ranges::views::slice(lower, upper)) { global_support = global_support.difference(support_layer.support_roof); } @@ -1754,14 +1758,14 @@ void AreaSupport::generateSupportRoof(SliceDataStorage& storage, const SliceMesh } void AreaSupport::generateSupportInterfaceLayer( - Polygons& support_areas, - const Polygons colliding_mesh_outlines, + Shape& support_areas, + const Shape colliding_mesh_outlines, const coord_t safety_offset, const coord_t outline_offset, const double minimum_interface_area, - Polygons& interface_polygons) + Shape& interface_polygons) { - Polygons model = colliding_mesh_outlines.unionPolygons(); + Shape model = colliding_mesh_outlines.unionPolygons(); interface_polygons = support_areas.offset(safety_offset / 2).intersection(model); interface_polygons = interface_polygons.offset(safety_offset).intersection(support_areas); // Make sure we don't generate any models that are not printable. if (outline_offset != 0) diff --git a/src/utils/AABB.cpp b/src/utils/AABB.cpp index f3a93cbf1d..6707775c3f 100644 --- a/src/utils/AABB.cpp +++ b/src/utils/AABB.cpp @@ -5,8 +5,9 @@ #include +#include "geometry/Polygon.h" +#include "geometry/Shape.h" #include "utils/linearAlg2D.h" -#include "utils/polygon.h" //To create the AABB of a polygon. namespace cura { @@ -24,14 +25,14 @@ AABB::AABB(const Point2LL& min, const Point2LL& max) { } -AABB::AABB(const Polygons& polys) +AABB::AABB(const Shape& shape) : min_(POINT_MAX, POINT_MAX) , max_(POINT_MIN, POINT_MIN) { - calculate(polys); + calculate(shape); } -AABB::AABB(ConstPolygonRef poly) +AABB::AABB(const Polygon& poly) : min_(POINT_MAX, POINT_MAX) , max_(POINT_MIN, POINT_MIN) { @@ -68,20 +69,20 @@ coord_t AABB::distanceSquared(const AABB& other) const }); } -void AABB::calculate(const Polygons& polys) +void AABB::calculate(const Shape& shape) { min_ = Point2LL(POINT_MAX, POINT_MAX); max_ = Point2LL(POINT_MIN, POINT_MIN); - for (unsigned int i = 0; i < polys.size(); i++) + for (const Polygon& poly : shape) { - for (unsigned int j = 0; j < polys[i].size(); j++) + for (const Point2LL& point : poly) { - include(polys[i][j]); + include(point); } } } -void AABB::calculate(ConstPolygonRef poly) +void AABB::calculate(const Polygon& poly) { min_ = Point2LL(POINT_MAX, POINT_MAX); max_ = Point2LL(POINT_MIN, POINT_MIN); @@ -131,7 +132,7 @@ bool AABB::hit(const AABB& other) const return true; } -void AABB::include(Point2LL point) +void AABB::include(const Point2LL& point) { min_.X = std::min(min_.X, point.X); min_.Y = std::min(min_.Y, point.Y); @@ -139,7 +140,15 @@ void AABB::include(Point2LL point) max_.Y = std::max(max_.Y, point.Y); } -void AABB::include(const AABB other) +void AABB::include(const Polygon& polygon) +{ + for (const Point2LL& point : polygon) + { + include(point); + } +} + +void AABB::include(const AABB& other) { // Note that this is different from including the min and max points, since when 'min > max' it's used to denote an negative/empty box. min_.X = std::min(min_.X, other.min_.X); @@ -162,12 +171,7 @@ void AABB::expand(int dist) Polygon AABB::toPolygon() const { - Polygon ret; - ret.add(min_); - ret.add(Point2LL(max_.X, min_.Y)); - ret.add(max_); - ret.add(Point2LL(min_.X, max_.Y)); - return ret; + return Polygon({ min_, Point2LL(max_.X, min_.Y), max_, Point2LL(min_.X, max_.Y) }, false); } } // namespace cura diff --git a/src/utils/ExtrusionLine.cpp b/src/utils/ExtrusionLine.cpp index 5183aab67b..0da925d771 100644 --- a/src/utils/ExtrusionLine.cpp +++ b/src/utils/ExtrusionLine.cpp @@ -12,7 +12,7 @@ namespace cura { -coord_t ExtrusionLine::getLength() const +coord_t ExtrusionLine::length() const { if (junctions_.empty()) { @@ -44,4 +44,20 @@ coord_t ExtrusionLine::getMinimalWidth() const ->w_; } +bool ExtrusionLine::shorterThan(const coord_t check_length) const +{ + const ExtrusionJunction* p0 = &back(); + int64_t length = 0; + for (const ExtrusionJunction& p1 : (*this)) + { + length += vSize(*p0 - p1); + if (length >= check_length) + { + return false; + } + p0 = &p1; + } + return true; +} + } // namespace cura diff --git a/src/utils/ExtrusionSegment.cpp b/src/utils/ExtrusionSegment.cpp index 71a2d59bef..0327f333db 100644 --- a/src/utils/ExtrusionSegment.cpp +++ b/src/utils/ExtrusionSegment.cpp @@ -1,8 +1,10 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "utils/ExtrusionSegment.h" +#include + #include #include "utils/macros.h" @@ -10,14 +12,14 @@ namespace cura { -Polygons ExtrusionSegment::toPolygons() +Shape ExtrusionSegment::toShape() { - return toPolygons(is_reduced_); + return toShape(is_reduced_); } -Polygons ExtrusionSegment::toPolygons(bool reduced) +Shape ExtrusionSegment::toShape(bool reduced) { - Polygons ret; + Shape ret; const Point2LL vec = to_.p_ - from_.p_; const coord_t vec_length = vSize(vec); @@ -26,7 +28,7 @@ Polygons ExtrusionSegment::toPolygons(bool reduced) return ret; } - PolygonRef poly = ret.newPoly(); + Polygon& poly = ret.newLine(); const double delta_r = 0.5 * std::abs(from_.w_ - to_.w_); const double vec_length_fixed = std::max(delta_r, static_cast(vec_length)); float alpha = std::acos(delta_r / vec_length_fixed); // Angle between the slope along the edge of the polygon (due to varying line width) and the centerline. @@ -49,7 +51,7 @@ Polygons ExtrusionSegment::toPolygons(bool reduced) // Draw the endcap on the "from" vertex's end. { - poly.emplace_back(from_.p_ + Point2LL(from_.w_ / 2 * cos(alpha + dir), from_.w_ / 2 * sin(alpha + dir))); + poly.push_back(from_.p_ + Point2LL(from_.w_ / 2 * cos(alpha + dir), from_.w_ / 2 * sin(alpha + dir))); double start_a = 2 * std::numbers::pi; while (start_a > alpha + dir) @@ -74,7 +76,7 @@ Polygons ExtrusionSegment::toPolygons(bool reduced) // Draw the endcap on the "to" vertex's end. { - poly.emplace_back( + poly.push_back( to_.p_ + Point2LL( to_.w_ / 2 * cos(2 * std::numbers::pi - alpha + dir), @@ -125,7 +127,7 @@ Polygons ExtrusionSegment::toPolygons(bool reduced) } #ifdef DEBUG - for (Point2LL p : poly) + for (const Point2LL& p : poly) { assert(p.X < 0x3FFFFFFFFFFFFFFFLL); assert(p.Y < 0x3FFFFFFFFFFFFFFFLL); diff --git a/src/utils/ListPolyIt.cpp b/src/utils/ListPolyIt.cpp index c2c4a7cf0a..f56159af5d 100644 --- a/src/utils/ListPolyIt.cpp +++ b/src/utils/ListPolyIt.cpp @@ -6,6 +6,7 @@ #include // isfinite #include // ostream +#include "geometry/Polygon.h" #include "utils/AABB.h" // for debug output svg html #include "utils/SVG.h" @@ -13,16 +14,16 @@ namespace cura { -void ListPolyIt::convertPolygonsToLists(const Polygons& polys, ListPolygons& result) +void ListPolyIt::convertPolygonsToLists(const Shape& shape, ListPolygons& result) { - for (ConstPolygonRef poly : polys) + for (const Polygon& poly : shape) { result.emplace_back(); convertPolygonToList(poly, result.back()); } } -void ListPolyIt::convertPolygonToList(ConstPolygonRef poly, ListPolygon& result) +void ListPolyIt::convertPolygonToList(const Polygon& poly, ListPolygon& result) { #ifdef DEBUG Point2LL last = poly.back(); @@ -41,7 +42,7 @@ void ListPolyIt::convertPolygonToList(ConstPolygonRef poly, ListPolygon& result) } -void ListPolyIt::convertListPolygonsToPolygons(const ListPolygons& list_polygons, Polygons& polygons) +void ListPolyIt::convertListPolygonsToPolygons(const ListPolygons& list_polygons, Shape& polygons) { for (unsigned int poly_idx = 0; poly_idx < polygons.size(); poly_idx++) { @@ -50,11 +51,11 @@ void ListPolyIt::convertListPolygonsToPolygons(const ListPolygons& list_polygons } } -void ListPolyIt::convertListPolygonToPolygon(const ListPolygon& list_polygon, PolygonRef polygon) +void ListPolyIt::convertListPolygonToPolygon(const ListPolygon& list_polygon, Polygon& polygon) { for (const Point2LL& p : list_polygon) { - polygon.add(p); + polygon.push_back(p); } } diff --git a/src/utils/Matrix4x3D.cpp b/src/utils/Matrix4x3D.cpp index 43b255379f..7e68fbfbbc 100644 --- a/src/utils/Matrix4x3D.cpp +++ b/src/utils/Matrix4x3D.cpp @@ -3,8 +3,8 @@ #include "utils/Matrix4x3D.h" //The definitions we're implementing. +#include "geometry/Point2LL.h" //Conversion directly into integer-based coordinates. #include "settings/types/Ratio.h" //Scale factor. -#include "utils/Point2LL.h" //Conversion directly into integer-based coordinates. #include "utils/Point3D.h" //This matrix gets applied to floating point coordinates. namespace cura diff --git a/src/utils/MixedPolylineStitcher.cpp b/src/utils/MixedPolylineStitcher.cpp new file mode 100644 index 0000000000..50e49a9323 --- /dev/null +++ b/src/utils/MixedPolylineStitcher.cpp @@ -0,0 +1,34 @@ +// Copyright (c) 2024 UltiMaker +// CuraEngine is released under the terms of the AGPLv3 or higher. + +#include "utils/MixedPolylineStitcher.h" + +#include "geometry/MixedLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" +#include "geometry/Shape.h" + + +namespace cura +{ + +void MixedPolylineStitcher::stitch(const OpenLinesSet& lines, MixedLinesSet& result, coord_t max_stitch_distance, coord_t snap_distance) +{ + OpenLinesSet open_lines; + ClosedLinesSet closed_lines; + + PolylineStitcher::stitch(lines, open_lines, closed_lines, max_stitch_distance, snap_distance); + + result.push_back(std::move(open_lines)); + + for (ClosedPolyline& closed_line : closed_lines) + { + // Base stitch method will create explicitely closed polylines, but won't tag them as such + // because it is a generic algorithm. Tag them now. + closed_line.setExplicitelyClosed(true); + } + + result.push_back(std::move(closed_lines)); +} + +} // namespace cura diff --git a/src/utils/Point3LL.cpp b/src/utils/Point3LL.cpp index 2420552fb6..bb8e7c6930 100644 --- a/src/utils/Point3LL.cpp +++ b/src/utils/Point3LL.cpp @@ -1,7 +1,7 @@ // Copyright (c) 2022 Ultimaker B.V. // CuraEngine is released under the terms of the AGPLv3 or higher. -#include "utils/Point3LL.h" //The headers we're implementing. +#include "geometry/Point3LL.h" //The headers we're implementing. namespace cura { @@ -63,14 +63,4 @@ Point3LL& Point3LL::operator/=(const Point3LL& p) return *this; } -bool Point3LL::operator==(const Point3LL& p) const -{ - return x_ == p.x_ && y_ == p.y_ && z_ == p.z_; -} - -bool Point3LL::operator!=(const Point3LL& p) const -{ - return x_ != p.x_ || y_ != p.y_ || z_ != p.z_; -} - } // namespace cura diff --git a/src/utils/PolygonConnector.cpp b/src/utils/PolygonConnector.cpp index 936ad69850..e8f3f3c33b 100644 --- a/src/utils/PolygonConnector.cpp +++ b/src/utils/PolygonConnector.cpp @@ -14,9 +14,9 @@ PolygonConnector::PolygonConnector(const coord_t line_width) { } -void PolygonConnector::add(const Polygons& input) +void PolygonConnector::add(const Shape& input) { - for (ConstPolygonRef poly : input) + for (const Polygon& poly : input) { input_polygons_.push_back(poly); } @@ -33,12 +33,12 @@ void PolygonConnector::add(const std::vector& input) } } -void PolygonConnector::connect(Polygons& output_polygons, std::vector& output_paths) +void PolygonConnector::connect(Shape& output_polygons, std::vector& output_paths) { std::vector result_polygons = connectGroup(input_polygons_); - for (Polygon& polygon : result_polygons) + for (const Polygon& polygon : result_polygons) { - output_polygons.add(polygon); + output_polygons.push_back(polygon); } std::vector result_paths = connectGroup(input_paths_); @@ -67,12 +67,12 @@ coord_t PolygonConnector::getWidth(const ExtrusionJunction& junction) const void PolygonConnector::addVertex(Polygon& polygonal, const Point2LL& position, const coord_t) const { - polygonal.add(position); + polygonal.push_back(position); } void PolygonConnector::addVertex(Polygon& polygonal, const Point2LL& vertex) const { - polygonal.add(vertex); + polygonal.push_back(vertex); } void PolygonConnector::addVertex(ExtrusionLine& polygonal, const Point2LL& position, const coord_t width) const diff --git a/src/utils/PolygonsPointIndex.cpp b/src/utils/PolygonsPointIndex.cpp index d41092baba..3618e53a96 100644 --- a/src/utils/PolygonsPointIndex.cpp +++ b/src/utils/PolygonsPointIndex.cpp @@ -3,13 +3,24 @@ #include "utils/PolygonsPointIndex.h" +#include "geometry/Shape.h" + namespace cura { template<> -ConstPolygonRef PathsPointIndex::getPolygon() const +const Polygon& PathsPointIndex::getPolygon() const { return (*polygons_)[poly_idx_]; } +std::pair PolygonsPointIndexSegmentLocator::operator()(const PolygonsPointIndex& val) const +{ + const Polygon& poly = (*val.polygons_)[val.poly_idx_]; + Point2LL start = poly[val.point_idx_]; + size_t next_point_idx = (val.point_idx_ + 1ul) % poly.size(); + Point2LL end = poly[next_point_idx]; + return std::pair(start, end); +} + } // namespace cura diff --git a/src/utils/PolygonsSegmentIndex.cpp b/src/utils/PolygonsSegmentIndex.cpp index 0b402a03f6..46f226a4c7 100644 --- a/src/utils/PolygonsSegmentIndex.cpp +++ b/src/utils/PolygonsSegmentIndex.cpp @@ -11,7 +11,7 @@ PolygonsSegmentIndex::PolygonsSegmentIndex() { } -PolygonsSegmentIndex::PolygonsSegmentIndex(const Polygons* polygons, unsigned int poly_idx, unsigned int point_idx) +PolygonsSegmentIndex::PolygonsSegmentIndex(const Shape* polygons, unsigned int poly_idx, unsigned int point_idx) : PolygonsPointIndex(polygons, poly_idx, point_idx) { } diff --git a/src/utils/PolylineStitcher.cpp b/src/utils/PolylineStitcher.cpp index 61a3b11594..6e67797c62 100644 --- a/src/utils/PolylineStitcher.cpp +++ b/src/utils/PolylineStitcher.cpp @@ -3,14 +3,18 @@ #include "utils/PolylineStitcher.h" -#include "utils/ExtrusionLine.h" -#include "utils/polygon.h" +#include "geometry/ClosedLinesSet.h" +#include "geometry/OpenLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "utils/ExtrusionLineStitcher.h" +#include "utils/OpenPolylineStitcher.h" +#include "utils/PolygonsPointIndex.h" namespace cura { template<> -bool PolylineStitcher::canReverse(const PathsPointIndex& ppi) +bool ExtrusionLineStitcher::canReverse(const PathsPointIndex& ppi) { if ((*ppi.polygons_)[ppi.poly_idx_].is_odd_) { @@ -23,33 +27,271 @@ bool PolylineStitcher::can } template<> -bool PolylineStitcher::canReverse(const PathsPointIndex&) +bool OpenPolylineStitcher::canReverse(const PathsPointIndex&) { return true; } template<> -bool PolylineStitcher::canConnect(const ExtrusionLine& a, const ExtrusionLine& b) +bool PolylineStitcher::canReverse(const PathsPointIndex&) +{ + return true; +} + +template<> +bool ExtrusionLineStitcher::canConnect(const ExtrusionLine& a, const ExtrusionLine& b) { return a.is_odd_ == b.is_odd_; } template<> -bool PolylineStitcher::canConnect(const Polygon&, const Polygon&) +bool OpenPolylineStitcher::canConnect(const OpenPolyline&, const OpenPolyline&) { return true; } template<> -bool PolylineStitcher::isOdd(const ExtrusionLine& line) +bool PolylineStitcher::canConnect(const OpenPolyline&, const OpenPolyline&) +{ + return true; +} + +template<> +bool ExtrusionLineStitcher::isOdd(const ExtrusionLine& line) { return line.is_odd_; } template<> -bool PolylineStitcher::isOdd(const Polygon&) +bool OpenPolylineStitcher::isOdd(const OpenPolyline&) +{ + return false; +} + +template<> +bool PolylineStitcher::isOdd(const OpenPolyline&) { return false; } +template<> +void ExtrusionLineStitcher::pushToClosedResult(VariableWidthLines& result_polygons, const ExtrusionLine& polyline) +{ + result_polygons.push_back(polyline); +} + +template<> +void OpenPolylineStitcher::pushToClosedResult(Shape& result_polygons, const OpenPolyline& polyline) +{ + result_polygons.emplace_back(polyline.getPoints(), true); +} + +template<> +void PolylineStitcher::pushToClosedResult(ClosedLinesSet& result_polygons, const OpenPolyline& polyline) +{ + result_polygons.emplace_back(polyline.getPoints(), true); +} + +template +void PolylineStitcher::stitch( + const InputPaths& lines, + InputPaths& result_lines, + OutputPaths& result_polygons, + coord_t max_stitch_distance, + coord_t snap_distance) +{ + if (lines.empty()) + { + return; + } + + SparsePointGrid, PathsPointIndexLocator> grid(max_stitch_distance, lines.size() * 2); + + // populate grid + for (size_t line_idx = 0; line_idx < lines.size(); line_idx++) + { + const auto line = lines[line_idx]; + grid.insert(PathsPointIndex(&lines, line_idx, 0)); + grid.insert(PathsPointIndex(&lines, line_idx, line.size() - 1)); + } + + std::vector processed(lines.size(), false); + + for (size_t line_idx = 0; line_idx < lines.size(); line_idx++) + { + if (processed[line_idx]) + { + continue; + } + processed[line_idx] = true; + const auto line = lines[line_idx]; + bool should_close = isOdd(line); + + Path chain = line; + bool closest_is_closing_polygon = false; + for (bool go_in_reverse_direction : { false, true }) // first go in the unreversed direction, to try to prevent the chain.reverse() operation. + { // NOTE: Implementation only works for this order; we currently only re-reverse the chain when it's closed. + if (go_in_reverse_direction) + { // try extending chain in the other direction + chain.reverse(); + } + coord_t chain_length = chain.length(); + + while (true) + { + Point2LL from = make_point(chain.back()); + + PathsPointIndex closest; + coord_t closest_distance = std::numeric_limits::max(); + grid.processNearby( + from, + max_stitch_distance, + std::function&)>( + [from, + &chain, + &closest, + &closest_is_closing_polygon, + &closest_distance, + &processed, + &chain_length, + go_in_reverse_direction, + max_stitch_distance, + snap_distance, + should_close](const PathsPointIndex& nearby) -> bool + { + bool is_closing_segment = false; + coord_t dist = vSize(nearby.p() - from); + if (dist > max_stitch_distance) + { + return true; // keep looking + } + if (vSize2(nearby.p() - make_point(chain.front())) < snap_distance * snap_distance) + { + if (chain_length + dist < 3 * max_stitch_distance // prevent closing of small poly, cause it might be able to continue making a larger polyline + || chain.size() <= 2) // don't make 2 vert polygons + { + return true; // look for a better next line + } + is_closing_segment = true; + if (! should_close) + { + dist += 10; // prefer continuing polyline over closing a polygon; avoids closed zigzags from being printed separately + // continue to see if closing segment is also the closest + // there might be a segment smaller than [max_stitch_distance] which closes the polygon better + } + else + { + dist -= 10; // Prefer closing the polygon if it's 100% even lines. Used to create closed contours. + // Continue to see if closing segment is also the closest. + } + } + else if (processed[nearby.poly_idx_]) + { // it was already moved to output + return true; // keep looking for a connection + } + bool nearby_would_be_reversed = nearby.point_idx_ != 0; + nearby_would_be_reversed = nearby_would_be_reversed != go_in_reverse_direction; // flip nearby_would_be_reversed when searching in the reverse direction + if (! canReverse(nearby) && nearby_would_be_reversed) + { // connecting the segment would reverse the polygon direction + return true; // keep looking for a connection + } + if (! canConnect(chain, (*nearby.polygons_)[nearby.poly_idx_])) + { + return true; // keep looking for a connection + } + if (dist < closest_distance) + { + closest_distance = dist; + closest = nearby; + closest_is_closing_polygon = is_closing_segment; + } + if (dist < snap_distance) + { // we have found a good enough next line + return false; // stop looking for alternatives + } + return true; // keep processing elements + })); + + if (! closest.initialized() // we couldn't find any next line + || closest_is_closing_polygon // we closed the polygon + ) + { + break; + } + + + coord_t segment_dist = vSize(make_point(chain.back()) - closest.p()); + assert(segment_dist <= max_stitch_distance + 10); + const size_t old_size = chain.size(); + if (closest.point_idx_ == 0) + { + auto start_pos = (*closest.polygons_)[closest.poly_idx_].begin(); + if (segment_dist < snap_distance) + { + ++start_pos; + } + chain.insert(chain.end(), start_pos, (*closest.polygons_)[closest.poly_idx_].end()); + } + else + { + auto start_pos = (*closest.polygons_)[closest.poly_idx_].rbegin(); + if (segment_dist < snap_distance) + { + ++start_pos; + } + chain.insert(chain.end(), start_pos, (*closest.polygons_)[closest.poly_idx_].rend()); + } + for (size_t i = old_size; i < chain.size(); ++i) // Update chain length. + { + chain_length += vSize(chain[i] - chain[i - 1]); + } + should_close = should_close & ! isOdd((*closest.polygons_)[closest.poly_idx_]); // If we connect an even to an odd line, we should no longer try to close it. + assert(! processed[closest.poly_idx_]); + processed[closest.poly_idx_] = true; + } + + if (closest_is_closing_polygon) + { + if (go_in_reverse_direction) + { // re-reverse chain to retain original direction + // NOTE: not sure if this code could ever be reached, since if a polygon can be closed that should be already possible in the forward direction + chain.reverse(); + } + + break; // don't consider reverse direction + } + } + if (closest_is_closing_polygon) + { + pushToClosedResult(result_polygons, chain); + } + else + { + PathsPointIndex ppi_here(&lines, line_idx, 0); + if (! canReverse(ppi_here)) + { // Since closest_is_closing_polygon is false we went through the second iterations of the for-loop, where go_in_reverse_direction is true + // the polyline isn't allowed to be reversed, so we re-reverse it. + chain.reverse(); + } + result_lines.emplace_back(chain); + } + } +} + +template void OpenPolylineStitcher::stitch(const OpenLinesSet& lines, OpenLinesSet& result_lines, Shape& result_polygons, coord_t max_stitch_distance, coord_t snap_distance); + +template void ExtrusionLineStitcher::stitch( + const VariableWidthLines& lines, + VariableWidthLines& result_lines, + VariableWidthLines& result_polygons, + coord_t max_stitch_distance, + coord_t snap_distance); + +template void PolylineStitcher::stitch( + const OpenLinesSet& lines, + OpenLinesSet& result_lines, + ClosedLinesSet& result_polygons, + coord_t max_stitch_distance, + coord_t snap_distance); + } // namespace cura diff --git a/src/utils/SVG.cpp b/src/utils/SVG.cpp index 4da036996a..69a115d9d4 100644 --- a/src/utils/SVG.cpp +++ b/src/utils/SVG.cpp @@ -7,9 +7,10 @@ #include +#include "geometry/Polygon.h" +#include "geometry/SingleShape.h" #include "utils/ExtrusionLine.h" #include "utils/Point3D.h" -#include "utils/polygon.h" namespace cura { @@ -87,7 +88,7 @@ SVG::SVG(std::string filename, AABB aabb, double scale, Point2LL canvas_size, Co out_ = fopen(filename.c_str(), "w"); if (! out_) { - spdlog::error("The file %s could not be opened for writing.", filename); + spdlog::error("The file {} could not be opened for writing.", filename); } if (output_is_html_) { @@ -155,12 +156,12 @@ void SVG::writeComment(const std::string& comment) const fprintf(out_, "\n", comment.c_str()); } -void SVG::writeAreas(const Polygons& polygons, const ColorObject color, const ColorObject outline_color, const double stroke_width) const +void SVG::writeAreas(const Shape& polygons, const ColorObject color, const ColorObject outline_color, const double stroke_width) const { - std::vector parts = polygons.splitIntoParts(); + std::vector parts = polygons.splitIntoParts(); for (auto part_it = parts.rbegin(); part_it != parts.rend(); ++part_it) { - PolygonsPart& part = *part_it; + SingleShape& part = *part_it; for (size_t j = 0; j < part.size(); j++) { fprintf(out_, " #include //Priority queue to prioritise removing unimportant vertices. +#include "geometry/ClosedPolyline.h" +#include "geometry/MixedLinesSet.h" +#include "geometry/OpenPolyline.h" +#include "settings/Settings.h" //To load the parameters from a Settings object. +#include "utils/ExtrusionLine.h" +#include "utils/linearAlg2D.h" //To calculate line deviations and intersecting lines. + namespace cura { @@ -23,12 +30,12 @@ Simplify::Simplify(const Settings& settings) { } -Polygons Simplify::polygon(const Polygons& polygons) const +Shape Simplify::polygon(const Shape& polygons) const { - Polygons result; + Shape result; for (size_t i = 0; i < polygons.size(); ++i) { - result.addIfNotEmpty(polygon(polygons[i])); + result.push_back(polygon(polygons[i]), CheckNonEmptyParam::OnlyIfNotEmpty); } return result; } @@ -45,17 +52,40 @@ ExtrusionLine Simplify::polygon(const ExtrusionLine& polygon) const return simplify(polygon, is_closed); } -Polygons Simplify::polyline(const Polygons& polylines) const +template +LinesSet Simplify::polyline(const LinesSet& polylines) const { - Polygons result; + LinesSet result; for (size_t i = 0; i < polylines.size(); ++i) { - result.addIfNotEmpty(polyline(polylines[i])); + result.push_back(polyline(polylines[i]), CheckNonEmptyParam::OnlyIfNotEmpty); + } + return result; +} + +MixedLinesSet Simplify::polyline(const MixedLinesSet& polylines) const +{ + MixedLinesSet result; + for (const PolylinePtr& polyline_ptr : polylines) + { + if (std::shared_ptr open_polyline = std::dynamic_pointer_cast(polyline_ptr)) + { + result.push_back(std::make_shared(polyline(*open_polyline))); + } + if (std::shared_ptr closed_polyline = std::dynamic_pointer_cast(polyline_ptr)) + { + result.push_back(std::make_shared(polyline(*closed_polyline))); + } } return result; } -Polygon Simplify::polyline(const Polygon& polyline) const +ClosedPolyline Simplify::polyline(const ClosedPolyline& polyline) const +{ + return simplify(polyline, polyline.isExplicitelyClosed()); +} + +OpenPolyline Simplify::polyline(const OpenPolyline& polyline) const { constexpr bool is_closed = false; return simplify(polyline, is_closed); @@ -83,21 +113,23 @@ size_t Simplify::previousNotDeleted(size_t index, const std::vector& to_de return index; } -Polygon Simplify::createEmpty([[maybe_unused]] const Polygon& original) const -{ - return Polygon(); -} - -ExtrusionLine Simplify::createEmpty(const ExtrusionLine& original) const +template<> +ExtrusionLine Simplify::createEmpty(const ExtrusionLine& original) { ExtrusionLine result(original.inset_idx_, original.is_odd_); result.is_closed_ = original.is_closed_; return result; } -void Simplify::appendVertex(Polygon& polygon, const Point2LL& vertex) const +template +Polygonal Simplify::createEmpty(const Polygonal& /*original*/) +{ + return Polygonal(); +} + +void Simplify::appendVertex(Polyline& polygon, const Point2LL& vertex) const { - polygon.add(vertex); + polygon.push_back(vertex); } void Simplify::appendVertex(ExtrusionLine& extrusion_line, const ExtrusionJunction& vertex) const @@ -176,4 +208,207 @@ coord_t Simplify::getAreaDeviation(const ExtrusionJunction& before, const Extrus } } +template +bool Simplify::detectSmall(const Polygonal& polygon, const coord_t& min_size) const +{ + if (polygon.size() < min_size) // For polygon, 2 or fewer vertices is degenerate. Delete it. For polyline, 1 vertex is degenerate. + { + return true; + } + if (polygon.size() == min_size) + { + const auto a = getPosition(polygon[0]); + const auto b = getPosition(polygon[1]); + const auto c = getPosition(polygon[polygon.size() - 1]); + if (std::max(std::max(vSize2(b - a), vSize2(c - a)), vSize2(c - b)) < min_resolution * min_resolution) + { + // ... unless they are degenetate. + return true; + } + } + return false; +} + +template +Polygonal Simplify::simplify(const Polygonal& polygon, const bool is_closed) const +{ + const size_t min_size = is_closed ? 3 : 2; + if (detectSmall(polygon, min_size)) + { + return createEmpty(polygon); + } + if (polygon.size() == min_size) // For polygon, don't reduce below 3. For polyline, not below 2. + { + return polygon; + } + + std::vector to_delete(polygon.size(), false); + auto comparator = [](const std::pair& vertex_a, const std::pair& vertex_b) + { + return vertex_a.second > vertex_b.second || (vertex_a.second == vertex_b.second && vertex_a.first > vertex_b.first); + }; + std::priority_queue, std::vector>, decltype(comparator)> by_importance(comparator); + + Polygonal result = polygon; // Make a copy so that we can also shift vertices. + for (int64_t current_removed = -1; (polygon.size() - current_removed) > min_size && current_removed != 0;) + { + current_removed = 0; + + // Add the initial points. + for (size_t i = 0; i < result.size(); ++i) + { + if (to_delete[i]) + { + continue; + } + const coord_t vertex_importance = importance(result, to_delete, i, is_closed); + by_importance.emplace(i, vertex_importance); + } + + // Iteratively remove the least important point until a threshold. + coord_t vertex_importance = 0; + while (! by_importance.empty() && (polygon.size() - current_removed) > min_size) + { + std::pair vertex = by_importance.top(); + by_importance.pop(); + // The importance may have changed since this vertex was inserted. Re-compute it now. + // If it doesn't change, it's safe to process. + vertex_importance = importance(result, to_delete, vertex.first, is_closed); + if (vertex_importance != vertex.second) + { + by_importance.emplace(vertex.first, vertex_importance); // Re-insert with updated importance. + continue; + } + + if (vertex_importance <= max_deviation_ * max_deviation_) + { + current_removed += remove(result, to_delete, vertex.first, vertex_importance, is_closed) ? 1 : 0; + } + } + } + + // Now remove the marked vertices in one sweep. + Polygonal filtered = createEmpty(polygon); + for (size_t i = 0; i < result.size(); ++i) + { + if (! to_delete[i]) + { + appendVertex(filtered, result[i]); + } + } + + if (detectSmall(filtered, min_size)) + { + return createEmpty(filtered); + } + return filtered; +} + +template +coord_t Simplify::importance(const Polygonal& polygon, const std::vector& to_delete, const size_t index, const bool is_closed) const +{ + const size_t poly_size = polygon.size(); + if (! is_closed && (index == 0 || index == poly_size - 1)) + { + return std::numeric_limits::max(); // Endpoints of the polyline must always be retained. + } + // From here on out we can safely look at the vertex neighbors and assume it's a polygon. We won't go out of bounds of the polyline. + + const Point2LL& vertex = getPosition(polygon[index]); + const size_t before_index = previousNotDeleted(index, to_delete); + const size_t after_index = nextNotDeleted(index, to_delete); + + const coord_t area_deviation = getAreaDeviation(polygon[before_index], polygon[index], polygon[after_index]); + if (area_deviation > max_area_deviation_) // Removing this line causes the variable line width to get flattened out too much. + { + return std::numeric_limits::max(); + } + + const Point2LL& before = getPosition(polygon[before_index]); + const Point2LL& after = getPosition(polygon[after_index]); + const coord_t deviation2 = LinearAlg2D::getDist2FromLine(vertex, before, after); + if (deviation2 <= min_resolution * min_resolution) // Deviation so small that it's always desired to remove them. + { + return deviation2; + } + if (vSize2(before - vertex) > max_resolution_ * max_resolution_ && vSize2(after - vertex) > max_resolution_ * max_resolution_) + { + return std::numeric_limits::max(); // Long line segments, no need to remove this one. + } + return deviation2; +} + +template +bool Simplify::remove(Polygonal& polygon, std::vector& to_delete, const size_t vertex, const coord_t deviation2, const bool is_closed) const +{ + if (deviation2 <= min_resolution * min_resolution) + { + // At less than the minimum resolution we're always allowed to delete the vertex. + // Even if the adjacent line segments are very long. + to_delete[vertex] = true; + return true; + } + + const size_t before = previousNotDeleted(vertex, to_delete); + const size_t after = nextNotDeleted(vertex, to_delete); + const Point2LL& vertex_position = getPosition(polygon[vertex]); + const Point2LL& before_position = getPosition(polygon[before]); + const Point2LL& after_position = getPosition(polygon[after]); + const coord_t length2_before = vSize2(vertex_position - before_position); + const coord_t length2_after = vSize2(vertex_position - after_position); + + if (length2_before <= max_resolution_ * max_resolution_ && length2_after <= max_resolution_ * max_resolution_) // Both adjacent line segments are short. + { + // Removing this vertex does little harm. No long lines will be shifted. + to_delete[vertex] = true; + return true; + } + + // Otherwise, one edge next to this vertex is longer than max_resolution. The other is shorter. + // In this case we want to remove the short edge by replacing it with a vertex where the two surrounding edges intersect. + // Find the two line segments surrounding the short edge here ("before" and "after" edges). + Point2LL before_from, before_to, after_from, after_to; + if (length2_before <= length2_after) // Before is the shorter line. + { + if (! is_closed && before == 0) // No edge before the short edge. + { + return false; // Edge cannot be deleted without shifting a long edge. Don't remove anything. + } + const size_t before_before = previousNotDeleted(before, to_delete); + before_from = getPosition(polygon[before_before]); + before_to = getPosition(polygon[before]); + after_from = getPosition(polygon[vertex]); + after_to = getPosition(polygon[after]); + } + else + { + if (! is_closed && after == polygon.size() - 1) // No edge after the short edge. + { + return false; // Edge cannot be deleted without shifting a long edge. Don't remove anything. + } + const size_t after_after = nextNotDeleted(after, to_delete); + before_from = getPosition(polygon[before]); + before_to = getPosition(polygon[vertex]); + after_from = getPosition(polygon[after]); + after_to = getPosition(polygon[after_after]); + } + Point2LL intersection; + const bool did_intersect = LinearAlg2D::lineLineIntersection(before_from, before_to, after_from, after_to, intersection); + if (! did_intersect) // Lines are parallel. + { + return false; // Cannot remove edge without shifting a long edge. Don't remove anything. + } + const coord_t intersection_deviation = LinearAlg2D::getDist2FromLineSegment(before_to, intersection, after_from); + if (intersection_deviation <= max_deviation_ * max_deviation_) // Intersection point doesn't deviate too much. Use it! + { + to_delete[vertex] = true; + polygon[length2_before <= length2_after ? before : after] = createIntersection(polygon[before], intersection, polygon[after]); + return true; + } + return false; +} + +template OpenLinesSet Simplify::polyline(const OpenLinesSet& polylines) const; +template ClosedLinesSet Simplify::polyline(const ClosedLinesSet& polylines) const; + } // namespace cura diff --git a/src/utils/ToolpathVisualizer.cpp b/src/utils/ToolpathVisualizer.cpp index 34526f8dca..88161c78c7 100644 --- a/src/utils/ToolpathVisualizer.cpp +++ b/src/utils/ToolpathVisualizer.cpp @@ -8,7 +8,7 @@ namespace cura { -void ToolpathVisualizer::outline(const Polygons& input) +void ToolpathVisualizer::outline(const Shape& input) { svg_.writeAreas(input, SVG::Color::GRAY, SVG::Color::NONE, 2); svg_.nextLayer(); @@ -18,20 +18,20 @@ void ToolpathVisualizer::toolpaths(const std::vector& all_segm { for (double w = .9; w > .25; w = 1.0 - (1.0 - w) * 1.2) { - Polygons polys; + Shape polys; for (size_t segment_idx = 0; segment_idx < all_segments.size(); segment_idx++) { ExtrusionSegment s = all_segments[segment_idx]; s.from_.w_ *= w / .9; s.to_.w_ *= w / .9; - Polygons covered = s.toPolygons(false); - polys.add(covered); + Shape covered = s.toShape(false); + polys.push_back(covered); } int c = 255 - 200 * (w - .25); SVG::ColorObject clr(c, c, c); polys = polys.execute(ClipperLib::pftNonZero); polys = PolygonUtils::connect(polys); - for (PolygonRef connected : polys) + for (const Polygon& connected : polys) svg_.writeAreas(connected, clr, SVG::Color::NONE); if (! rounded_visualization) break; @@ -40,12 +40,12 @@ void ToolpathVisualizer::toolpaths(const std::vector& all_segm } -void ToolpathVisualizer::underfill(const Polygons& underfills) +void ToolpathVisualizer::underfill(const Shape& underfills) { svg_.writeAreas(underfills, SVG::ColorObject(0, 128, 255), SVG::Color::NONE); svg_.nextLayer(); } -void ToolpathVisualizer::overfill(const Polygons& overfills, const Polygons& double_overfills) +void ToolpathVisualizer::overfill(const Shape& overfills, const Shape& double_overfills) { svg_.writeAreas(overfills, SVG::ColorObject(255, 128, 0), SVG::Color::NONE); svg_.nextLayer(); @@ -56,7 +56,7 @@ void ToolpathVisualizer::overfill(const Polygons& overfills, const Polygons& dou } } -void ToolpathVisualizer::width_legend(const Polygons& input, coord_t nozzle_size, coord_t max_dev, coord_t min_w, bool rounded_visualization) +void ToolpathVisualizer::width_legend(const Shape& input, coord_t nozzle_size, coord_t max_dev, coord_t min_w, bool rounded_visualization) { auto to_string = [](double v) { @@ -71,7 +71,7 @@ void ToolpathVisualizer::width_legend(const Polygons& input, coord_t nozzle_size legend_btm.p_ += (legend_mid.p_ - legend_btm.p_) / 4; legend_top.p_ += (legend_mid.p_ - legend_top.p_) / 4; ExtrusionSegment legend_segment(legend_btm, legend_top, true, false); - svg_.writeAreas(legend_segment.toPolygons(false), SVG::ColorObject(200, 200, 200), SVG::Color::NONE); // real outline + svg_.writeAreas(legend_segment.toShape(false), SVG::ColorObject(200, 200, 200), SVG::Color::NONE); // real outline std::vector all_segments_plus; all_segments_plus.emplace_back(legend_segment); // colored @@ -143,7 +143,7 @@ void ToolpathVisualizer::widths( // } s.from_.w_ *= w / .9; s.to_.w_ *= w / .9; - Polygons covered = s.toPolygons(); + Shape covered = s.toShape(); svg_.writeAreas(covered, SVG::ColorObject(clr.x_, clr.y_, clr.z_), SVG::Color::NONE); } } diff --git a/src/utils/VoronoiUtils.cpp b/src/utils/VoronoiUtils.cpp index 564a48e99b..3533a98d00 100644 --- a/src/utils/VoronoiUtils.cpp +++ b/src/utils/VoronoiUtils.cpp @@ -8,6 +8,7 @@ #include +#include "geometry/PointMatrix.h" #include "utils/linearAlg2D.h" #include "utils/macros.h" diff --git a/src/utils/VoxelUtils.cpp b/src/utils/VoxelUtils.cpp index ddac6713d5..6d2f28173d 100644 --- a/src/utils/VoxelUtils.cpp +++ b/src/utils/VoxelUtils.cpp @@ -95,9 +95,9 @@ bool VoxelUtils::walkLine(Point3LL start, Point3LL end, const std::function& process_cell_func) const +bool VoxelUtils::walkPolygons(const Shape& polys, coord_t z, const std::function& process_cell_func) const { - for (ConstPolygonRef poly : polys) + for (const Polygon& poly : polys) { Point2LL last = poly.back(); for (Point2LL p : poly) @@ -113,9 +113,9 @@ bool VoxelUtils::walkPolygons(const Polygons& polys, coord_t z, const std::funct return true; } -bool VoxelUtils::walkDilatedPolygons(const Polygons& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const +bool VoxelUtils::walkDilatedPolygons(const Shape& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const { - Polygons translated = polys; + Shape translated = polys; const Point3LL translation = (Point3LL(1, 1, 1) - kernel.kernel_size_ % 2) * cell_size_ / 2; if (translation.x_ && translation.y_) { @@ -124,9 +124,9 @@ bool VoxelUtils::walkDilatedPolygons(const Polygons& polys, coord_t z, const Dil return walkPolygons(translated, z + translation.z_, dilate(kernel, process_cell_func)); } -bool VoxelUtils::walkAreas(const Polygons& polys, coord_t z, const std::function& process_cell_func) const +bool VoxelUtils::walkAreas(const Shape& polys, coord_t z, const std::function& process_cell_func) const { - Polygons translated = polys; + Shape translated = polys; const Point3LL translation = -cell_size_ / 2; // offset half a cell so that the dots of spreadDotsArea are centered on the middle of the cell isntead of the lower corners. if (translation.x_ && translation.y_) { @@ -135,7 +135,7 @@ bool VoxelUtils::walkAreas(const Polygons& polys, coord_t z, const std::function return _walkAreas(translated, z, process_cell_func); } -bool VoxelUtils::_walkAreas(const Polygons& polys, coord_t z, const std::function& process_cell_func) const +bool VoxelUtils::_walkAreas(const Shape& polys, coord_t z, const std::function& process_cell_func) const { std::vector skin_points = PolygonUtils::spreadDotsArea(polys, Point2LL(cell_size_.x_, cell_size_.y_)); for (Point2LL p : skin_points) @@ -149,9 +149,9 @@ bool VoxelUtils::_walkAreas(const Polygons& polys, coord_t z, const std::functio return true; } -bool VoxelUtils::walkDilatedAreas(const Polygons& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const +bool VoxelUtils::walkDilatedAreas(const Shape& polys, coord_t z, const DilationKernel& kernel, const std::function& process_cell_func) const { - Polygons translated = polys; + Shape translated = polys; const Point3LL translation = (Point3LL(1, 1, 1) - kernel.kernel_size_ % 2) * cell_size_ / 2 // offset half a cell when using a n even kernel - cell_size_ / 2; // offset half a cell so that the dots of spreadDotsArea are centered on the middle of the cell isntead of the lower corners. if (translation.x_ && translation.y_) diff --git a/src/utils/LinearAlg2D.cpp b/src/utils/linearAlg2D.cpp similarity index 81% rename from src/utils/LinearAlg2D.cpp rename to src/utils/linearAlg2D.cpp index 2292323251..64b50cee10 100644 --- a/src/utils/LinearAlg2D.cpp +++ b/src/utils/linearAlg2D.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #include "utils/linearAlg2D.h" @@ -6,8 +6,11 @@ #include // swap #include #include // atan2 +#include -#include "utils/Point2LL.h" // dot +#include "geometry/Point3Matrix.h" +#include "geometry/PointMatrix.h" +#include "utils/math.h" namespace cura { @@ -262,4 +265,43 @@ Point2LL LinearAlg2D::getBisectorVector(const Point2LL& intersect, const Point2L return (((a0 * vec_len) / std::max(1LL, vSize(a0))) + ((b0 * vec_len) / std::max(1LL, vSize(b0)))) / 2; } +Point3Matrix LinearAlg2D::rotateAround(const Point2LL& middle, double rotation) +{ + PointMatrix rotation_matrix(rotation); + Point3Matrix rotation_matrix_homogeneous(rotation_matrix); + return Point3Matrix::translate(middle).compose(rotation_matrix_homogeneous).compose(Point3Matrix::translate(-middle)); +} + +bool LinearAlg2D::lineLineIntersection(const Point2LL& a, const Point2LL& b, const Point2LL& c, const Point2LL& d, Point2LL& output) +{ + // Adapted from Apex: https://github.com/Ghostkeeper/Apex/blob/eb75f0d96e36c7193d1670112826842d176d5214/include/apex/line_segment.hpp#L91 + // Adjusted to work with lines instead of line segments. + const Point2LL l1_delta = b - a; + const Point2LL l2_delta = d - c; + const coord_t divisor = cross(l1_delta, l2_delta); // Pre-compute divisor needed for the intersection check. + if (divisor == 0) + { + // The lines are parallel if the cross product of their directions is zero. + return false; + } + + // Create a parametric representation of each line. + // We'll equate the parametric equations to each other to find the intersection then. + // Parametric equation is L = P + Vt (where P and V are a starting point and directional vector). + // We'll map the starting point of one line onto the parameter system of the other line. + // Then using the divisor we can see whether and where they cross. + const Point2LL starts_delta = a - c; + const coord_t l1_parametric = cross(l2_delta, starts_delta); + Point2LL result = a + Point2LL(round_divide_signed(l1_parametric * l1_delta.X, divisor), round_divide_signed(l1_parametric * l1_delta.Y, divisor)); + + if (std::abs(result.X) > std::numeric_limits::max() || std::abs(result.Y) > std::numeric_limits::max()) + { + // Intersection is so far away that it could lead to integer overflows. + // Even though the lines aren't 100% parallel, it's better to pretend they are. They are practically parallel. + return false; + } + output = result; + return true; +} + } // namespace cura diff --git a/src/utils/polygon.cpp b/src/utils/polygon.cpp deleted file mode 100644 index 99cb2be1b9..0000000000 --- a/src/utils/polygon.cpp +++ /dev/null @@ -1,1727 +0,0 @@ -// Copyright (c) 2024 UltiMaker -// CuraEngine is released under the terms of the AGPLv3 or higher. - -#include "utils/polygon.h" - -#include -#include - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "utils/ListPolyIt.h" -#include "utils/PolylineStitcher.h" -#include "utils/linearAlg2D.h" // pointLiesOnTheRightOfLine - -namespace cura -{ - -size_t ConstPolygonRef::size() const -{ - return path->size(); -} - -bool ConstPolygonRef::empty() const -{ - return path->empty(); -} - -bool ConstPolygonRef::shorterThan(const coord_t check_length) const -{ - return cura::shorterThan(*this, check_length); -} - -bool ConstPolygonRef::_inside(Point2LL p, bool border_result) const -{ - const ConstPolygonRef thiss = *this; - if (size() < 1) - { - return false; - } - - int crossings = 0; - Point2LL p0 = back(); - for (unsigned int n = 0; n < size(); n++) - { - Point2LL p1 = thiss[n]; - // no tests unless the segment p0-p1 is at least partly at, or to right of, p.X - short comp = LinearAlg2D::pointLiesOnTheRightOfLine(p, p0, p1); - if (comp == 1) - { - crossings++; - } - else if (comp == 0) - { - return border_result; - } - p0 = p1; - } - return (crossings % 2) == 1; -} - - -Polygons ConstPolygonRef::intersection(const ConstPolygonRef& other) const -{ - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPath(*path, ClipperLib::ptSubject, true); - clipper.AddPath(*other.path, ClipperLib::ptClip, true); - clipper.Execute(ClipperLib::ctIntersection, ret.paths); - return ret; -} - -bool Polygons::empty() const -{ - return paths.empty(); -} - -Polygons Polygons::approxConvexHull(int extra_outset) -{ - constexpr int overshoot = MM2INT(100); // 10cm (hard-coded value). - - Polygons convex_hull; - // Perform the offset for each polygon one at a time. - // This is necessary because the polygons may overlap, in which case the offset could end up in an infinite loop. - // See http://www.angusj.com/delphi/clipper/documentation/Docs/Units/ClipperLib/Classes/ClipperOffset/_Body.htm - for (const ClipperLib::Path& path : paths) - { - Polygons offset_result; - ClipperLib::ClipperOffset offsetter(1.2, 10.0); - offsetter.AddPath(path, ClipperLib::jtRound, ClipperLib::etClosedPolygon); - offsetter.Execute(offset_result.paths, overshoot); - convex_hull.add(offset_result); - } - return convex_hull.unionPolygons().offset(-overshoot + extra_outset, ClipperLib::jtRound); -} - -void Polygons::makeConvex() -{ - // early out if there is nothing to do - if (empty()) - { - return; - } - - // Andrew’s Monotone Chain Convex Hull Algorithm - std::vector points; - - for (const auto& poly : this->paths) - { - points.insert(points.end(), poly.begin(), poly.end()); - } - - ClipperLib::Path convexified; - auto make_sorted_poly_convex = [&convexified](std::vector& poly) - { - convexified.push_back(poly[0]); - - for (const auto window : poly | ranges::views::sliding(2)) - { - const Point2LL& current = window[0]; - const Point2LL& after = window[1]; - - if (LinearAlg2D::pointIsLeftOfLine(current, convexified.back(), after) < 0) - { - // Track backwards to make sure we haven't been in a concave pocket for multiple vertices already. - while (convexified.size() >= 2 - && (LinearAlg2D::pointIsLeftOfLine(convexified.back(), convexified[convexified.size() - 2], current) >= 0 - || LinearAlg2D::pointIsLeftOfLine(convexified.back(), convexified[convexified.size() - 2], convexified.front()) > 0)) - { - convexified.pop_back(); - } - convexified.push_back(current); - } - } - }; - - std::sort( - points.begin(), - points.end(), - [](Point2LL a, Point2LL b) - { - return a.X == b.X ? a.Y < b.Y : a.X < b.X; - }); - make_sorted_poly_convex(points); - std::reverse(points.begin(), points.end()); - make_sorted_poly_convex(points); - - this->paths = { convexified }; -} - -size_t Polygons::pointCount() const -{ - size_t count = 0; - for (const ClipperLib::Path& path : paths) - { - count += path.size(); - } - return count; -} - -bool Polygons::inside(Point2LL p, bool border_result) const -{ - int poly_count_inside = 0; - for (const ClipperLib::Path& poly : *this) - { - const int is_inside_this_poly = ClipperLib::PointInPolygon(p, poly); - if (is_inside_this_poly == -1) - { - return border_result; - } - poly_count_inside += is_inside_this_poly; - } - return (poly_count_inside % 2) == 1; -} - -bool PolygonsPart::inside(Point2LL p, bool border_result) const -{ - if (size() < 1) - { - return false; - } - if (! (*this)[0].inside(p, border_result)) - { - return false; - } - for (unsigned int n = 1; n < paths.size(); n++) - { - if ((*this)[n].inside(p, border_result)) - { - return false; - } - } - return true; -} - -bool Polygons::insideOld(Point2LL p, bool border_result) const -{ - const Polygons& thiss = *this; - if (size() < 1) - { - return false; - } - - int crossings = 0; - for (const ClipperLib::Path& poly : thiss) - { - Point2LL p0 = poly.back(); - for (const Point2LL& p1 : poly) - { - short comp = LinearAlg2D::pointLiesOnTheRightOfLine(p, p0, p1); - if (comp == 1) - { - crossings++; - } - else if (comp == 0) - { - return border_result; - } - p0 = p1; - } - } - return (crossings % 2) == 1; -} - -size_t Polygons::findInside(Point2LL p, bool border_result) -{ - Polygons& thiss = *this; - if (size() < 1) - { - return false; - } - - // NOTE: Keep these vectors fixed-size, they replace an (non-standard, sized at runtime) arrays. - std::vector min_x(size(), std::numeric_limits::max()); - std::vector crossings(size()); - - for (size_t poly_idx = 0; poly_idx < size(); poly_idx++) - { - PolygonRef poly = thiss[poly_idx]; - Point2LL p0 = poly.back(); - for (Point2LL& p1 : poly) - { - short comp = LinearAlg2D::pointLiesOnTheRightOfLine(p, p0, p1); - if (comp == 1) - { - crossings[poly_idx]++; - int64_t x; - if (p1.Y == p0.Y) - { - x = p0.X; - } - else - { - x = p0.X + (p1.X - p0.X) * (p.Y - p0.Y) / (p1.Y - p0.Y); - } - if (x < min_x[poly_idx]) - { - min_x[poly_idx] = x; - } - } - else if (border_result && comp == 0) - { - return poly_idx; - } - p0 = p1; - } - } - - int64_t min_x_uneven = std::numeric_limits::max(); - size_t ret = NO_INDEX; - size_t n_unevens = 0; - for (size_t array_idx = 0; array_idx < size(); array_idx++) - { - if (crossings[array_idx] % 2 == 1) - { - n_unevens++; - if (min_x[array_idx] < min_x_uneven) - { - min_x_uneven = min_x[array_idx]; - ret = array_idx; - } - } - } - if (n_unevens % 2 == 0) - { - ret = NO_INDEX; - } - return ret; -} - -Polygons Polygons::intersectionPolyLines(const Polygons& polylines, bool restitch, const coord_t max_stitch_distance) const -{ - Polygons split_polylines = polylines.splitPolylinesIntoSegments(); - - ClipperLib::PolyTree result; - ClipperLib::Clipper clipper(clipper_init); - clipper.AddPaths(split_polylines.paths, ClipperLib::ptSubject, false); - clipper.AddPaths(paths, ClipperLib::ptClip, true); - clipper.Execute(ClipperLib::ctIntersection, result); - Polygons ret; - ClipperLib::OpenPathsFromPolyTree(result, ret.paths); - - if (restitch) - { - Polygons result_lines, result_polygons; - const coord_t snap_distance = 10_mu; - PolylineStitcher::stitch(ret, result_lines, result_polygons, max_stitch_distance, snap_distance); - ret = result_lines; - // if polylines got stitched into polygons, split them back up into a polyline again, because the result only admits polylines - for (PolygonRef poly : result_polygons) - { - if (poly.empty()) - continue; - if (poly.size() > 2) - { - poly.emplace_back(poly[0]); - } - ret.add(poly); - } - } - - return ret; -} - -void Polygons::toPolylines() -{ - for (PolygonRef poly : *this) - { - if (poly.empty()) - continue; - poly.emplace_back(poly.front()); - } -} - -void Polygons::splitPolylinesIntoSegments(Polygons& result) const -{ - for (ConstPolygonRef poly : *this) - { - poly.splitPolylineIntoSegments(result); - } -} -Polygons Polygons::splitPolylinesIntoSegments() const -{ - Polygons ret; - splitPolylinesIntoSegments(ret); - return ret; -} - -void Polygons::splitPolygonsIntoSegments(Polygons& result) const -{ - for (ConstPolygonRef poly : *this) - { - poly.splitPolygonIntoSegments(result); - } -} -Polygons Polygons::splitPolygonsIntoSegments() const -{ - Polygons ret; - splitPolygonsIntoSegments(ret); - return ret; -} - -coord_t Polygons::polyLineLength() const -{ - coord_t length = 0; - for (ConstPolygonRef poly : *this) - { - length += poly.polylineLength(); - } - return length; -} - -Polygons Polygons::offset(coord_t distance, ClipperLib::JoinType join_type, double miter_limit) const -{ - if (distance == 0) - { - return *this; - } - Polygons ret; - ClipperLib::ClipperOffset clipper(miter_limit, 10.0); - clipper.AddPaths(unionPolygons().paths, join_type, ClipperLib::etClosedPolygon); - clipper.MiterLimit = miter_limit; - clipper.Execute(ret.paths, distance); - return ret; -} - -Polygons Polygons::offset(const std::vector& offset_dists) const -{ - // we need as many offset-dists as points - assert(this->pointCount() == offset_dists.size()); - - Polygons ret; - int i = 0; - for (auto& poly_line : this->paths - | ranges::views::filter( - [](const auto& path) - { - return ! path.empty(); - })) - { - std::vector ret_poly_line; - - auto prev_p = poly_line.back(); - auto prev_dist = offset_dists[i + poly_line.size() - 1]; - - for (const auto& p : poly_line) - { - auto offset_dist = offset_dists[i]; - - auto vec_dir = prev_p - p; - - constexpr coord_t min_vec_len = 10; - if (vSize2(vec_dir) > min_vec_len * min_vec_len) - { - auto offset_p1 = turn90CCW(normal(vec_dir, prev_dist)); - auto offset_p2 = turn90CCW(normal(vec_dir, offset_dist)); - - ret_poly_line.emplace_back(prev_p + offset_p1); - ret_poly_line.emplace_back(p + offset_p2); - } - - prev_p = p; - prev_dist = offset_dist; - i++; - } - - ret.add(ret_poly_line); - } - - ClipperLib::SimplifyPolygons(ret.paths, ClipperLib::PolyFillType::pftPositive); - - return ret; -} - -Polygons ConstPolygonRef::offset(int distance, ClipperLib::JoinType join_type, double miter_limit) const -{ - if (distance == 0) - { - Polygons ret; - ret.add(*this); - return ret; - } - Polygons ret; - ClipperLib::ClipperOffset clipper(miter_limit, 10.0); - clipper.AddPath(*path, join_type, ClipperLib::etClosedPolygon); - clipper.MiterLimit = miter_limit; - clipper.Execute(ret.paths, distance); - return ret; -} - -void PolygonRef::removeColinearEdges(const AngleRadians max_deviation_angle) -{ - // TODO: Can be made more efficient (for example, use pointer-types for process-/skip-indices, so we can swap them without copy). - - size_t num_removed_in_iteration = 0; - do - { - num_removed_in_iteration = 0; - - std::vector process_indices(path->size(), true); - - bool go = true; - while (go) - { - go = false; - - const auto& rpath = *path; - const size_t pathlen = rpath.size(); - if (pathlen <= 3) - { - return; - } - - std::vector skip_indices(path->size(), false); - - ClipperLib::Path new_path; - for (size_t point_idx = 0; point_idx < pathlen; ++point_idx) - { - // Don't iterate directly over process-indices, but do it this way, because there are points _in_ process-indices that should nonetheless be skipped: - if (! process_indices[point_idx]) - { - new_path.push_back(rpath[point_idx]); - continue; - } - - // Should skip the last point for this iteration if the old first was removed (which can be seen from the fact that the new first was skipped): - if (point_idx == (pathlen - 1) && skip_indices[0]) - { - skip_indices[new_path.size()] = true; - go = true; - new_path.push_back(rpath[point_idx]); - break; - } - - const Point2LL& prev = rpath[(point_idx - 1 + pathlen) % pathlen]; - const Point2LL& pt = rpath[point_idx]; - const Point2LL& next = rpath[(point_idx + 1) % pathlen]; - - double angle = LinearAlg2D::getAngleLeft(prev, pt, next); // [0 : 2 * pi] - if (angle >= std::numbers::pi) - { - angle -= std::numbers::pi; - } // map [pi : 2 * pi] to [0 : pi] - - // Check if the angle is within limits for the point to 'make sense', given the maximum deviation. - // If the angle indicates near-parallel segments ignore the point 'pt' - if (angle > max_deviation_angle && angle < std::numbers::pi - max_deviation_angle) - { - new_path.push_back(pt); - } - else if (point_idx != (pathlen - 1)) - { - // Skip the next point, since the current one was removed: - skip_indices[new_path.size()] = true; - go = true; - new_path.push_back(next); - ++point_idx; - } - } - *path = new_path; - num_removed_in_iteration += pathlen - path->size(); - - process_indices.clear(); - process_indices.insert(process_indices.end(), skip_indices.begin(), skip_indices.end()); - } - } while (num_removed_in_iteration > 0); -} - -void PolygonRef::applyMatrix(const PointMatrix& matrix) -{ - for (unsigned int path_idx = 0; path_idx < path->size(); path_idx++) - { - (*path)[path_idx] = matrix.apply((*path)[path_idx]); - } -} -void PolygonRef::applyMatrix(const Point3Matrix& matrix) -{ - for (unsigned int path_idx = 0; path_idx < path->size(); path_idx++) - { - (*path)[path_idx] = matrix.apply((*path)[path_idx]); - } -} - -Polygons Polygons::getOutsidePolygons() const -{ - Polygons ret; - ClipperLib::Clipper clipper(clipper_init); - ClipperLib::PolyTree poly_tree; - constexpr bool paths_are_closed_polys = true; - clipper.AddPaths(paths, ClipperLib::ptSubject, paths_are_closed_polys); - clipper.Execute(ClipperLib::ctUnion, poly_tree); - - for (int outer_poly_idx = 0; outer_poly_idx < poly_tree.ChildCount(); outer_poly_idx++) - { - ClipperLib::PolyNode* child = poly_tree.Childs[outer_poly_idx]; - ret.emplace_back(child->Contour); - } - return ret; -} - -void Polygons::removeEmptyHoles_processPolyTreeNode(const ClipperLib::PolyNode& node, const bool remove_holes, Polygons& ret) const -{ - for (int outer_poly_idx = 0; outer_poly_idx < node.ChildCount(); outer_poly_idx++) - { - ClipperLib::PolyNode* child = node.Childs[outer_poly_idx]; - if (remove_holes) - { - ret.emplace_back(child->Contour); - } - for (int hole_node_idx = 0; hole_node_idx < child->ChildCount(); hole_node_idx++) - { - ClipperLib::PolyNode& hole_node = *child->Childs[hole_node_idx]; - if ((hole_node.ChildCount() > 0) == remove_holes) - { - ret.emplace_back(hole_node.Contour); - removeEmptyHoles_processPolyTreeNode(hole_node, remove_holes, ret); - } - } - } -} - -void Polygons::removeSmallAreas(const double min_area_size, const bool remove_holes) -{ - auto new_end = paths.end(); - if (remove_holes) - { - for (auto it = paths.begin(); it < new_end;) - { - // All polygons smaller than target are removed by replacing them with a polygon from the back of the vector - if (std::abs(INT2MM2(ClipperLib::Area(*it))) < min_area_size) - { - *it = std::move(*--new_end); - continue; - } - it++; // Skipped on removal such that the polygon just swaped in is checked next - } - } - else - { - // For each polygon, computes the signed area, move small outlines at the end of the vector and keep references on small holes - std::vector small_holes; - for (auto it = paths.begin(); it < new_end;) - { - double area = INT2MM2(ClipperLib::Area(*it)); - if (std::abs(area) < min_area_size) - { - if (area >= 0) - { - --new_end; - if (it < new_end) - { - std::swap(*new_end, *it); - continue; - } - else - { // Don't self-swap the last Path - break; - } - } - else - { - small_holes.push_back(*it); - } - } - it++; // Skipped on removal such that the polygon just swaped in is checked next - } - - // Removes small holes that have their first point inside one of the removed outlines - // Iterating in reverse ensures that unprocessed small holes won't be moved - const auto removed_outlines_start = new_end; - for (auto hole_it = small_holes.rbegin(); hole_it < small_holes.rend(); hole_it++) - { - for (auto outline_it = removed_outlines_start; outline_it < paths.end(); outline_it++) - { - if (PolygonRef(*outline_it).inside(*hole_it->begin())) - { - **hole_it = std::move(*--new_end); - break; - } - } - } - } - paths.resize(new_end - paths.begin()); -} - -void Polygons::removeSmallCircumference(const coord_t min_circumference_size, const bool remove_holes) -{ - removeSmallAreaCircumference(0.0, min_circumference_size, remove_holes); -} - -void Polygons::removeSmallAreaCircumference(const double min_area_size, const coord_t min_circumference_size, const bool remove_holes) -{ - Polygons new_polygon; - - bool outline_is_removed = false; - for (ConstPolygonRef poly : paths) - { - double area = poly.area(); - auto circumference = poly.polygonLength(); - bool is_outline = area >= 0; - - if (is_outline) - { - if (circumference >= min_circumference_size && std::abs(area) >= min_area_size) - { - new_polygon.add(poly); - outline_is_removed = false; - } - else - { - outline_is_removed = true; - } - } - else if (outline_is_removed) - { - // containing parent outline is removed; hole should be removed as well - } - else if (! remove_holes || (circumference >= min_circumference_size && std::abs(area) >= min_area_size)) - { - // keep hole-polygon if we do not remove holes, or if its - // circumference is bigger then the minimum circumference size - new_polygon.add(poly); - } - } - - *this = new_polygon; -} - -void Polygons::removeDegenerateVerts() -{ - _removeDegenerateVerts(false); -} - -void Polygons::removeDegenerateVertsPolyline() -{ - _removeDegenerateVerts(true); -} - -void Polygons::_removeDegenerateVerts(const bool for_polyline) -{ - Polygons& thiss = *this; - for (size_t poly_idx = 0; poly_idx < size(); poly_idx++) - { - PolygonRef poly = thiss[poly_idx]; - Polygon result; - - auto isDegenerate = [](const Point2LL& last, const Point2LL& now, const Point2LL& next) - { - Point2LL last_line = now - last; - Point2LL next_line = next - now; - return dot(last_line, next_line) == -1 * vSize(last_line) * vSize(next_line); - }; - - // With polylines, skip the first and last vertex. - const size_t start_vertex = for_polyline ? 1 : 0; - const size_t end_vertex = for_polyline ? poly.size() - 1 : poly.size(); - for (size_t i = 0; i < start_vertex; ++i) - { - result.add(poly[i]); // Add everything before the start vertex. - } - - bool isChanged = false; - for (size_t idx = start_vertex; idx < end_vertex; idx++) - { - const Point2LL& last = (result.size() == 0) ? poly.back() : result.back(); - if (idx + 1 >= poly.size() && result.size() == 0) - { - break; - } - const Point2LL& next = (idx + 1 >= poly.size()) ? result[0] : poly[idx + 1]; - if (isDegenerate(last, poly[idx], next)) - { // lines are in the opposite direction - // don't add vert to the result - isChanged = true; - while (result.size() > 1 && isDegenerate(result[result.size() - 2], result.back(), next)) - { - result.pop_back(); - } - } - else - { - result.add(poly[idx]); - } - } - - for (size_t i = end_vertex; i < poly.size(); ++i) - { - result.add(poly[i]); // Add everything after the end vertex. - } - - if (isChanged) - { - if (for_polyline || result.size() > 2) - { - *poly = *result; - } - else - { - thiss.remove(poly_idx); - poly_idx--; // effectively the next iteration has the same poly_idx (referring to a new poly which is not yet processed) - } - } - } -} - -Polygons Polygons::toPolygons(ClipperLib::PolyTree& poly_tree) -{ - Polygons ret; - ClipperLib::PolyTreeToPaths(poly_tree, ret.paths); - return ret; -} - -[[maybe_unused]] Polygons Polygons::fromWkt(const std::string& wkt) -{ - typedef boost::geometry::model::d2::point_xy point_type; - typedef boost::geometry::model::polygon polygon_type; - - polygon_type poly; - boost::geometry::read_wkt(wkt, poly); - - Polygons ret; - - Polygon outer; - for (const auto& point : poly.outer()) - { - outer.add(Point2LL(point.x(), point.y())); - } - ret.add(outer); - - for (const auto& hole : poly.inners()) - { - Polygon inner; - for (const auto& point : hole) - { - inner.add(Point2LL(point.x(), point.y())); - } - ret.add(inner); - } - - return ret; -} - -[[maybe_unused]] void Polygons::writeWkt(std::ostream& stream) const -{ - stream << "POLYGON ("; - const auto paths_str = paths - | ranges::views::transform( - [](const auto& path) - { - const auto line_string = ranges::views::concat(path, path | ranges::views::take(1)) - | ranges::views::transform( - [](const auto& point) - { - return fmt::format("{} {}", point.X, point.Y); - }) - | ranges::views::join(ranges::views::c_str(", ")) | ranges::to(); - return "(" + line_string + ")"; - }) - | ranges::views::join(ranges::views::c_str(", ")) | ranges::to(); - stream << paths_str; - stream << ")"; -} - -bool ConstPolygonRef::smooth_corner_complex(const Point2LL p1, ListPolyIt& p0_it, ListPolyIt& p2_it, const int64_t shortcut_length) -{ - // walk away from the corner until the shortcut > shortcut_length or it would smooth a piece inward - // - walk in both directions untill shortcut > shortcut_length - // - stop walking in one direction if it would otherwise cut off a corner in that direction - // - same in the other direction - // - stop if both are cut off - // walk by updating p0_it and p2_it - int64_t shortcut_length2 = shortcut_length * shortcut_length; - bool forward_is_blocked = false; - bool forward_is_too_far = false; - bool backward_is_blocked = false; - bool backward_is_too_far = false; - while (true) - { - const bool forward_has_converged = forward_is_blocked || forward_is_too_far; - const bool backward_has_converged = backward_is_blocked || backward_is_too_far; - if (forward_has_converged && backward_has_converged) - { - if (forward_is_too_far && backward_is_too_far && vSize2(p0_it.prev().p() - p2_it.next().p()) < shortcut_length2) - { - // o - // / \ . - // o o - // | | - // \ / . - // | | - // \ / . - // | | - // o o - --p0_it; - ++p2_it; - forward_is_too_far = false; // invalidate data - backward_is_too_far = false; // invalidate data - continue; - } - else - { - break; - } - } - smooth_outward_step(p1, shortcut_length2, p0_it, p2_it, forward_is_blocked, backward_is_blocked, forward_is_too_far, backward_is_too_far); - if (p0_it.prev() == p2_it || p0_it == p2_it) - { // stop if we went all the way around the polygon - // this should only be the case for hole polygons (?) - if (forward_is_too_far && backward_is_too_far) - { - // in case p0_it.prev() == p2_it : - // / . - // / /| - // | becomes | | - // \ \| - // \ . - // in case p0_it == p2_it : - // / . - // / becomes /| - // \ \| - // \ . - break; - } - else - { - // this whole polygon can be removed - return true; - } - } - } - - const Point2LL v02 = p2_it.p() - p0_it.p(); - const int64_t v02_size2 = vSize2(v02); - // set the following: - // p0_it = start point of line - // p2_it = end point of line - if (std::abs(v02_size2 - shortcut_length2) < shortcut_length * 10) // i.e. if (size2 < l * (l+10) && size2 > l * (l-10)) - { // v02 is approximately shortcut length - // handle this separately to avoid rounding problems below in the getPointOnLineWithDist function - // p0_it and p2_it are already correct - } - else if (! backward_is_blocked && ! forward_is_blocked) - { // introduce two new points - // 1----b---->2 - // ^ / - // | / - // | / - // |/ - // |a - // | - // 0 - const int64_t v02_size = sqrt(v02_size2); - - const ListPolyIt p0_2_it = p0_it.prev(); - const ListPolyIt p2_2_it = p2_it.next(); - const Point2LL p2_2 = p2_2_it.p(); - const Point2LL p0_2 = p0_2_it.p(); - const Point2LL v02_2 = p0_2 - p2_2; - const int64_t v02_2_size = vSize(v02_2); - double progress - = std::min(1.0, INT2MM(shortcut_length - v02_size) / INT2MM(v02_2_size - v02_size)); // account for rounding error when v02_2_size is approx equal to v02_size - assert(progress >= 0.0f && progress <= 1.0f && "shortcut length must be between last length and new length"); - const Point2LL new_p0 = p0_it.p() + (p0_2 - p0_it.p()) * progress; - p0_it = ListPolyIt::insertPointNonDuplicate(p0_2_it, p0_it, new_p0); - const Point2LL new_p2 = p2_it.p() + (p2_2 - p2_it.p()) * progress; - p2_it = ListPolyIt::insertPointNonDuplicate(p2_it, p2_2_it, new_p2); - } - else if (! backward_is_blocked) - { // forward is blocked, back is open - // | - // 1->b - // ^ : - // | / - // 0 : - // |/ - // |a - // | - // 0_2 - const ListPolyIt p0_2_it = p0_it.prev(); - const Point2LL p0 = p0_it.p(); - const Point2LL p0_2 = p0_2_it.p(); - const Point2LL p2 = p2_it.p(); - Point2LL new_p0; - bool success = LinearAlg2D::getPointOnLineWithDist(p2, p0, p0_2, shortcut_length, new_p0); - // shortcut length must be possible given that last length was ok and new length is too long - if (success) - { -#ifdef ASSERT_INSANE_OUTPUT - assert(new_p0.X < 400000 && new_p0.Y < 400000); -#endif // #ifdef ASSERT_INSANE_OUTPUT - p0_it = ListPolyIt::insertPointNonDuplicate(p0_2_it, p0_it, new_p0); - } - else - { // if not then a rounding error occured - if (vSize(p2 - p0_2) < vSize2(p2 - p0)) - { - p0_it = p0_2_it; // start shortcut at 0 - } - } - } - else if (! forward_is_blocked) - { // backward is blocked, front is open - // 1----2----b----------->2_2 - // ^ ,-' - // | ,-' - //--0.-' - // a - const ListPolyIt p2_2_it = p2_it.next(); - const Point2LL p0 = p0_it.p(); - const Point2LL p2 = p2_it.p(); - const Point2LL p2_2 = p2_2_it.p(); - Point2LL new_p2; - bool success = LinearAlg2D::getPointOnLineWithDist(p0, p2, p2_2, shortcut_length, new_p2); - // shortcut length must be possible given that last length was ok and new length is too long - if (success) - { -#ifdef ASSERT_INSANE_OUTPUT - assert(new_p2.X < 400000 && new_p2.Y < 400000); -#endif // #ifdef ASSERT_INSANE_OUTPUT - p2_it = ListPolyIt::insertPointNonDuplicate(p2_it, p2_2_it, new_p2); - } - else - { // if not then a rounding error occured - if (vSize(p2_2 - p0) < vSize2(p2 - p0)) - { - p2_it = p2_2_it; // start shortcut at 0 - } - } - } - else - { - // | - // __|2 - // | / > shortcut cannot be of the desired length - // ___|/ . - // 0 - // both are blocked and p0_it and p2_it are already correct - } - // delete all cut off points - while (p0_it.next() != p2_it) - { - p0_it.next().remove(); - } - return false; -} - -void ConstPolygonRef::smooth_outward_step( - const Point2LL p1, - const int64_t shortcut_length2, - ListPolyIt& p0_it, - ListPolyIt& p2_it, - bool& forward_is_blocked, - bool& backward_is_blocked, - bool& forward_is_too_far, - bool& backward_is_too_far) -{ - const bool forward_has_converged = forward_is_blocked || forward_is_too_far; - const bool backward_has_converged = backward_is_blocked || backward_is_too_far; - const Point2LL p0 = p0_it.p(); - const Point2LL p2 = p2_it.p(); - bool walk_forward - = ! forward_has_converged && (backward_has_converged || (vSize2(p2 - p1) < vSize2(p0 - p1))); // whether to walk along the p1-p2 direction or in the p1-p0 direction - - if (walk_forward) - { - const ListPolyIt p2_2_it = p2_it.next(); - const Point2LL p2_2 = p2_2_it.p(); - bool p2_is_left = LinearAlg2D::pointIsLeftOfLine(p2, p0, p2_2) >= 0; - if (! p2_is_left) - { - forward_is_blocked = true; - return; - } - - const Point2LL v02_2 = p2_2 - p0_it.p(); - if (vSize2(v02_2) > shortcut_length2) - { - forward_is_too_far = true; - return; - } - - p2_it = p2_2_it; // make one step in the forward direction - backward_is_blocked = false; // invalidate data about backward walking - backward_is_too_far = false; - return; - } - else - { - const ListPolyIt p0_2_it = p0_it.prev(); - const Point2LL p0_2 = p0_2_it.p(); - bool p0_is_left = LinearAlg2D::pointIsLeftOfLine(p0, p0_2, p2) >= 0; - if (! p0_is_left) - { - backward_is_blocked = true; - return; - } - - const Point2LL v02_2 = p2_it.p() - p0_2; - if (vSize2(v02_2) > shortcut_length2) - { - backward_is_too_far = true; - return; - } - - p0_it = p0_2_it; // make one step in the backward direction - forward_is_blocked = false; // invalidate data about forward walking - forward_is_too_far = false; - return; - } -} - -void ConstPolygonRef::smooth_corner_simple( - const Point2LL p0, - const Point2LL p1, - const Point2LL p2, - const ListPolyIt p0_it, - const ListPolyIt p1_it, - const ListPolyIt p2_it, - const Point2LL v10, - const Point2LL v12, - const Point2LL v02, - const int64_t shortcut_length, - double cos_angle) -{ - // 1----b---->2 - // ^ / - // | / - // | / - // |/ - // |a - // | - // 0 - // ideally a1_size == b1_size - if (vSize2(v02) <= shortcut_length * (shortcut_length + 10) // v02 is approximately shortcut length - || (cos_angle > 0.9999 && LinearAlg2D::getDist2FromLine(p2, p0, p1) < 20 * 20)) // p1 is degenerate - { - // handle this separately to avoid rounding problems below in the getPointOnLineWithDist function - p1_it.remove(); - // don't insert new elements - } - else - { - // compute the distance a1 == b1 to get vSize(ab)==shortcut_length with the given angle between v10 and v12 - // 1 - // /|\ . - // / | \ . - // / | \ . - // / | \ . - // a/____|____\b . - // m - // use trigonometry on the right-angled triangle am1 - double a1m_angle = acos(cos_angle) / 2; - const int64_t a1_size = shortcut_length / 2 / sin(a1m_angle); - if (a1_size * a1_size < vSize2(v10) && a1_size * a1_size < vSize2(v12)) - { - Point2LL a = p1 + normal(v10, a1_size); - Point2LL b = p1 + normal(v12, a1_size); -#ifdef ASSERT_INSANE_OUTPUT - assert(vSize(a) < 4000000); - assert(vSize(b) < 4000000); -#endif // #ifdef ASSERT_INSANE_OUTPUT - ListPolyIt::insertPointNonDuplicate(p0_it, p1_it, a); - ListPolyIt::insertPointNonDuplicate(p1_it, p2_it, b); - p1_it.remove(); - } - else if (vSize2(v12) < vSize2(v10)) - { - // b - // 1->2 - // ^ | - // | / - // | | - // |/ - // |a - // | - // 0 - const Point2LL& b = p2_it.p(); - Point2LL a; - bool success = LinearAlg2D::getPointOnLineWithDist(b, p1, p0, shortcut_length, a); - // v02 has to be longer than ab! - if (success) - { // if not success then assume a is negligibly close to 0, but rounding errors caused a problem -#ifdef ASSERT_INSANE_OUTPUT - assert(vSize(a) < 4000000); -#endif // #ifdef ASSERT_INSANE_OUTPUT - ListPolyIt::insertPointNonDuplicate(p0_it, p1_it, a); - } - p1_it.remove(); - } - else - { - // 1---------b----------->2 - // ^ ,-' - // | ,-' - // 0.-' - // a - const Point2LL& a = p0_it.p(); - Point2LL b; - bool success = LinearAlg2D::getPointOnLineWithDist(a, p1, p2, shortcut_length, b); - // v02 has to be longer than ab! - if (success) - { // if not success then assume b is negligibly close to 2, but rounding errors caused a problem -#ifdef ASSERT_INSANE_OUTPUT - assert(vSize(b) < 4000000); -#endif // #ifdef ASSERT_INSANE_OUTPUT - ListPolyIt::insertPointNonDuplicate(p1_it, p2_it, b); - } - p1_it.remove(); - } - } -} - -void ConstPolygonRef::smooth_outward(const AngleDegrees min_angle, int shortcut_length, PolygonRef result) const -{ - // example of smoothed out corner: - // - // 6 - // ^ - // | - // inside | outside - // 2>3>4>5 - // ^ / . - // | / . - // 1 / . - // ^ / . - // |/ . - // | - // | - // 0 - - int shortcut_length2 = shortcut_length * shortcut_length; - double cos_min_angle = cos(min_angle / 180 * std::numbers::pi); - - ListPolygon poly; - ListPolyIt::convertPolygonToList(*this, poly); - - { // remove duplicate vertices - ListPolyIt p1_it(poly, poly.begin()); - do - { - ListPolyIt next = p1_it.next(); - if (vSize2(p1_it.p() - next.p()) < 10 * 10) - { - p1_it.remove(); - } - p1_it = next; - } while (p1_it != ListPolyIt(poly, poly.begin())); - } - - ListPolyIt p1_it(poly, poly.begin()); - do - { - const Point2LL p1 = p1_it.p(); - ListPolyIt p0_it = p1_it.prev(); - ListPolyIt p2_it = p1_it.next(); - const Point2LL p0 = p0_it.p(); - const Point2LL p2 = p2_it.p(); - - const Point2LL v10 = p0 - p1; - const Point2LL v12 = p2 - p1; - double cos_angle = INT2MM(INT2MM(dot(v10, v12))) / vSizeMM(v10) / vSizeMM(v12); - bool is_left_angle = LinearAlg2D::pointIsLeftOfLine(p1, p0, p2) > 0; - if (cos_angle > cos_min_angle && is_left_angle) - { - // angle is so sharp that it can be removed - Point2LL v02 = p2_it.p() - p0_it.p(); - if (vSize2(v02) >= shortcut_length2) - { - smooth_corner_simple(p0, p1, p2, p0_it, p1_it, p2_it, v10, v12, v02, shortcut_length, cos_angle); - } - else - { - bool remove_poly = smooth_corner_complex(p1, p0_it, p2_it, shortcut_length); // edits p0_it and p2_it! - if (remove_poly) - { - // don't convert ListPolygon into result - return; - } - } - // update: - p1_it = p2_it; // next point to consider for whether it's an internal corner - } - else - { - ++p1_it; - } - } while (p1_it != ListPolyIt(poly, poly.begin())); - - ListPolyIt::convertListPolygonToPolygon(poly, result); -} - -Polygons Polygons::smooth_outward(const AngleDegrees max_angle, int shortcut_length) -{ - Polygons ret; - for (unsigned int p = 0; p < size(); p++) - { - PolygonRef poly(paths[p]); - if (poly.size() < 3) - { - continue; - } - if (poly.size() == 3) - { - ret.add(poly); - continue; - } - poly.smooth_outward(max_angle, shortcut_length, ret.newPoly()); - if (ret.back().size() < 3) - { - ret.paths.resize(ret.paths.size() - 1); - } - } - return ret; -} - - -void ConstPolygonRef::splitPolylineIntoSegments(Polygons& result) const -{ - Point2LL last = front(); - for (size_t idx = 1; idx < size(); idx++) - { - Point2LL p = (*this)[idx]; - result.addLine(last, p); - last = p; - } -} - -Polygons ConstPolygonRef::splitPolylineIntoSegments() const -{ - Polygons ret; - splitPolylineIntoSegments(ret); - return ret; -} - -void ConstPolygonRef::splitPolygonIntoSegments(Polygons& result) const -{ - splitPolylineIntoSegments(result); - result.addLine(back(), front()); -} - -Polygons ConstPolygonRef::splitPolygonIntoSegments() const -{ - Polygons ret; - splitPolygonIntoSegments(ret); - return ret; -} - -void ConstPolygonRef::smooth(int remove_length, PolygonRef result) const -{ - // a typical zigzag with the middle part to be removed by removing (1) : - // - // 3 - // ^ - // | - // | - // inside | outside - // 1--->2 - // ^ - // | - // | - // | - // 0 - const ConstPolygonRef& thiss = *path; - ClipperLib::Path* poly = result.path; - if (size() > 0) - { - poly->push_back(thiss[0]); - } - auto is_zigzag = [remove_length](const int64_t v02_size, const int64_t v12_size, const int64_t v13_size, const int64_t dot1, const int64_t dot2) - { - if (v12_size > remove_length) - { // v12 or v13 is too long - return false; - } - const bool p1_is_left_of_v02 = dot1 < 0; - if (! p1_is_left_of_v02) - { // removing p1 wouldn't smooth outward - return false; - } - const bool p2_is_left_of_v13 = dot2 > 0; - if (p2_is_left_of_v13) - { // l0123 doesn't constitute a zigzag ''|,, - return false; - } - if (-dot1 <= v02_size * v12_size / 2) - { // angle at p1 isn't sharp enough - return false; - } - if (-dot2 <= v13_size * v12_size / 2) - { // angle at p2 isn't sharp enough - return false; - } - return true; - }; - Point2LL v02 = thiss[2] - thiss[0]; - Point2LL v02T = turn90CCW(v02); - int64_t v02_size = vSize(v02); - bool force_push = false; - for (unsigned int poly_idx = 1; poly_idx < size(); poly_idx++) - { - const Point2LL& p1 = thiss[poly_idx]; - const Point2LL& p2 = thiss[(poly_idx + 1) % size()]; - const Point2LL& p3 = thiss[(poly_idx + 2) % size()]; - // v02 computed in last iteration - // v02_size as well - const Point2LL v12 = p2 - p1; - const int64_t v12_size = vSize(v12); - const Point2LL v13 = p3 - p1; - const int64_t v13_size = vSize(v13); - - // v02T computed in last iteration - const int64_t dot1 = dot(v02T, v12); - const Point2LL v13T = turn90CCW(v13); - const int64_t dot2 = dot(v13T, v12); - bool push_point = force_push || ! is_zigzag(v02_size, v12_size, v13_size, dot1, dot2); - force_push = false; - if (push_point) - { - poly->push_back(p1); - } - else - { - // do not add the current one to the result - force_push = true; // ensure the next point is added; it cannot also be a zigzag - } - v02T = v13T; - v02 = v13; - v02_size = v13_size; - } -} - -Polygons Polygons::smooth(int remove_length) const -{ - Polygons ret; - for (unsigned int p = 0; p < size(); p++) - { - ConstPolygonRef poly(paths[p]); - if (poly.size() < 3) - { - continue; - } - if (poly.size() == 3) - { - ret.add(poly); - continue; - } - poly.smooth(remove_length, ret.newPoly()); - PolygonRef back = ret.back(); - if (back.size() < 3) - { - back.path->resize(back.path->size() - 1); - } - } - return ret; -} - -void ConstPolygonRef::smooth2(int remove_length, PolygonRef result) const -{ - const ConstPolygonRef& thiss = *this; - ClipperLib::Path* poly = result.path; - if (thiss.size() > 0) - { - poly->push_back(thiss[0]); - } - for (unsigned int poly_idx = 1; poly_idx < thiss.size(); poly_idx++) - { - const Point2LL& last = thiss[poly_idx - 1]; - const Point2LL& now = thiss[poly_idx]; - const Point2LL& next = thiss[(poly_idx + 1) % thiss.size()]; - if (shorterThen(last - now, remove_length) && shorterThen(now - next, remove_length)) - { - poly_idx++; // skip the next line piece (dont escalate the removal of edges) - if (poly_idx < thiss.size()) - { - poly->push_back(thiss[poly_idx]); - } - } - else - { - poly->push_back(thiss[poly_idx]); - } - } -} - -Polygons Polygons::smooth2(int remove_length, int min_area) const -{ - Polygons ret; - for (unsigned int p = 0; p < size(); p++) - { - ConstPolygonRef poly(paths[p]); - if (poly.size() == 0) - { - continue; - } - if (poly.area() < min_area || poly.size() <= 5) // when optimally removing, a poly with 5 pieces results in a triangle. Smaller polys dont have area! - { - ret.add(poly); - continue; - } - if (poly.size() < 4) - { - ret.add(poly); - } - else - { - poly.smooth2(remove_length, ret.newPoly()); - } - } - return ret; -} - -double Polygons::area() const -{ - double area = 0.0; - for (unsigned int poly_idx = 0; poly_idx < size(); poly_idx++) - { - area += operator[](poly_idx).area(); - // note: holes already have negative area - } - return area; -} - -std::vector Polygons::splitIntoParts(bool unionAll) const -{ - std::vector ret; - ClipperLib::Clipper clipper(clipper_init); - ClipperLib::PolyTree resultPolyTree; - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - if (unionAll) - clipper.Execute(ClipperLib::ctUnion, resultPolyTree, ClipperLib::pftNonZero, ClipperLib::pftNonZero); - else - clipper.Execute(ClipperLib::ctUnion, resultPolyTree); - - splitIntoParts_processPolyTreeNode(&resultPolyTree, ret); - return ret; -} - -void Polygons::splitIntoParts_processPolyTreeNode(ClipperLib::PolyNode* node, std::vector& ret) const -{ - for (int n = 0; n < node->ChildCount(); n++) - { - ClipperLib::PolyNode* child = node->Childs[n]; - PolygonsPart part; - part.add(child->Contour); - for (int i = 0; i < child->ChildCount(); i++) - { - part.add(child->Childs[i]->Contour); - splitIntoParts_processPolyTreeNode(child->Childs[i], ret); - } - ret.push_back(part); - } -} - -std::vector Polygons::sortByNesting() const -{ - std::vector ret; - ClipperLib::Clipper clipper(clipper_init); - ClipperLib::PolyTree resultPolyTree; - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - clipper.Execute(ClipperLib::ctUnion, resultPolyTree); - - sortByNesting_processPolyTreeNode(&resultPolyTree, 0, ret); - return ret; -} - -void Polygons::sortByNesting_processPolyTreeNode(ClipperLib::PolyNode* node, const size_t nesting_idx, std::vector& ret) const -{ - for (int n = 0; n < node->ChildCount(); n++) - { - ClipperLib::PolyNode* child = node->Childs[n]; - if (nesting_idx >= ret.size()) - { - ret.resize(nesting_idx + 1); - } - ret[nesting_idx].add(child->Contour); - sortByNesting_processPolyTreeNode(child, nesting_idx + 1, ret); - } -} - -Polygons Polygons::tubeShape(const coord_t inner_offset, const coord_t outer_offset) const -{ - return this->offset(outer_offset).difference(this->offset(-inner_offset)); -} - -Polygons Polygons::removeNearSelfIntersections() const -{ - using map_pt = mapbox::geometry::point; - using map_ring = mapbox::geometry::linear_ring; - using map_poly = mapbox::geometry::polygon; - using map_mpoly = mapbox::geometry::multi_polygon; - - map_mpoly mwpoly; - - mapbox::geometry::wagyu::wagyu wagyu; - - for (auto& polygon : splitIntoParts()) - { - mwpoly.emplace_back(); - map_poly& wpoly = mwpoly.back(); - for (auto& path : polygon) - { - wpoly.push_back(std::move(*reinterpret_cast>*>(&path))); - for (auto& point : wpoly.back()) - { - point.x /= 4; - point.y /= 4; - } - wagyu.add_ring(wpoly.back()); - } - } - - map_mpoly sln; - - wagyu.execute(mapbox::geometry::wagyu::clip_type_union, sln, mapbox::geometry::wagyu::fill_type_even_odd, mapbox::geometry::wagyu::fill_type_even_odd); - - Polygons polys; - - for (auto& poly : sln) - { - for (auto& ring : poly) - { - ring.pop_back(); - for (auto& point : ring) - { - point.x *= 4; - point.y *= 4; - } - polys.add(*reinterpret_cast*>(&ring)); // NOTE: 'add' already moves the vector - } - } - polys = polys.unionPolygons(); - polys.removeColinearEdges(); - - return polys; -} - -size_t PartsView::getPartContaining(size_t poly_idx, size_t* boundary_poly_idx) const -{ - const PartsView& partsView = *this; - for (size_t part_idx_now = 0; part_idx_now < partsView.size(); part_idx_now++) - { - const std::vector& partView = partsView[part_idx_now]; - if (partView.size() == 0) - { - continue; - } - std::vector::const_iterator result = std::find(partView.begin(), partView.end(), poly_idx); - if (result != partView.end()) - { - if (boundary_poly_idx) - { - *boundary_poly_idx = partView[0]; - } - return part_idx_now; - } - } - return NO_INDEX; -} - -PolygonsPart PartsView::assemblePart(size_t part_idx) const -{ - const PartsView& partsView = *this; - PolygonsPart ret; - if (part_idx != NO_INDEX) - { - for (size_t poly_idx_ff : partsView[part_idx]) - { - ret.add(polygons_[poly_idx_ff]); - } - } - return ret; -} - -PolygonsPart PartsView::assemblePartContaining(size_t poly_idx, size_t* boundary_poly_idx) const -{ - PolygonsPart ret; - size_t part_idx = getPartContaining(poly_idx, boundary_poly_idx); - if (part_idx != NO_INDEX) - { - return assemblePart(part_idx); - } - return ret; -} - -PartsView Polygons::splitIntoPartsView(bool unionAll) -{ - Polygons reordered; - PartsView partsView(*this); - ClipperLib::Clipper clipper(clipper_init); - ClipperLib::PolyTree resultPolyTree; - clipper.AddPaths(paths, ClipperLib::ptSubject, true); - if (unionAll) - clipper.Execute(ClipperLib::ctUnion, resultPolyTree, ClipperLib::pftNonZero, ClipperLib::pftNonZero); - else - clipper.Execute(ClipperLib::ctUnion, resultPolyTree); - - splitIntoPartsView_processPolyTreeNode(partsView, reordered, &resultPolyTree); - - (*this) = reordered; - return partsView; -} - -void Polygons::splitIntoPartsView_processPolyTreeNode(PartsView& partsView, Polygons& reordered, ClipperLib::PolyNode* node) const -{ - for (int n = 0; n < node->ChildCount(); n++) - { - ClipperLib::PolyNode* child = node->Childs[n]; - partsView.emplace_back(); - size_t pos = partsView.size() - 1; - partsView[pos].push_back(reordered.size()); - reordered.add(child->Contour); // TODO: should this steal the internal representation for speed? - for (int i = 0; i < child->ChildCount(); i++) - { - partsView[pos].push_back(reordered.size()); - reordered.add(child->Childs[i]->Contour); - splitIntoPartsView_processPolyTreeNode(partsView, reordered, child->Childs[i]); - } - } -} - -void Polygons::ensureManifold() -{ - const Polygons& polys = *this; - std::vector duplicate_locations; - std::unordered_set poly_locations; - for (size_t poly_idx = 0; poly_idx < polys.size(); poly_idx++) - { - ConstPolygonRef poly = polys[poly_idx]; - for (size_t point_idx = 0; point_idx < poly.size(); point_idx++) - { - Point2LL p = poly[point_idx]; - if (poly_locations.find(p) != poly_locations.end()) - { - duplicate_locations.push_back(p); - } - poly_locations.emplace(p); - } - } - Polygons removal_dots; - for (Point2LL p : duplicate_locations) - { - PolygonRef dot = removal_dots.newPoly(); - dot.add(p + Point2LL(0, 5)); - dot.add(p + Point2LL(5, 0)); - dot.add(p + Point2LL(0, -5)); - dot.add(p + Point2LL(-5, 0)); - } - if (! removal_dots.empty()) - { - *this = polys.difference(removal_dots); - } -} - -} // namespace cura diff --git a/src/utils/polygonUtils.cpp b/src/utils/polygonUtils.cpp index 7f7bc2466a..84f367c586 100644 --- a/src/utils/polygonUtils.cpp +++ b/src/utils/polygonUtils.cpp @@ -1,15 +1,19 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "utils/polygonUtils.h" #include #include +#include #include #include #include +#include "geometry/OpenPolyline.h" +#include "geometry/PointMatrix.h" +#include "geometry/SingleShape.h" #include "infill.h" #include "utils/SparsePointGridInclusive.h" #include "utils/linearAlg2D.h" @@ -18,7 +22,6 @@ #include #include "utils/AABB.h" -#include "utils/SVG.h" #endif namespace cura @@ -34,7 +37,7 @@ int64_t PolygonUtils::segmentLength(PolygonsPointIndex start, PolygonsPointIndex assert(start.poly_idx_ == end.poly_idx_); int64_t segment_length = 0; Point2LL prev_vert = start.p(); - ConstPolygonRef poly = (*start.polygons_)[start.poly_idx_]; + const Polygon& poly = start.getPolygon(); for (unsigned int point_idx = 1; point_idx <= poly.size(); point_idx++) { unsigned int vert_idx = (start.point_idx_ + point_idx) % poly.size(); @@ -51,16 +54,16 @@ int64_t PolygonUtils::segmentLength(PolygonsPointIndex start, PolygonsPointIndex return segment_length; } -void PolygonUtils::spreadDots(PolygonsPointIndex start, PolygonsPointIndex end, unsigned int n_dots, std::vector& result) +void PolygonUtils::spreadDots(PolygonsPointIndex start, PolygonsPointIndex end, unsigned int n_dots, std::vector& result) { assert(start.poly_idx_ == end.poly_idx_); int64_t segment_length = segmentLength(start, end); - ConstPolygonRef poly = (*start.polygons_)[start.poly_idx_]; + const Polygon& poly = start.getPolygon(); unsigned int n_dots_in_between = n_dots; if (start == end) { - result.emplace_back(start.p(), start.point_idx_, poly); + result.emplace_back(start.p(), start.point_idx_, &poly); n_dots_in_between--; // generate one less below, because we already pushed a point to the result } @@ -78,7 +81,7 @@ void PolygonUtils::spreadDots(PolygonsPointIndex start, PolygonsPointIndex end, for (; dist_past_vert_to_insert_point < p0p1_length && n_points_generated < n_dots_in_between; dist_past_vert_to_insert_point += wipe_point_dist) { - result.emplace_back(p0 + normal(p0p1, dist_past_vert_to_insert_point), vert.point_idx_, poly); + result.emplace_back(p0 + normal(p0p1, dist_past_vert_to_insert_point), vert.point_idx_, &poly); n_points_generated++; } dist_past_vert_to_insert_point -= p0p1_length; @@ -92,21 +95,21 @@ void PolygonUtils::spreadDots(PolygonsPointIndex start, PolygonsPointIndex end, assert(result.size() == n_dots && "we didn't generate as many wipe locations as we asked for."); } -std::vector PolygonUtils::spreadDotsArea(const Polygons& polygons, coord_t grid_size) +std::vector PolygonUtils::spreadDotsArea(const Shape& polygons, coord_t grid_size) { return spreadDotsArea(polygons, Point2LL(grid_size, grid_size)); } -std::vector PolygonUtils::spreadDotsArea(const Polygons& polygons, Point2LL grid_size) +std::vector PolygonUtils::spreadDotsArea(const Shape& polygons, Point2LL grid_size) { std::vector dummy_toolpaths; Settings dummy_settings; Infill infill_gen(EFillMethod::LINES, false, false, polygons, 0, grid_size.X, 0, 1, 0, 0, 0, 0, 0); - Polygons result_polygons; - Polygons result_lines; + Shape result_polygons; + OpenLinesSet result_lines; infill_gen.generate(dummy_toolpaths, result_polygons, result_lines, dummy_settings, 0, SectionType::DOTS); // FIXME: @jellespijker make sure the propper layer nr is used std::vector result; - for (PolygonRef line : result_lines) + for (const OpenPolyline& line : result_lines) { assert(line.size() == 2); Point2LL a = line[0]; @@ -129,7 +132,7 @@ std::vector PolygonUtils::spreadDotsArea(const Polygons& polygons, Poi bool PolygonUtils::lineSegmentPolygonsIntersection( const Point2LL& a, const Point2LL& b, - const Polygons& current_outlines, + const Shape& current_outlines, const LocToLineGrid& outline_locator, Point2LL& result, const coord_t within_max_dist) @@ -179,7 +182,7 @@ bool PolygonUtils::lineSegmentPolygonsIntersection( return closest_dist2 < within_max_dist2; } -Point2LL PolygonUtils::getVertexInwardNormal(ConstPolygonRef poly, unsigned int point_idx) +Point2LL PolygonUtils::getVertexInwardNormal(const Polyline& poly, unsigned int point_idx) { Point2LL p1 = poly[point_idx]; @@ -217,18 +220,19 @@ Point2LL PolygonUtils::getVertexInwardNormal(ConstPolygonRef poly, unsigned int return n; } -Point2LL PolygonUtils::getBoundaryPointWithOffset(ConstPolygonRef poly, unsigned int point_idx, int64_t offset) +Point2LL PolygonUtils::getBoundaryPointWithOffset(const Polyline& poly, unsigned int point_idx, int64_t offset) { return poly[point_idx] + normal(getVertexInwardNormal(poly, point_idx), -offset); } -Point2LL PolygonUtils::moveInsideDiagonally(ClosestPolygonPoint point_on_boundary, int64_t inset) +Point2LL PolygonUtils::moveInsideDiagonally(ClosestPointPolygon point_on_boundary, int64_t inset) { if (! point_on_boundary.isValid()) { return no_point; } - ConstPolygonRef poly = **point_on_boundary.poly_; + + const Polygon& poly = *point_on_boundary.poly_; Point2LL p0 = poly[point_on_boundary.point_idx_]; Point2LL p1 = poly[(point_on_boundary.point_idx_ + 1) % poly.size()]; if (vSize2(p0 - point_on_boundary.location_) < vSize2(p1 - point_on_boundary.location_)) @@ -241,21 +245,21 @@ Point2LL PolygonUtils::moveInsideDiagonally(ClosestPolygonPoint point_on_boundar } } -unsigned int PolygonUtils::moveOutside(const Polygons& polygons, Point2LL& from, int distance, int64_t maxDist2) +unsigned int PolygonUtils::moveOutside(const Shape& polygons, Point2LL& from, int distance, int64_t maxDist2) { return moveInside(polygons, from, -distance, maxDist2); } -ClosestPolygonPoint PolygonUtils::moveInside2( - const Polygons& polygons, +ClosestPointPolygon PolygonUtils::moveInside2( + const Shape& polygons, Point2LL& from, const int distance, const int64_t max_dist2, - const Polygons* loc_to_line_polygons, + const Shape* loc_to_line_polygons, const LocToLineGrid* loc_to_line_grid, const std::function& penalty_function) { - std::optional closest_polygon_point; + std::optional closest_polygon_point; if (loc_to_line_grid) { closest_polygon_point = findClose(from, *loc_to_line_polygons, *loc_to_line_grid, penalty_function); @@ -267,16 +271,16 @@ ClosestPolygonPoint PolygonUtils::moveInside2( return _moveInside2(*closest_polygon_point, distance, from, max_dist2); } -ClosestPolygonPoint PolygonUtils::moveInside2( - const Polygons& loc_to_line_polygons, - ConstPolygonRef polygon, +ClosestPointPolygon PolygonUtils::moveInside2( + const Shape& loc_to_line_polygons, + const Polygon& polygon, Point2LL& from, const int distance, const int64_t max_dist2, const LocToLineGrid* loc_to_line_grid, const std::function& penalty_function) { - std::optional closest_polygon_point; + std::optional closest_polygon_point; if (loc_to_line_grid) { closest_polygon_point = findClose(from, loc_to_line_polygons, *loc_to_line_grid, penalty_function); @@ -288,11 +292,11 @@ ClosestPolygonPoint PolygonUtils::moveInside2( return _moveInside2(*closest_polygon_point, distance, from, max_dist2); } -ClosestPolygonPoint PolygonUtils::_moveInside2(const ClosestPolygonPoint& closest_polygon_point, const int distance, Point2LL& from, const int64_t max_dist2) +ClosestPointPolygon PolygonUtils::_moveInside2(const ClosestPointPolygon& closest_polygon_point, const int distance, Point2LL& from, const int64_t max_dist2) { if (! closest_polygon_point.isValid()) { - return ClosestPolygonPoint(); // stub with invalid indices to signify we haven't found any + return ClosestPointPolygon(); // stub with invalid indices to signify we haven't found any } const Point2LL v_boundary_from = from - closest_polygon_point.location_; Point2LL result = moveInside(closest_polygon_point, distance); @@ -314,7 +318,7 @@ ClosestPolygonPoint PolygonUtils::_moveInside2(const ClosestPolygonPoint& closes { if (vSize2(v_boundary_from) > max_dist2) { - return ClosestPolygonPoint(*closest_polygon_point.poly_); // stub with invalid indices to signify we haven't found any + return ClosestPointPolygon(closest_polygon_point.poly_); // stub with invalid indices to signify we haven't found any } else { @@ -327,7 +331,7 @@ ClosestPolygonPoint PolygonUtils::_moveInside2(const ClosestPolygonPoint& closes /* * Implementation assumes moving inside, but moving outside should just as well be possible. */ -size_t PolygonUtils::moveInside(const Polygons& polygons, Point2LL& from, int distance, int64_t maxDist2) +size_t PolygonUtils::moveInside(const Shape& polygons, Point2LL& from, int distance, int64_t maxDist2) { Point2LL ret = from; int64_t bestDist2 = std::numeric_limits::max(); @@ -335,7 +339,7 @@ size_t PolygonUtils::moveInside(const Polygons& polygons, Point2LL& from, int di bool is_already_on_correct_side_of_boundary = false; // whether [from] is already on the right side of the boundary for (size_t poly_idx = 0; poly_idx < polygons.size(); poly_idx++) { - ConstPolygonRef poly = polygons[poly_idx]; + const Polygon& poly = polygons[poly_idx]; if (poly.size() < 2) continue; Point2LL p0 = poly[poly.size() - 2]; @@ -447,7 +451,7 @@ size_t PolygonUtils::moveInside(const Polygons& polygons, Point2LL& from, int di } // Version that works on single PolygonRef. -unsigned int PolygonUtils::moveInside(const ConstPolygonRef polygon, Point2LL& from, int distance, int64_t maxDist2) +unsigned int PolygonUtils::moveInside(const ClosedPolyline& polygon, Point2LL& from, int distance, int64_t maxDist2) { // TODO: This is copied from the moveInside of Polygons. /* @@ -565,12 +569,12 @@ unsigned int PolygonUtils::moveInside(const ConstPolygonRef polygon, Point2LL& f return 0; } -Point2LL PolygonUtils::moveOutside(const ClosestPolygonPoint& cpp, const int distance) +Point2LL PolygonUtils::moveOutside(const ClosestPointPolygon& cpp, const int distance) { return moveInside(cpp, -distance); } -Point2LL PolygonUtils::moveInside(const ClosestPolygonPoint& cpp, const int distance) +Point2LL PolygonUtils::moveInside(const ClosestPointPolygon& cpp, const int distance) { if (! cpp.isValid()) { @@ -580,7 +584,7 @@ Point2LL PolygonUtils::moveInside(const ClosestPolygonPoint& cpp, const int dist { // the point which is assumed to be on the boundary doesn't have to be moved return cpp.location_; } - ConstPolygonRef poly = *cpp.poly_; + const Polyline& poly = *cpp.poly_; unsigned int point_idx = cpp.point_idx_; const Point2LL& on_boundary = cpp.location_; @@ -616,33 +620,33 @@ Point2LL PolygonUtils::moveInside(const ClosestPolygonPoint& cpp, const int dist } } -ClosestPolygonPoint PolygonUtils::ensureInsideOrOutside( - const Polygons& polygons, +ClosestPointPolygon PolygonUtils::ensureInsideOrOutside( + const Shape& polygons, Point2LL& from, int preferred_dist_inside, int64_t max_dist2, - const Polygons* loc_to_line_polygons, + const Shape* loc_to_line_polygons, const LocToLineGrid* loc_to_line_grid, const std::function& penalty_function) { - const ClosestPolygonPoint closest_polygon_point = moveInside2(polygons, from, preferred_dist_inside, max_dist2, loc_to_line_polygons, loc_to_line_grid, penalty_function); + const ClosestPointPolygon closest_polygon_point = moveInside2(polygons, from, preferred_dist_inside, max_dist2, loc_to_line_polygons, loc_to_line_grid, penalty_function); return ensureInsideOrOutside(polygons, from, closest_polygon_point, preferred_dist_inside, loc_to_line_polygons, loc_to_line_grid, penalty_function); } -ClosestPolygonPoint PolygonUtils::ensureInsideOrOutside( - const Polygons& polygons, +ClosestPointPolygon PolygonUtils::ensureInsideOrOutside( + const Shape& polygons, Point2LL& from, - const ClosestPolygonPoint& closest_polygon_point, + const ClosestPointPolygon& closest_polygon_point, int preferred_dist_inside, - const Polygons* loc_to_line_polygons, + const Shape* loc_to_line_polygons, const LocToLineGrid* loc_to_line_grid, const std::function& penalty_function) { if (! closest_polygon_point.isValid()) { - return ClosestPolygonPoint(); // we couldn't move inside + return ClosestPointPolygon(); // we couldn't move inside } - ConstPolygonRef closest_poly = *closest_polygon_point.poly_; + const Polygon& closest_poly = *closest_polygon_point.poly_; bool is_outside_boundary = closest_poly.orientation(); { @@ -670,13 +674,13 @@ ClosestPolygonPoint PolygonUtils::ensureInsideOrOutside( // The offset is performed on the closest reference polygon in order to save computation time { const coord_t offset = (is_outside_boundary) ? -preferred_dist_inside : preferred_dist_inside; // perform inset on outer boundary and outset on holes - Polygons insetted + Shape insetted = closest_poly.offset(offset / 2); // perform less inset, because chances are (thin parts of) the polygon will disappear, given that moveInside did an overshoot if (insetted.size() == 0) { - return ClosestPolygonPoint(); // we couldn't move inside + return ClosestPointPolygon(); // we couldn't move inside } - ClosestPolygonPoint inside = findClosest(from, insetted, penalty_function); + ClosestPointPolygon inside = findClosest(from, insetted, penalty_function); if (inside.isValid()) { bool is_inside = polygons.inside(inside.location_); @@ -684,73 +688,29 @@ ClosestPolygonPoint PolygonUtils::ensureInsideOrOutside( { // Insetting from the reference polygon ended up outside another polygon. // Perform an offset on all polygons instead. - Polygons all_insetted = polygons.offset(-preferred_dist_inside); - ClosestPolygonPoint overall_inside = findClosest(from, all_insetted, penalty_function); + Shape all_insetted = polygons.offset(-preferred_dist_inside); + ClosestPointPolygon overall_inside = findClosest(from, all_insetted, penalty_function); bool overall_is_inside = polygons.inside(overall_inside.location_); if (overall_is_inside != (preferred_dist_inside > 0)) { -#ifdef DEBUG - static bool has_run = false; - if (! has_run) - { - try - { - int offset_performed = offset / 2; - AABB aabb(polygons); - aabb.expand(std::abs(preferred_dist_inside) * 2); - SVG svg("debug.html", aabb); - svg.writeComment("Original polygon in black"); - svg.writePolygons(polygons, SVG::Color::BLACK); - for (auto poly : polygons) - { - for (auto point : poly) - { - svg.writePoint(point, true, 2); - } - } - std::stringstream ss; - svg.writeComment("Reference polygon in yellow"); - svg.writePolygon(closest_poly, SVG::Color::YELLOW); - ss << "Offsetted polygon in blue with offset " << offset_performed; - svg.writeComment(ss.str()); - svg.writePolygons(insetted, SVG::Color::BLUE); - for (auto poly : insetted) - { - for (auto point : poly) - { - svg.writePoint(point, true, 2); - } - } - svg.writeComment("From location"); - svg.writePoint(from, true, 5, SVG::Color::GREEN); - svg.writeComment("Location computed to be inside the black polygon"); - svg.writePoint(inside.location_, true, 5, SVG::Color::RED); - } - catch (...) - { - } - spdlog::error("Clipper::offset failed. See generated debug.html! Black is original Blue is offsetted polygon"); - has_run = true; - } -#endif - return ClosestPolygonPoint(); + return ClosestPointPolygon(); } inside = overall_inside; } from = inside.location_; } // otherwise we just return the closest polygon point without modifying the from location - return closest_polygon_point; // don't return a ClosestPolygonPoint with a reference to the above local polygons variable + return closest_polygon_point; // don't return a ClosestPoint with a reference to the above local polygons variable } } -void PolygonUtils::walkToNearestSmallestConnection(ClosestPolygonPoint& poly1_result, ClosestPolygonPoint& poly2_result) +void PolygonUtils::walkToNearestSmallestConnection(ClosestPointPolygon& poly1_result, ClosestPointPolygon& poly2_result) { if (! poly1_result.isValid() || ! poly2_result.isValid()) { return; } - ConstPolygonRef poly1 = *poly1_result.poly_; - ConstPolygonRef poly2 = *poly2_result.poly_; + const Polygon& poly1 = *poly1_result.poly_; + const Polygon& poly2 = *poly2_result.poly_; size_t poly1_idx = poly1_result.poly_idx_; size_t poly2_idx = poly2_result.poly_idx_; if (poly1_result.point_idx_ == NO_INDEX || poly2_result.point_idx_ == NO_INDEX) @@ -781,10 +741,10 @@ void PolygonUtils::walkToNearestSmallestConnection(ClosestPolygonPoint& poly1_re // o o >> should find connection here coord_t best_distance2 = vSize2(poly1_result.p() - poly2_result.p()); auto check_neighboring_vert - = [&best_distance2](ConstPolygonRef from_poly, ConstPolygonRef to_poly, ClosestPolygonPoint& from_poly_result, ClosestPolygonPoint& to_poly_result, bool vertex_after) + = [&best_distance2](const Polygon& from_poly, const Polygon& to_poly, ClosestPointPolygon& from_poly_result, ClosestPointPolygon& to_poly_result, bool vertex_after) { const Point2LL after_poly2_result = to_poly[(to_poly_result.point_idx_ + vertex_after) % to_poly.size()]; - const ClosestPolygonPoint poly1_after_poly2_result = findNearestClosest(after_poly2_result, from_poly, from_poly_result.point_idx_); + const ClosestPointPolygon poly1_after_poly2_result = findNearestClosest(after_poly2_result, from_poly, from_poly_result.point_idx_); const coord_t poly1_after_poly2_result_dist2 = vSize2(poly1_after_poly2_result.p() - after_poly2_result); if (poly1_after_poly2_result_dist2 < best_distance2) { @@ -802,14 +762,14 @@ void PolygonUtils::walkToNearestSmallestConnection(ClosestPolygonPoint& poly1_re poly2_result.poly_idx_ = poly2_idx; } -ClosestPolygonPoint PolygonUtils::findNearestClosest(Point2LL from, ConstPolygonRef polygon, int start_idx) +ClosestPointPolygon PolygonUtils::findNearestClosest(Point2LL from, const Polygon& polygon, int start_idx) { - ClosestPolygonPoint forth = findNearestClosest(from, polygon, start_idx, 1); + ClosestPointPolygon forth = findNearestClosest(from, polygon, start_idx, 1); if (! forth.isValid()) { return forth; // stop computation } - ClosestPolygonPoint back = findNearestClosest(from, polygon, start_idx, -1); + ClosestPointPolygon back = findNearestClosest(from, polygon, start_idx, -1); assert(back.isValid()); if (vSize2(forth.location_ - from) < vSize2(back.location_ - from)) { @@ -821,11 +781,11 @@ ClosestPolygonPoint PolygonUtils::findNearestClosest(Point2LL from, ConstPolygon } } -ClosestPolygonPoint PolygonUtils::findNearestClosest(Point2LL from, ConstPolygonRef polygon, int start_idx, int direction) +ClosestPointPolygon PolygonUtils::findNearestClosest(Point2LL from, const Polygon& polygon, int start_idx, int direction) { if (polygon.size() == 0) { - return ClosestPolygonPoint(polygon); + return ClosestPointPolygon(&polygon); } Point2LL aPoint = polygon[0]; Point2LL best = aPoint; @@ -851,28 +811,28 @@ ClosestPolygonPoint PolygonUtils::findNearestClosest(Point2LL from, ConstPolygon } else { - return ClosestPolygonPoint(best, bestPos, polygon); + return ClosestPointPolygon(best, bestPos, &polygon); } } - return ClosestPolygonPoint(best, bestPos, polygon); + return ClosestPointPolygon(best, bestPos, &polygon); } -ClosestPolygonPoint PolygonUtils::findClosest(Point2LL from, const Polygons& polygons, const std::function& penalty_function) +ClosestPointPolygon PolygonUtils::findClosest(Point2LL from, const Shape& polygons, const std::function& penalty_function) { - ClosestPolygonPoint none; + ClosestPointPolygon none; if (polygons.size() == 0) { return none; } - ConstPolygonPointer any_polygon = polygons[0]; + const Polygon* any_polygon = &(polygons[0]); unsigned int any_poly_idx; for (any_poly_idx = 0; any_poly_idx < polygons.size(); any_poly_idx++) { // find first point in all polygons if (polygons[any_poly_idx].size() > 0) { - any_polygon = polygons[any_poly_idx]; + any_polygon = &(polygons[any_poly_idx]); break; } } @@ -880,16 +840,16 @@ ClosestPolygonPoint PolygonUtils::findClosest(Point2LL from, const Polygons& pol { return none; } - ClosestPolygonPoint best((*any_polygon)[0], 0, *any_polygon, any_poly_idx); + ClosestPointPolygon best((*any_polygon)[0], 0, any_polygon, any_poly_idx); int64_t closestDist2_score = vSize2(from - best.location_) + penalty_function(best.location_); for (unsigned int ply = 0; ply < polygons.size(); ply++) { - ConstPolygonRef poly = polygons[ply]; + const Polygon& poly = polygons[ply]; if (poly.size() == 0) continue; - ClosestPolygonPoint closest_here = findClosest(from, poly, penalty_function); + ClosestPointPolygon closest_here = findClosest(from, poly, penalty_function); if (! closest_here.isValid()) { continue; @@ -906,11 +866,11 @@ ClosestPolygonPoint PolygonUtils::findClosest(Point2LL from, const Polygons& pol return best; } -ClosestPolygonPoint PolygonUtils::findClosest(Point2LL from, ConstPolygonRef polygon, const std::function& penalty_function) +ClosestPointPolygon PolygonUtils::findClosest(Point2LL from, const Polygon& polygon, const std::function& penalty_function) { if (polygon.size() == 0) { - return ClosestPolygonPoint(polygon); + return ClosestPointPolygon(&polygon); } Point2LL aPoint = polygon[0]; Point2LL best = aPoint; @@ -937,16 +897,16 @@ ClosestPolygonPoint PolygonUtils::findClosest(Point2LL from, ConstPolygonRef pol } } - return ClosestPolygonPoint(best, bestPos, polygon); + return ClosestPointPolygon(best, bestPos, &polygon); } -PolygonsPointIndex PolygonUtils::findNearestVert(const Point2LL from, const Polygons& polys) +PolygonsPointIndex PolygonUtils::findNearestVert(const Point2LL from, const Shape& polys) { int64_t best_dist2 = std::numeric_limits::max(); PolygonsPointIndex closest_vert; for (unsigned int poly_idx = 0; poly_idx < polys.size(); poly_idx++) { - ConstPolygonRef poly = polys[poly_idx]; + const Polygon& poly = polys[poly_idx]; for (unsigned int point_idx = 0; point_idx < poly.size(); point_idx++) { int64_t dist2 = vSize2(poly[point_idx] - from); @@ -960,7 +920,7 @@ PolygonsPointIndex PolygonUtils::findNearestVert(const Point2LL from, const Poly return closest_vert; } -unsigned int PolygonUtils::findNearestVert(const Point2LL from, ConstPolygonRef poly) +unsigned int PolygonUtils::findNearestVert(const Point2LL from, const Polygon& poly) { int64_t best_dist2 = std::numeric_limits::max(); unsigned int closest_vert_idx = -1; @@ -976,7 +936,7 @@ unsigned int PolygonUtils::findNearestVert(const Point2LL from, ConstPolygonRef return closest_vert_idx; } -std::unique_ptr PolygonUtils::createLocToLineGrid(const Polygons& polygons, int square_size) +std::unique_ptr PolygonUtils::createLocToLineGrid(const Shape& polygons, int square_size) { unsigned int n_points = 0; for (const auto& poly : polygons) @@ -988,7 +948,7 @@ std::unique_ptr PolygonUtils::createLocToLineGrid(const Polygons& for (unsigned int poly_idx = 0; poly_idx < polygons.size(); poly_idx++) { - ConstPolygonRef poly = polygons[poly_idx]; + const Polygon& poly = polygons[poly_idx]; for (unsigned int point_idx = 0; point_idx < poly.size(); point_idx++) { ret->insert(PolygonsPointIndex(&polygons, poly_idx, point_idx)); @@ -1004,8 +964,8 @@ std::unique_ptr PolygonUtils::createLocToLineGrid(const Polygons& * * We could skip the duplication by keeping a vector of vectors of bools. */ -std::optional - PolygonUtils::findClose(Point2LL from, const Polygons& polygons, const LocToLineGrid& loc_to_line, const std::function& penalty_function) +std::optional + PolygonUtils::findClose(Point2LL from, const Shape& polygons, const LocToLineGrid& loc_to_line, const std::function& penalty_function) { std::vector near_lines = loc_to_line.getNearby(from, loc_to_line.getCellSize()); @@ -1015,7 +975,7 @@ std::optional PolygonsPointIndex best_point_poly_idx(nullptr, NO_INDEX, NO_INDEX); for (PolygonsPointIndex& point_poly_index : near_lines) { - ConstPolygonRef poly = polygons[point_poly_index.poly_idx_]; + const Polygon& poly = polygons[point_poly_index.poly_idx_]; const Point2LL& p1 = poly[point_poly_index.point_idx_]; const Point2LL& p2 = poly[(point_poly_index.point_idx_ + 1) % poly.size()]; @@ -1030,28 +990,28 @@ std::optional } if (best_point_poly_idx.poly_idx_ == NO_INDEX) { - return std::optional(); + return std::optional(); } else { - return std::optional(std::in_place, best, best_point_poly_idx.point_idx_, polygons[best_point_poly_idx.poly_idx_], best_point_poly_idx.poly_idx_); + return std::optional(std::in_place, best, best_point_poly_idx.point_idx_, &(polygons[best_point_poly_idx.poly_idx_]), best_point_poly_idx.poly_idx_); } } -std::vector> - PolygonUtils::findClose(ConstPolygonRef from, const Polygons& destination, const LocToLineGrid& destination_loc_to_line, const std::function& penalty_function) +std::vector> + PolygonUtils::findClose(const Polygon& from, const Shape& destination, const LocToLineGrid& destination_loc_to_line, const std::function& penalty_function) { - std::vector> ret; + std::vector> ret; int p0_idx = from.size() - 1; Point2LL p0(from[p0_idx]); int grid_size = destination_loc_to_line.getCellSize(); for (unsigned int p1_idx = 0; p1_idx < from.size(); p1_idx++) { const Point2LL& p1 = from[p1_idx]; - std::optional best_here = findClose(p1, destination, destination_loc_to_line, penalty_function); + std::optional best_here = findClose(p1, destination, destination_loc_to_line, penalty_function); if (best_here) { - ret.push_back(std::make_pair(ClosestPolygonPoint(p1, p1_idx, from), *best_here)); + ret.push_back(std::make_pair(ClosestPointPolygon(p1, p1_idx, &from), *best_here)); } Point2LL p0p1 = p1 - p0; int dist_to_p1 = vSize(p0p1); @@ -1063,7 +1023,7 @@ std::vector> best_here = findClose(x, destination, destination_loc_to_line, penalty_function); if (best_here) { - ret.push_back(std::make_pair(ClosestPolygonPoint(x, p0_idx, from), *best_here)); + ret.push_back(std::make_pair(ClosestPointPolygon(x, p0_idx, &from), *best_here)); } } p0 = p1; @@ -1072,7 +1032,7 @@ std::vector> return ret; } -bool PolygonUtils::getNextPointWithDistance(Point2LL from, int64_t dist, ConstPolygonRef poly, int start_idx, int poly_start_idx, GivenDistPoint& result) +bool PolygonUtils::getNextPointWithDistance(Point2LL from, int64_t dist, const OpenPolyline& poly, int start_idx, int poly_start_idx, GivenDistPoint& result) { Point2LL prev_poly_point = poly[(start_idx + poly_start_idx) % poly.size()]; @@ -1146,9 +1106,10 @@ bool PolygonUtils::getNextPointWithDistance(Point2LL from, int64_t dist, ConstPo return false; } -ClosestPolygonPoint PolygonUtils::walk(const ClosestPolygonPoint& from, coord_t distance) +template +ClosestPoint PolygonUtils::walk(const ClosestPoint& from, coord_t distance) { - ConstPolygonRef poly = *from.poly_; + const LineType& poly = *from.poly_; Point2LL last_vertex = from.p(); Point2LL next_vertex; size_t last_point_idx = from.point_idx_; @@ -1166,10 +1127,10 @@ ClosestPolygonPoint PolygonUtils::walk(const ClosestPolygonPoint& from, coord_t last_point_idx = point_idx; } Point2LL result = next_vertex + normal(last_vertex - next_vertex, -distance); - return ClosestPolygonPoint(result, last_point_idx, poly, from.poly_idx_); + return ClosestPoint(result, last_point_idx, &poly, from.poly_idx_); } -std::optional PolygonUtils::getNextParallelIntersection(const ClosestPolygonPoint& start, const Point2LL& line_to, const coord_t dist, const bool forward) +std::optional PolygonUtils::getNextParallelIntersection(const ClosestPointPolygon& start, const Point2LL& line_to, const coord_t dist, const bool forward) { // <--o--t-----y----< poly 1 // : : @@ -1183,7 +1144,7 @@ std::optional PolygonUtils::getNextParallelIntersection(con // r=result // t=line_to - ConstPolygonRef poly = *start.poly_; + const Polygon& poly = *start.poly_; const Point2LL s = start.p(); const Point2LL t = line_to; @@ -1217,14 +1178,14 @@ std::optional PolygonUtils::getNextParallelIntersection(con vert_before_idx = (next_point_idx > 0) ? vert_before_idx - 1 : poly.size() - 1; } assert(vert_before_idx < poly.size()); - return ClosestPolygonPoint(intersection, vert_before_idx, poly); + return ClosestPointPolygon(intersection, vert_before_idx, &poly); } prev_vert = next_vert; prev_projected = projected; } - return std::optional(); + return std::optional(); } @@ -1267,7 +1228,7 @@ bool PolygonUtils::polygonCollidesWithLineSegment(const Point2LL from, const Poi } bool PolygonUtils::polygonCollidesWithLineSegment( - ConstPolygonRef poly, + const Polygon& poly, const Point2LL& transformed_startPoint, const Point2LL& transformed_endPoint, PointMatrix transformation_matrix) @@ -1285,7 +1246,7 @@ bool PolygonUtils::polygonCollidesWithLineSegment( return false; } -bool PolygonUtils::polygonCollidesWithLineSegment(ConstPolygonRef poly, const Point2LL& startPoint, const Point2LL& endPoint) +bool PolygonUtils::polygonCollidesWithLineSegment(const Polygon& poly, const Point2LL& startPoint, const Point2LL& endPoint) { Point2LL diff = endPoint - startPoint; @@ -1297,12 +1258,12 @@ bool PolygonUtils::polygonCollidesWithLineSegment(ConstPolygonRef poly, const Po } bool PolygonUtils::polygonCollidesWithLineSegment( - const Polygons& polys, + const Shape& polys, const Point2LL& transformed_startPoint, const Point2LL& transformed_endPoint, PointMatrix transformation_matrix) { - for (ConstPolygonRef poly : polys) + for (const Polygon& poly : polys) { if (poly.size() == 0) { @@ -1318,7 +1279,7 @@ bool PolygonUtils::polygonCollidesWithLineSegment( } -bool PolygonUtils::polygonCollidesWithLineSegment(const Polygons& polys, const Point2LL& startPoint, const Point2LL& endPoint) +bool PolygonUtils::polygonCollidesWithLineSegment(const Shape& polys, const Point2LL& startPoint, const Point2LL& endPoint) { if (endPoint == startPoint) { @@ -1333,7 +1294,7 @@ bool PolygonUtils::polygonCollidesWithLineSegment(const Polygons& polys, const P return polygonCollidesWithLineSegment(polys, transformed_startPoint, transformed_endPoint, transformation_matrix); } -bool PolygonUtils::polygonsIntersect(const ConstPolygonRef& poly_a, const ConstPolygonRef& poly_b) +bool PolygonUtils::polygonsIntersect(const Polygon& poly_a, const Polygon& poly_b) { // only do the full intersection when the polys' BBs overlap AABB bba(poly_a); @@ -1341,7 +1302,7 @@ bool PolygonUtils::polygonsIntersect(const ConstPolygonRef& poly_a, const ConstP return bba.hit(bbb) && poly_a.intersection(poly_b).size() > 0; } -bool PolygonUtils::polygonOutlinesAdjacent(const ConstPolygonRef inner_poly, const ConstPolygonRef outer_poly, const coord_t max_gap) +bool PolygonUtils::polygonOutlinesAdjacent(const Polygon& inner_poly, const Polygon& outer_poly, const coord_t max_gap) { // Heuristic check if their AABBs are near first. AABB inner_aabb(inner_poly); @@ -1373,8 +1334,8 @@ bool PolygonUtils::polygonOutlinesAdjacent(const ConstPolygonRef inner_poly, con void PolygonUtils::findAdjacentPolygons( std::vector& adjacent_poly_indices, - const ConstPolygonRef& poly, - const std::vector& possible_adjacent_polys, + const Polygon& poly, + const std::vector& possible_adjacent_polys, const coord_t max_gap) { // given a polygon, and a vector of polygons, return a vector containing the indices of the polygons that are adjacent to the given polygon @@ -1387,7 +1348,7 @@ void PolygonUtils::findAdjacentPolygons( } } -double PolygonUtils::relativeHammingDistance(const Polygons& poly_a, const Polygons& poly_b) +double PolygonUtils::relativeHammingDistance(const Shape& poly_a, const Shape& poly_b) { const double area_a = std::abs(poly_a.area()); const double area_b = std::abs(poly_b.area()); @@ -1397,7 +1358,7 @@ double PolygonUtils::relativeHammingDistance(const Polygons& poly_a, const Polyg constexpr bool borders_allowed = true; if (total_area == 0.0) { - for (const ConstPolygonRef& polygon_a : poly_a) + for (const Polygon& polygon_a : poly_a) { for (Point2LL point : polygon_a) { @@ -1407,7 +1368,7 @@ double PolygonUtils::relativeHammingDistance(const Polygons& poly_a, const Polyg } } } - for (const ConstPolygonRef& polygon_b : poly_b) + for (const Polygon& polygon_b : poly_b) { for (Point2LL point : polygon_b) { @@ -1420,7 +1381,7 @@ double PolygonUtils::relativeHammingDistance(const Polygons& poly_a, const Polyg return 0.0; // All points are inside the other polygon, regardless of where the vertices are along the edges. } - const Polygons symmetric_difference = poly_a.xorPolygons(poly_b); + const Shape symmetric_difference = poly_a.xorPolygons(poly_b); const double hamming_distance = symmetric_difference.area(); return hamming_distance / total_area; } @@ -1472,18 +1433,18 @@ Polygon PolygonUtils::makeWheel(const Point2LL& mid, const coord_t inner_radius, } -Polygons PolygonUtils::connect(const Polygons& input) +Shape PolygonUtils::connect(const Shape& input) { - Polygons ret; - std::vector parts = input.splitIntoParts(true); - for (PolygonsPart& part : parts) + Shape ret; + std::vector parts = input.splitIntoParts(true); + for (SingleShape& part : parts) { - PolygonRef outline = part.outerPolygon(); + Polygon& outline = part.outerPolygon(); for (size_t hole_idx = 1; hole_idx < part.size(); hole_idx++) { - PolygonRef hole = part[hole_idx]; + Polygon& hole = part[hole_idx]; Point2LL hole_point = hole[0]; - hole.add(hole_point); + hole.push_back(hole_point); // find where the scanline passes the Y size_t best_segment_to_idx = 0; coord_t best_dist = std::numeric_limits::max(); @@ -1506,21 +1467,21 @@ Polygons PolygonUtils::connect(const Polygons& input) } prev = here; } - (*outline).insert(outline.begin() + best_segment_to_idx, 2, best_intersection_point); - (*outline).insert(outline.begin() + best_segment_to_idx + 1, hole.begin(), hole.end()); + outline.insert(outline.begin() + best_segment_to_idx, 2, best_intersection_point); + outline.insert(outline.begin() + best_segment_to_idx + 1, hole.begin(), hole.end()); } - ret.add(outline); + ret.push_back(outline); } return ret; } /* Note: Also tries to solve for near-self intersections, when epsilon >= 1 */ -void PolygonUtils::fixSelfIntersections(const coord_t epsilon, Polygons& thiss) +void PolygonUtils::fixSelfIntersections(const coord_t epsilon, Shape& polygon) { if (epsilon < 1) { - ClipperLib::SimplifyPolygons(thiss.paths); + polygon.simplify(); return; } @@ -1529,32 +1490,32 @@ void PolygonUtils::fixSelfIntersections(const coord_t epsilon, Polygons& thiss) // Points too close to line segments should be moved a little away from those line segments, but less than epsilon, // so at least half-epsilon distance between points can still be guaranteed. constexpr coord_t grid_size = 2000; - auto query_grid = PolygonUtils::createLocToLineGrid(thiss, grid_size); + auto query_grid = PolygonUtils::createLocToLineGrid(polygon, grid_size); const coord_t move_dist = half_epsilon - 2; const coord_t half_epsilon_sqrd = half_epsilon * half_epsilon; - const size_t n = thiss.size(); + const size_t n = polygon.size(); for (size_t poly_idx = 0; poly_idx < n; poly_idx++) { - const size_t pathlen = thiss[poly_idx].size(); + const size_t pathlen = polygon[poly_idx].size(); for (size_t point_idx = 0; point_idx < pathlen; ++point_idx) { - Point2LL& pt = thiss[poly_idx][point_idx]; + Point2LL& pt = polygon[poly_idx][point_idx]; for (const auto& line : query_grid->getNearby(pt, epsilon * 2)) { - const size_t line_next_idx = (line.point_idx_ + 1) % thiss[line.poly_idx_].size(); + const size_t line_next_idx = (line.point_idx_ + 1) % polygon[line.poly_idx_].size(); if (poly_idx == line.poly_idx_ && (point_idx == line.point_idx_ || point_idx == line_next_idx)) { continue; } - const Point2LL& a = thiss[line.poly_idx_][line.point_idx_]; - const Point2LL& b = thiss[line.poly_idx_][line_next_idx]; + const Point2LL& a = polygon[line.poly_idx_][line.point_idx_]; + const Point2LL& b = polygon[line.poly_idx_][line_next_idx]; if (half_epsilon_sqrd >= vSize2(pt - LinearAlg2D::getClosestOnLineSegment(pt, a, b))) { - const Point2LL& other = thiss[poly_idx][(point_idx + 1) % pathlen]; + const Point2LL& other = polygon[poly_idx][(point_idx + 1) % pathlen]; const Point2LL vec = LinearAlg2D::pointIsLeftOfLine(other, a, b) > 0 ? b - a : a - b; const coord_t len = vSize(vec); pt.X += (-vec.Y * move_dist) / len; @@ -1564,29 +1525,29 @@ void PolygonUtils::fixSelfIntersections(const coord_t epsilon, Polygons& thiss) } } - ClipperLib::SimplifyPolygons(thiss.paths); + polygon.simplify(); } -Polygons PolygonUtils::unionManySmall(const Polygons& p) +Shape PolygonUtils::unionManySmall(const Shape& polygon) { - if (p.paths.size() < 8) + if (polygon.size() < 8) { - return p.unionPolygons(); + return polygon.unionPolygons(); } - Polygons a, b; - a.paths.reserve(p.paths.size() / 2); - b.paths.reserve(a.paths.size() + 1); - for (const auto& [i, path] : p.paths | ranges::views::enumerate) + Shape a, b; + a.reserve(polygon.size() / 2); + b.reserve(a.size() + 1); + for (const auto& [i, path] : polygon | ranges::views::enumerate) { - (i % 2 == 0 ? b : a).paths.push_back(path); + (i % 2 == 0 ? b : a).push_back(path); } return unionManySmall(a).unionPolygons(unionManySmall(b)); } -Polygons PolygonUtils::clipPolygonWithAABB(const Polygons& src, const AABB& aabb) +Shape PolygonUtils::clipPolygonWithAABB(const Shape& src, const AABB& aabb) { - Polygons out; + Shape out; out.reserve(src.size()); for (const auto path : src) { @@ -1595,7 +1556,7 @@ Polygons PolygonUtils::clipPolygonWithAABB(const Polygons& src, const AABB& aabb const size_t cnt = path.size(); if (cnt < 3) { - return Polygons(); + return Shape(); } enum class Side @@ -1625,7 +1586,7 @@ Polygons PolygonUtils::clipPolygonWithAABB(const Polygons& src, const AABB& aabb // the edge possibly cuts corner of the bounding box. (sides_prev & sides_this & sides_next) == 0) { - poly.add(path[i]); + poly.push_back(path[i]); sides_prev = sides_this; } else @@ -1643,43 +1604,46 @@ Polygons PolygonUtils::clipPolygonWithAABB(const Polygons& src, const AABB& aabb if (sides_this == 0 || // The last point is inside. Take it. (sides_prev & sides_this & sides_next) == 0) // Either this point is outside and previous or next is inside, or the edge possibly cuts corner of the bounding box. - poly.add(path.back()); + poly.push_back(path.back()); } if (! poly.empty()) { - out.add(poly); + out.push_back(poly); } } return out; } -Polygons PolygonUtils::generateOutset(const Polygons& inner_poly, size_t count, coord_t line_width) +Shape PolygonUtils::generateOutset(const Shape& inner_poly, size_t count, coord_t line_width) { - Polygons outset; + Shape outset; - Polygons current_outset; + Shape current_outset; for (size_t index = 0; index < count; ++index) { current_outset = index == 0 ? inner_poly.offset(line_width / 2) : current_outset.offset(line_width); - outset.add(current_outset); + outset.push_back(current_outset); } return outset; } -Polygons PolygonUtils::generateInset(const Polygons& outer_poly, coord_t line_width, coord_t initial_inset) +Shape PolygonUtils::generateInset(const Shape& outer_poly, coord_t line_width, coord_t initial_inset) { - Polygons inset; + Shape inset; - Polygons current_inset = outer_poly.offset(-(initial_inset + line_width / 2)); + Shape current_inset = outer_poly.offset(-(initial_inset + line_width / 2)); while (! current_inset.empty()) { - inset.add(current_inset); + inset.push_back(current_inset); current_inset = current_inset.offset(-line_width); } return inset; } +template ClosestPoint PolygonUtils::walk(const ClosestPoint& from, coord_t distance); +template ClosestPoint PolygonUtils::walk(const ClosestPoint& from, coord_t distance); + } // namespace cura diff --git a/stress_benchmark/stress_benchmark.cpp b/stress_benchmark/stress_benchmark.cpp index c2dc5614bf..cd3cd4ce33 100644 --- a/stress_benchmark/stress_benchmark.cpp +++ b/stress_benchmark/stress_benchmark.cpp @@ -20,12 +20,13 @@ #include #include "WallsComputation.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" #include "rapidjson/document.h" #include "rapidjson/stringbuffer.h" #include "rapidjson/writer.h" #include "settings/Settings.h" #include "sliceDataStorage.h" -#include "utils/polygon.h" constexpr std::string_view USAGE = R"(Stress Benchmark. @@ -53,7 +54,7 @@ struct Resource return wkt_file.stem().string(); } - std::vector polygons() const + std::vector polygons() const { using point_type = boost::geometry::model::d2::point_xy; using polygon_type = boost::geometry::model::polygon; @@ -71,27 +72,27 @@ struct Resource boost::geometry::read_wkt(buffer.str(), boost_polygons); - std::vector polygons; + std::vector polygons; for (const auto& boost_polygon : boost_polygons) { - cura::Polygons polygon; + cura::Shape polygon; cura::Polygon outer; for (const auto& point : boost_polygon.outer()) { - outer.add(cura::Point2LL(point.x(), point.y())); + outer.push_back(cura::Point2LL(point.x(), point.y())); } - polygon.add(outer); + polygon.push_back(outer); for (const auto& hole : boost_polygon.inners()) { cura::Polygon inner; for (const auto& point : hole) { - inner.add(cura::Point2LL(point.x(), point.y())); + inner.push_back(cura::Point2LL(point.x(), point.y())); } - polygon.add(inner); + polygon.push_back(inner); } polygons.push_back(polygon); @@ -145,11 +146,11 @@ std::vector getResources() void handleChildProcess(const auto& shapes, const auto& settings) { cura::SliceLayer layer; - for (const cura::Polygons& shape : shapes) + for (const cura::Shape& shape : shapes) { layer.parts.emplace_back(); cura::SliceLayerPart& part = layer.parts.back(); - part.outline.add(shape); + part.outline.push_back(shape); } cura::LayerIndex layer_idx(100); cura::WallsComputation walls_computation(settings, layer_idx); diff --git a/tests/ClipperTest.cpp b/tests/ClipperTest.cpp index 14e54ac6ed..7bcc755eda 100644 --- a/tests/ClipperTest.cpp +++ b/tests/ClipperTest.cpp @@ -9,7 +9,7 @@ // #define TEST_INFILL_SVG_OUTPUT #ifdef TEST_INFILL_SVG_OUTPUT #include "utils/SVG.h" -#include "utils/polygon.h" +#include "geometry/Polygon.h" #include #endif // TEST_INFILL_SVG_OUTPUT diff --git a/tests/InfillTest.cpp b/tests/InfillTest.cpp index 8d95705934..ef6e730fdf 100644 --- a/tests/InfillTest.cpp +++ b/tests/InfillTest.cpp @@ -2,20 +2,25 @@ // CuraEngine is released under the terms of the AGPLv3 or higher #include "infill.h" -#include "ReadTestPolygons.h" -#include "slicer.h" -#include "utils/Coord_t.h" -#include + #include #include -#include +#include #include +#include + +#include "ReadTestPolygons.h" +#include "geometry/OpenPolyline.h" +#include "slicer.h" +#include "utils/Coord_t.h" + // #define TEST_INFILL_SVG_OUTPUT #ifdef TEST_INFILL_SVG_OUTPUT -#include "utils/SVG.h" #include + +#include "utils/SVG.h" #endif // TEST_INFILL_SVG_OUTPUT // NOLINTBEGIN(*-magic-numbers) @@ -69,19 +74,23 @@ class InfillTestParameters // Parameters used to generate the infill: InfillParameters params; - Polygons outline_polygons; + Shape outline_polygons; // Resulting infill: - Polygons result_lines; - Polygons result_polygons; + OpenLinesSet result_lines; + Shape result_polygons; std::string name; - InfillTestParameters() : valid(false), fail_reason("Read of file with test polygons failed (see generateInfillTests), can't continue tests."), params(InfillParameters(EFillMethod::NONE, false, false, 0)), name("UNNAMED") + InfillTestParameters() + : valid(false) + , fail_reason("Read of file with test polygons failed (see generateInfillTests), can't continue tests.") + , params(InfillParameters(EFillMethod::NONE, false, false, 0)) + , name("UNNAMED") { } - InfillTestParameters(const InfillParameters& params, const size_t& test_polygon_id, Polygons outline_polygons, Polygons result_lines, Polygons result_polygons) + InfillTestParameters(const InfillParameters& params, const size_t& test_polygon_id, Shape outline_polygons, OpenLinesSet result_lines, Shape result_polygons) : valid(true) , fail_reason("__") , params(params) @@ -89,7 +98,13 @@ class InfillTestParameters , result_lines(std::move(result_lines)) , result_polygons(std::move(result_polygons)) { - name = fmt::format("InfillTestParameters_P{:d}_Z{:d}_C{:d}_L{:d}__{:d}", static_cast(params.pattern), params.zig_zagify, params.connect_polygons, params.line_distance, test_polygon_id); + name = fmt::format( + "InfillTestParameters_P{:d}_Z{:d}_C{:d}_L{:d}__{:d}", + static_cast(params.pattern), + params.zig_zagify, + params.connect_polygons, + params.line_distance, + test_polygon_id); } friend std::ostream& operator<<(std::ostream& os, const InfillTestParameters& params) @@ -108,11 +123,12 @@ constexpr coord_t Z = 100; // Future improvement: Also take an uneven layer, so constexpr coord_t SHIFT = 0; constexpr coord_t MAX_RESOLUTION = 10; constexpr coord_t MAX_DEVIATION = 5; -const std::vector POLYGON_FILENAMES = { - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave.txt").string(), std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave_hole.txt").string(), - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square.txt").string(), std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square_hole.txt").string(), - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_triangle.txt").string(), std::filesystem::path(__FILE__).parent_path().append("resources/polygon_two_squares.txt").string() -}; +const std::vector POLYGON_FILENAMES = { std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave_hole.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square_hole.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_triangle.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_two_squares.txt").string() }; #ifdef TEST_INFILL_SVG_OUTPUT void writeTestcaseSVG(const InfillTestParameters& params) @@ -134,7 +150,7 @@ void writeTestcaseSVG(const InfillTestParameters& params) } #endif // TEST_INFILL_SVG_OUTPUT -InfillTestParameters generateInfillToTest(const InfillParameters& params, const size_t& test_polygon_id, const Polygons& outline_polygons) +InfillTestParameters generateInfillToTest(const InfillParameters& params, const size_t& test_polygon_id, const Shape& outline_polygons) { auto layers = std::vector(200, SlicerLayer{}); scripta::setAll(layers); @@ -144,24 +160,25 @@ InfillTestParameters generateInfillToTest(const InfillParameters& params, const const bool connect_polygons = params.connect_polygons; const coord_t line_distance = params.line_distance; - Infill infill(pattern, - zig_zagify, - connect_polygons, - outline_polygons, - INFILL_LINE_WIDTH, - line_distance, - INFILL_OVERLAP, - INFILL_MULTIPLIER, - FILL_ANGLE, - Z, - SHIFT, - MAX_RESOLUTION, - MAX_DEVIATION); // There are some optional parameters, but these will do for now (future improvement?). + Infill infill( + pattern, + zig_zagify, + connect_polygons, + outline_polygons, + INFILL_LINE_WIDTH, + line_distance, + INFILL_OVERLAP, + INFILL_MULTIPLIER, + FILL_ANGLE, + Z, + SHIFT, + MAX_RESOLUTION, + MAX_DEVIATION); // There are some optional parameters, but these will do for now (future improvement?). Settings infill_settings; std::vector result_paths; - Polygons result_polygons; - Polygons result_lines; + Shape result_polygons; + OpenLinesSet result_lines; infill.generate(result_paths, result_polygons, result_lines, infill_settings, 1, SectionType::INFILL, nullptr, nullptr); InfillTestParameters result = InfillTestParameters(params, test_polygon_id, outline_polygons, result_lines, result_polygons); @@ -174,7 +191,7 @@ std::vector generateInfillTests() constexpr bool do_connect_polygons = true; constexpr bool dont_connect_polygons = false; - std::vector shapes; + std::vector shapes; if (! readTestPolygons(POLYGON_FILENAMES, shapes)) { return { InfillTestParameters() }; // return an invalid singleton, that'll trip up the 'file read' assertion in the TEST_P's @@ -187,7 +204,8 @@ std::vector generateInfillTests() * - Gyroid, since it doesn't handle the 100% infill and related cases well * - Concentric and ZigZag, since they now use a method that starts from an extra infill wall, which fail these tests (TODO!) */ - std::vector skip_methods = { EFillMethod::CONCENTRIC, EFillMethod::ZIG_ZAG, EFillMethod::CROSS, EFillMethod::CROSS_3D, EFillMethod::CUBICSUBDIV, EFillMethod::GYROID, EFillMethod::LIGHTNING }; + std::vector skip_methods + = { EFillMethod::CONCENTRIC, EFillMethod::ZIG_ZAG, EFillMethod::CROSS, EFillMethod::CROSS_3D, EFillMethod::CUBICSUBDIV, EFillMethod::GYROID, EFillMethod::LIGHTNING }; std::vector methods; for (int i_method = 0; i_method < static_cast(EFillMethod::NONE); ++i_method) @@ -203,7 +221,7 @@ std::vector generateInfillTests() std::vector parameters_list; size_t test_polygon_id = 0; - for (const Polygons& polygons : shapes) + for (const Shape& polygons : shapes) { for (const EFillMethod& method : methods) { @@ -227,7 +245,14 @@ class InfillTest : public testing::TestWithParam { }; -INSTANTIATE_TEST_SUITE_P(InfillTestcases, InfillTest, testing::ValuesIn(generateInfillTests()), [](const testing::TestParamInfo& info) { return info.param.name; }); +INSTANTIATE_TEST_SUITE_P( + InfillTestcases, + InfillTest, + testing::ValuesIn(generateInfillTests()), + [](const testing::TestParamInfo& info) + { + return info.param.name; + }); TEST_P(InfillTest, TestInfillSanity) { @@ -243,7 +268,7 @@ TEST_P(InfillTest, TestInfillSanity) long double worst_case_zig_zag_added_area = 0; if (params.params.zig_zagify || params.params.pattern == EFillMethod::ZIG_ZAG) { - worst_case_zig_zag_added_area = params.outline_polygons.polygonLength() * INFILL_LINE_WIDTH; + worst_case_zig_zag_added_area = params.outline_polygons.length() * INFILL_LINE_WIDTH; } const double min_available_area = std::abs(params.outline_polygons.offset(static_cast(-params.params.line_distance) / 2).area()); @@ -251,23 +276,26 @@ TEST_P(InfillTest, TestInfillSanity) const long double min_expected_infill_area = (min_available_area * static_cast(INFILL_LINE_WIDTH)) / params.params.line_distance; const long double max_expected_infill_area = (max_available_area * INFILL_LINE_WIDTH) / params.params.line_distance + worst_case_zig_zag_added_area; - const long double out_infill_area = ((params.result_polygons.polygonLength() + params.result_lines.polyLineLength()) * static_cast(INFILL_LINE_WIDTH)) / getPatternMultiplier(params.params.pattern); + const long double out_infill_area + = ((params.result_polygons.length() + params.result_lines.length()) * static_cast(INFILL_LINE_WIDTH)) / getPatternMultiplier(params.params.pattern); ASSERT_GT((coord_t)max_available_area, (coord_t)out_infill_area) << "Infill area should allways be less than the total area available."; ASSERT_GT((coord_t)out_infill_area, (coord_t)min_expected_infill_area) << "Infill area should be greater than the minimum area expected to be covered."; ASSERT_LT((coord_t)out_infill_area, (coord_t)max_expected_infill_area) << "Infill area should be less than the maximum area to be covered."; const coord_t maximum_error = 10_mu; // potential rounding error - const Polygons padded_shape_outline = params.outline_polygons.offset(INFILL_LINE_WIDTH / 2); + const Shape padded_shape_outline = params.outline_polygons.offset(INFILL_LINE_WIDTH / 2); constexpr bool restitch = false; // No need to restitch polylines - that would introduce stitching errors. - ASSERT_LE(std::abs(padded_shape_outline.intersectionPolyLines(params.result_lines, restitch).polyLineLength() - params.result_lines.polyLineLength()), maximum_error) << "Infill (lines) should not be outside target polygon."; - Polygons result_polygon_lines = params.result_polygons; - for (PolygonRef poly : result_polygon_lines) + ASSERT_LE(std::abs(padded_shape_outline.intersection(params.result_lines, restitch).length() - params.result_lines.length()), maximum_error) + << "Infill (lines) should not be outside target polygon."; + Shape result_polygon_lines = params.result_polygons; + for (Polygon& poly : result_polygon_lines) { - poly.add(poly.front()); + poly.push_back(poly.front()); } - ASSERT_LE(std::abs(padded_shape_outline.intersectionPolyLines(result_polygon_lines, restitch).polyLineLength() - result_polygon_lines.polyLineLength()), maximum_error) << "Infill (lines) should not be outside target polygon."; + ASSERT_LE(std::abs(padded_shape_outline.intersection(result_polygon_lines, restitch).length() - result_polygon_lines.length()), maximum_error) + << "Infill (lines) should not be outside target polygon."; } } // namespace cura -// NOLINTEND(*-magic-numbers) \ No newline at end of file +// NOLINTEND(*-magic-numbers) diff --git a/tests/LayerPlanTest.cpp b/tests/LayerPlanTest.cpp index 421a2caf51..7c7afa13e7 100644 --- a/tests/LayerPlanTest.cpp +++ b/tests/LayerPlanTest.cpp @@ -320,30 +320,30 @@ class AddTravelTest : public LayerPlanTest, public testing::WithParamInterface("false", "false", "off", false, false, AddTravelTestScene::OPEN)) { - around_start_end.add(Point2LL(-100, -100)); - around_start_end.add(Point2LL(500100, -100)); - around_start_end.add(Point2LL(500100, 500100)); - around_start_end.add(Point2LL(-100, 500100)); - - around_start.add(Point2LL(-100, -100)); - around_start.add(Point2LL(100, -100)); - around_start.add(Point2LL(100, 100)); - around_start.add(Point2LL(-100, 100)); - - around_end.add(Point2LL(249900, 249900)); - around_end.add(Point2LL(250100, 249900)); - around_end.add(Point2LL(250100, 250100)); - around_end.add(Point2LL(249900, 249900)); - - between.add(Point2LL(250000, 240000)); - between.add(Point2LL(260000, 240000)); - between.add(Point2LL(260000, 300000)); - between.add(Point2LL(250000, 300000)); - - between_hole.add(Point2LL(250000, 240000)); - between_hole.add(Point2LL(250000, 300000)); - between_hole.add(Point2LL(260000, 300000)); - between_hole.add(Point2LL(260000, 240000)); + around_start_end.push_back(Point2LL(-100, -100)); + around_start_end.push_back(Point2LL(500100, -100)); + around_start_end.push_back(Point2LL(500100, 500100)); + around_start_end.push_back(Point2LL(-100, 500100)); + + around_start.push_back(Point2LL(-100, -100)); + around_start.push_back(Point2LL(100, -100)); + around_start.push_back(Point2LL(100, 100)); + around_start.push_back(Point2LL(-100, 100)); + + around_end.push_back(Point2LL(249900, 249900)); + around_end.push_back(Point2LL(250100, 249900)); + around_end.push_back(Point2LL(250100, 250100)); + around_end.push_back(Point2LL(249900, 249900)); + + between.push_back(Point2LL(250000, 240000)); + between.push_back(Point2LL(260000, 240000)); + between.push_back(Point2LL(260000, 300000)); + between.push_back(Point2LL(250000, 300000)); + + between_hole.push_back(Point2LL(250000, 240000)); + between_hole.push_back(Point2LL(250000, 300000)); + between_hole.push_back(Point2LL(260000, 300000)); + between_hole.push_back(Point2LL(260000, 240000)); } /*! @@ -363,7 +363,7 @@ class AddTravelTest : public LayerPlanTest, public testing::WithParamInterfaceget("retraction_min_travel"); // Update the copy that the storage has of this. settings->add("retraction_combing_max_distance", parameters.is_long_combing ? "1" : "10000"); - Polygons slice_data; + Shape slice_data; switch (parameters.scene) { case OPEN: @@ -371,24 +371,24 @@ class AddTravelTest : public LayerPlanTest, public testing::WithParamInterface + #include +#include #include #include #include +#include + +#include "ReadTestPolygons.h" +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" +#include "infill.h" +#include "slicer.h" +#include "utils/Coord_t.h" +#include "utils/math.h" + // To diagnose failing tests with visual images, uncomment the following line: // #define TEST_PATHS_SVG_OUTPUT #ifdef TEST_PATHS_SVG_OUTPUT -#include "utils/SVG.h" #include + +#include "utils/SVG.h" #endif // TEST_PATHS_SVG_OUTPUT namespace cura @@ -30,22 +36,22 @@ class PathOrderMonotonicTest : public testing::TestWithParam& path) +inline Point2LL startVertex(const PathOrdering& path) { return (*path.vertices_)[path.start_vertex_]; } -inline Point2LL endVertex(const PathOrdering& path) +inline Point2LL endVertex(const PathOrdering& path) { return (*path.vertices_)[path.vertices_->size() - (1 + path.start_vertex_)]; } -coord_t projectPathAlongAxis(const PathOrdering& path, const Point2LL& vector) +coord_t projectPathAlongAxis(const PathOrdering& path, const Point2LL& vector) { return dot(startVertex(path), vector); } -coord_t projectEndAlongAxis(const PathOrdering& path, const Point2LL& vector) +coord_t projectEndAlongAxis(const PathOrdering& path, const Point2LL& vector) { return dot(endVertex(path), vector); } @@ -54,7 +60,8 @@ bool rangeOverlaps(const std::pair& range_b, const std::pair shapes; + std::vector shapes; if (! readTestPolygons(filename, shapes)) { return false; @@ -81,17 +88,34 @@ bool getInfillLines(const std::string& filename, const AngleRadians& angle, Poly for (const auto& shape : shapes) { - Infill infill_comp(pattern, zig_zagify, connect_polygons, shape, infill_line_width, line_distance, infill_overlap, infill_multiplier, AngleDegrees(angle), z, shift, max_resolution, max_deviation); + Infill infill_comp( + pattern, + zig_zagify, + connect_polygons, + shape, + infill_line_width, + line_distance, + infill_overlap, + infill_multiplier, + AngleDegrees(angle), + z, + shift, + max_resolution, + max_deviation); Settings infill_settings; std::vector result_paths; - Polygons dummy_polys; + Shape dummy_polys; infill_comp.generate(result_paths, dummy_polys, output, infill_settings, 1, SectionType::INFILL, nullptr, nullptr); } return true; } #ifdef TEST_PATHS_SVG_OUTPUT -void writeDebugSVG(const std::string& original_filename, const AngleRadians& angle, const Point& monotonic_vec, const std::vector>>& sections) +void writeDebugSVG( + const std::string& original_filename, + const AngleRadians& angle, + const Point& monotonic_vec, + const std::vector>>& sections) { constexpr int buff_size = 1024; char buff[buff_size]; @@ -136,7 +160,7 @@ TEST_P(PathOrderMonotonicTest, SectionsTest) const auto params = GetParam(); const double angle_radians{ std::get<1>(params) }; const auto& filename = std::get<0>(params); - Polygons polylines; + OpenLinesSet polylines; ASSERT_TRUE(getInfillLines(filename, angle_radians, polylines)) << "Input test-file could not be read, check setup."; const Point2LL& pt_r = polylines.begin()->at(0); @@ -146,15 +170,15 @@ TEST_P(PathOrderMonotonicTest, SectionsTest) const Point2LL perpendicular_axis{ turn90CCW(monotonic_axis) }; constexpr coord_t max_adjacent_distance = line_distance + 1; - PathOrderMonotonic object_under_test(angle_from_first_line, max_adjacent_distance, monotonic_axis * -1000); + PathOrderMonotonic object_under_test(angle_from_first_line, max_adjacent_distance, monotonic_axis * -1000); for (const auto& polyline : polylines) { - object_under_test.addPolyline(ConstPolygonPointer(polyline)); + object_under_test.addPolyline(&polyline); } object_under_test.optimize(); // Collect sections: - std::vector>> sections; + std::vector>> sections; sections.emplace_back(); coord_t last_path_mono_projection = projectPathAlongAxis(object_under_test.paths_.front(), monotonic_axis); for (const auto& path : object_under_test.paths_) @@ -202,7 +226,8 @@ TEST_P(PathOrderMonotonicTest, SectionsTest) const coord_t mono_b = projectPathAlongAxis(*it_b, monotonic_axis); if (mono_a < mono_b) { - EXPECT_FALSE(rangeOverlaps(perp_b_range, perp_a_range)) << "Perpendicular range overlaps for neighboring lines in different sections (next line of A / line in B)."; + EXPECT_FALSE(rangeOverlaps(perp_b_range, perp_a_range)) + << "Perpendicular range overlaps for neighboring lines in different sections (next line of A / line in B)."; } } } @@ -210,14 +235,28 @@ TEST_P(PathOrderMonotonicTest, SectionsTest) } } -const std::vector polygon_filenames = { - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave.txt").string(), std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave_hole.txt").string(), - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square.txt").string(), std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square_hole.txt").string(), - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_triangle.txt").string(), std::filesystem::path(__FILE__).parent_path().append("resources/polygon_two_squares.txt").string(), - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_slant_gap.txt").string(), std::filesystem::path(__FILE__).parent_path().append("resources/polygon_sawtooth.txt").string(), - std::filesystem::path(__FILE__).parent_path().append("resources/polygon_letter_y.txt").string() -}; -const std::vector angle_radians = { 0, 0.1, 0.25 * std::numbers::pi, 1.0, 0.5 * std::numbers::pi, 0.75 * std::numbers::pi, std::numbers::pi, 1.25 * std::numbers::pi, 4.0, 1.5 * std::numbers::pi, 1.75 * std::numbers::pi, 5.0, (2.0 * std::numbers::pi) - 0.1 }; +const std::vector polygon_filenames = { std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_concave_hole.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_square_hole.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_triangle.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_two_squares.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_slant_gap.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_sawtooth.txt").string(), + std::filesystem::path(__FILE__).parent_path().append("resources/polygon_letter_y.txt").string() }; +const std::vector angle_radians = { 0, + 0.1, + 0.25 * std::numbers::pi, + 1.0, + 0.5 * std::numbers::pi, + 0.75 * std::numbers::pi, + std::numbers::pi, + 1.25 * std::numbers::pi, + 4.0, + 1.5 * std::numbers::pi, + 1.75 * std::numbers::pi, + 5.0, + (2.0 * std::numbers::pi) - 0.1 }; INSTANTIATE_TEST_SUITE_P(PathOrderMonotonicTestInstantiation, PathOrderMonotonicTest, testing::Combine(testing::ValuesIn(polygon_filenames), testing::ValuesIn(angle_radians))); // NOLINTEND(*-magic-numbers) diff --git a/tests/PathOrderOptimizerTest.cpp b/tests/PathOrderOptimizerTest.cpp index b4d4a32ee1..c8621ba3d8 100644 --- a/tests/PathOrderOptimizerTest.cpp +++ b/tests/PathOrderOptimizerTest.cpp @@ -15,7 +15,7 @@ class PathOrderOptimizerTest : public testing::Test /*! * A blank optimizer with no polygons added yet. Fresh and virgin. */ - PathOrderOptimizer optimizer; + PathOrderOptimizer optimizer; /*! * A simple isosceles triangle. Base length and height 50. @@ -29,12 +29,12 @@ class PathOrderOptimizerTest : public testing::Test void SetUp() override { - optimizer = PathOrderOptimizer(Point2LL(0, 0)); + optimizer = PathOrderOptimizer(Point2LL(0, 0)); triangle.clear(); - triangle.add(Point2LL(0, 0)); - triangle.add(Point2LL(50, 0)); - triangle.add(Point2LL(25, 50)); + triangle.push_back(Point2LL(0, 0)); + triangle.push_back(Point2LL(50, 0)); + triangle.push_back(Point2LL(25, 50)); } }; // NOLINTEND(misc-non-private-member-variables-in-classes) @@ -62,9 +62,9 @@ TEST_F(PathOrderOptimizerTest, ThreeTrianglesShortestOrder) far.translate(Point2LL(1000, 1000)); // Add them out of order so that it's clear that the optimization changes the order. - optimizer.addPolygon(middle); - optimizer.addPolygon(far); - optimizer.addPolygon(near); + optimizer.addPolygon(&middle); + optimizer.addPolygon(&far); + optimizer.addPolygon(&near); optimizer.optimize(); diff --git a/tests/ReadTestPolygons.cpp b/tests/ReadTestPolygons.cpp index baa3d71147..be4b5bbe87 100644 --- a/tests/ReadTestPolygons.cpp +++ b/tests/ReadTestPolygons.cpp @@ -2,16 +2,19 @@ // CuraEngine is released under the terms of the AGPLv3 or higher. #include "ReadTestPolygons.h" -#include "utils/Coord_t.h" + #include +#include "geometry/Shape.h" +#include "utils/Coord_t.h" + // NOTE: See the documentation in the header-file for an explanation of this simple file format. namespace cura { // Read multiple files to the collection of polygons. // Returns boolean success/failure (read errors, not found, etc.). -bool readTestPolygons(const std::vector& filenames, std::vector& polygons_out) +bool readTestPolygons(const std::vector& filenames, std::vector& polygons_out) { for (const std::string& filename : filenames) { @@ -25,7 +28,7 @@ bool readTestPolygons(const std::vector& filenames, std::vector& polygons_out) +bool readTestPolygons(const std::string& filename, std::vector& polygons_out) { FILE* handle = std::fopen(filename.c_str(), "r"); if (! handle) @@ -34,7 +37,7 @@ bool readTestPolygons(const std::string& filename, std::vector& polygo } Polygon next_path; - Polygons next_shape; + Shape next_shape; char command = '_'; int read = 0; @@ -68,7 +71,7 @@ bool readTestPolygons(const std::string& filename, std::vector& polygo case '#': // end of file if (! next_path.empty()) { - next_shape.add(Polygon(next_path)); // copy and add + next_shape.push_back(Polygon(next_path)); // copy and add next_path.clear(); } if (command != 'x' && ! next_shape.empty()) diff --git a/tests/ReadTestPolygons.h b/tests/ReadTestPolygons.h index cf7f7b2884..5c237025b1 100644 --- a/tests/ReadTestPolygons.h +++ b/tests/ReadTestPolygons.h @@ -4,7 +4,7 @@ #ifndef READ_TEST_POLYGONS_H #define READ_TEST_POLYGONS_H -#include "utils/polygon.h" +#include "geometry/Polygon.h" #include #include @@ -34,8 +34,8 @@ v 50000 50000 namespace cura { -bool readTestPolygons(const std::vector& filenames, std::vector& polygons_out); -bool readTestPolygons(const std::string& filename, std::vector& polygons_out); +bool readTestPolygons(const std::vector& filenames, std::vector& polygons_out); +bool readTestPolygons(const std::string& filename, std::vector& polygons_out); } // namespace cura #endif // READ_TEST_POLYGONS_H diff --git a/tests/WallsComputationTest.cpp b/tests/WallsComputationTest.cpp index a45cb5464a..98061d4cdc 100644 --- a/tests/WallsComputationTest.cpp +++ b/tests/WallsComputationTest.cpp @@ -2,21 +2,25 @@ // CuraEngine is released under the terms of the AGPLv3 or higher #include "WallsComputation.h" //Unit under test. + +#include + +#include +#include + +#include + #include "InsetOrderOptimizer.h" //Unit also under test. +#include "geometry/OpenPolyline.h" +#include "geometry/Polygon.h" //To create example polygons. #include "settings/Settings.h" //Settings to generate walls with. #include "sliceDataStorage.h" //Sl #include "slicer.h" -#include "utils/polygon.h" //To create example polygons. -#include -#include -#include - -#include #ifdef WALLS_COMPUTATION_TEST_SVG_OUTPUT -#include "utils/SVG.h" -#include "utils/polygon.h" #include + +#include "utils/SVG.h" #endif // WALLS_COMPUTATION_TEST_SVG_OUTPUT // NOLINTBEGIN(*-magic-numbers) @@ -41,14 +45,15 @@ class WallsComputationTest : public testing::Test /*! * Basic 10x10mm square shape to work with. */ - Polygons square_shape; + Shape square_shape; /*! * A rectangle enclosing two triangular holes; */ - Polygons ff_holes; + Shape ff_holes; - WallsComputationTest() : walls_computation(settings, LayerIndex(100)) + WallsComputationTest() + : walls_computation(settings, LayerIndex(100)) { square_shape.emplace_back(); square_shape.back().emplace_back(0, 0); @@ -104,7 +109,7 @@ TEST_F(WallsComputationTest, GenerateWallsForLayerSinglePart) SliceLayer layer; layer.parts.emplace_back(); SliceLayerPart& part = layer.parts.back(); - part.outline.add(square_shape); + part.outline.push_back(square_shape); // Run the test. walls_computation.generateWalls(&layer, SectionType::WALL); @@ -113,7 +118,8 @@ TEST_F(WallsComputationTest, GenerateWallsForLayerSinglePart) EXPECT_FALSE(part.wall_toolpaths.empty()) << "There must be some walls."; EXPECT_GT(part.print_outline.area(), 0) << "The print outline must encompass the outer wall, so it must be more than 0."; EXPECT_LE(part.print_outline.area(), square_shape.area()) << "The print outline must stay within the bounds of the original part."; - EXPECT_GT(part.inner_area.area(), 0) << "The inner area must be within the innermost wall. There are not enough walls to fill the entire part, so there is a positive inner area."; + EXPECT_GT(part.inner_area.area(), 0) + << "The inner area must be within the innermost wall. There are not enough walls to fill the entire part, so there is a positive inner area."; EXPECT_EQ(layer.parts.size(), 1) << "There is still just 1 part."; } @@ -126,7 +132,7 @@ TEST_F(WallsComputationTest, GenerateWallsZeroWalls) SliceLayer layer; layer.parts.emplace_back(); SliceLayerPart& part = layer.parts.back(); - part.outline.add(square_shape); + part.outline.push_back(square_shape); // Run the test. walls_computation.generateWalls(&layer, SectionType::WALL); @@ -147,7 +153,7 @@ TEST_F(WallsComputationTest, WallToolPathsGetWeakOrder) SliceLayer layer; layer.parts.emplace_back(); SliceLayerPart& part = layer.parts.back(); - part.outline.add(ff_holes); + part.outline.push_back(ff_holes); // Run the test. walls_computation.generateWalls(&layer, SectionType::WALL); diff --git a/tests/arcus/ArcusCommunicationTest.cpp b/tests/arcus/ArcusCommunicationTest.cpp index 7c1036166e..a05b88d98d 100644 --- a/tests/arcus/ArcusCommunicationTest.cpp +++ b/tests/arcus/ArcusCommunicationTest.cpp @@ -1,15 +1,19 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. +#include +#include +#include + +#include + #include "FffProcessor.h" #include "MockSocket.h" //To mock out the communication with the front-end. #include "communication/ArcusCommunicationPrivate.h" //To access the private fields of this communication class. +#include "geometry/Polygon.h" //Create test shapes to send over the socket. +#include "geometry/Shape.h" #include "settings/types/LayerIndex.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" //Create test shapes to send over the socket. -#include -#include -#include // NOLINTBEGIN(*-magic-numbers) namespace cura @@ -33,7 +37,7 @@ class ArcusCommunicationTest : public testing::Test Polygon test_circle; Polygon test_convex_shape; - Polygons test_shapes; // all above polygons + Shape test_shapes; // all above polygons void SetUp() override { @@ -47,24 +51,24 @@ class ArcusCommunicationTest : public testing::Test test_square.emplace_back(1000, 0); test_square.emplace_back(1000, 1000); test_square.emplace_back(0, 1000); - test_shapes.add(test_square); + test_shapes.push_back(test_square); test_square2.emplace_back(1100, 1500); test_square2.emplace_back(2000, 1500); test_square2.emplace_back(2000, -500); test_square2.emplace_back(1100, -500); - test_shapes.add(test_square2); + test_shapes.push_back(test_square2); test_triangle.emplace_back(0, 2100); test_triangle.emplace_back(500, 1100); test_triangle.emplace_back(1500, 2100); - test_shapes.add(test_triangle); + test_shapes.push_back(test_triangle); for (double a = 0; a < 1.0; a += .05) { - test_circle.add(Point2LL(2050, 2050) + Point2LL(std::cos(a * 2 * std::numbers::pi) * 500, std::sin(a * 2 * std::numbers::pi) * 500)); + test_circle.push_back(Point2LL(2050, 2050) + Point2LL(std::cos(a * 2 * std::numbers::pi) * 500, std::sin(a * 2 * std::numbers::pi) * 500)); } - test_shapes.add(test_circle); + test_shapes.push_back(test_circle); test_convex_shape.emplace_back(-300, 0); test_convex_shape.emplace_back(-100, 500); @@ -75,7 +79,7 @@ class ArcusCommunicationTest : public testing::Test test_convex_shape.emplace_back(-1500, 1500); test_convex_shape.emplace_back(-1600, 1100); test_convex_shape.emplace_back(-700, 200); - test_shapes.add(test_convex_shape); + test_shapes.push_back(test_convex_shape); } void TearDown() override @@ -170,4 +174,4 @@ TEST_F(ArcusCommunicationTest, SendProgress) } } // namespace cura -// NOLINTEND(*-magic-numbers) \ No newline at end of file +// NOLINTEND(*-magic-numbers) diff --git a/tests/arcus/MockCommunication.h b/tests/arcus/MockCommunication.h index f2401025fd..0f95cee769 100644 --- a/tests/arcus/MockCommunication.h +++ b/tests/arcus/MockCommunication.h @@ -7,9 +7,10 @@ #include #include "communication/Communication.h" //The interface we're implementing. +#include "geometry/Polygon.h" //In the signature of Communication. +#include "geometry/Shape.h" #include "settings/types/LayerIndex.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" //In the signature of Communication. namespace cura { @@ -24,10 +25,8 @@ class MockCommunication : public Communication MOCK_CONST_METHOD0(isSequential, bool()); MOCK_CONST_METHOD1(sendProgress, void(double progress)); MOCK_METHOD3(sendLayerComplete, void(const LayerIndex::value_type& layer_nr, const coord_t& z, const coord_t& thickness)); - MOCK_METHOD5(sendPolygons, void(const PrintFeatureType& type, const Polygons& polygons, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity)); - MOCK_METHOD5( - sendPolygon, - void(const PrintFeatureType& type, const ConstPolygonRef& polygon, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity)); + MOCK_METHOD5(sendPolygons, void(const PrintFeatureType& type, const Shape& polygons, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity)); + MOCK_METHOD5(sendPolygon, void(const PrintFeatureType& type, const Polygon& polygon, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity)); MOCK_METHOD5(sendLineTo, void(const PrintFeatureType& type, const Point2LL& to, const coord_t& line_width, const coord_t& line_thickness, const Velocity& velocity)); MOCK_METHOD1(sendCurrentPosition, void(const Point2LL& position)); MOCK_METHOD1(setExtruderForSend, void(const ExtruderTrain& extruder)); diff --git a/tests/integration/SlicePhaseTest.cpp b/tests/integration/SlicePhaseTest.cpp index de814b5099..cfb9557024 100644 --- a/tests/integration/SlicePhaseTest.cpp +++ b/tests/integration/SlicePhaseTest.cpp @@ -1,17 +1,18 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include +#include #include #include "Application.h" // To set up a slice with settings. #include "Slice.h" // To set up a scene to slice. +#include "geometry/Polygon.h" // Creating polygons to compare to sliced layers. #include "slicer.h" // Starts the slicing phase that we want to test. #include "utils/Coord_t.h" #include "utils/Matrix4x3D.h" // To load STL files. -#include "utils/polygon.h" // Creating polygons to compare to sliced layers. -#include "utils/polygonUtils.h" // Comparing similarity of polygons. +#include "utils/polygonUtils.h" // Comparing similarity of polygons_. namespace cura { @@ -95,10 +96,10 @@ TEST_F(SlicePhaseTest, Cube) for (size_t layer_nr = 0; layer_nr < num_layers; layer_nr++) { const SlicerLayer& layer = slicer.layers[layer_nr]; - EXPECT_EQ(layer.polygons.size(), 1); - if (layer.polygons.size() == 1) + EXPECT_EQ(layer.polygons_.size(), 1); + if (layer.polygons_.size() == 1) { - Polygon sliced_polygon = layer.polygons[0]; + Polygon sliced_polygon = layer.polygons_[0]; EXPECT_EQ(sliced_polygon.size(), square.size()); if (sliced_polygon.size() == square.size()) { @@ -157,19 +158,19 @@ TEST_F(SlicePhaseTest, Cylinder1000) const coord_t y = std::sin(std::numbers::pi * 2 / num_vertices * i) * radius; circle.emplace_back(x, y); } - Polygons circles; - circles.add(circle); + Shape circles; + circles.push_back(circle); for (size_t layer_nr = 0; layer_nr < num_layers; layer_nr++) { const SlicerLayer& layer = slicer.layers[layer_nr]; - EXPECT_EQ(layer.polygons.size(), 1); - if (layer.polygons.size() == 1) + EXPECT_EQ(layer.polygons_.size(), 1); + if (layer.polygons_.size() == 1) { - Polygon sliced_polygon = layer.polygons[0]; + Polygon sliced_polygon = layer.polygons_[0]; // Due to the reduction in resolution, the final slice will not have the same vertices as the input. // Let's say that are allowed to be up to 1/500th of the surface area off. - EXPECT_LE(PolygonUtils::relativeHammingDistance(layer.polygons, circles), 0.002); + EXPECT_LE(PolygonUtils::relativeHammingDistance(layer.polygons_, circles), 0.002); } } } diff --git a/tests/settings/SettingsTest.cpp b/tests/settings/SettingsTest.cpp index c8f7814db8..0c6c98f458 100644 --- a/tests/settings/SettingsTest.cpp +++ b/tests/settings/SettingsTest.cpp @@ -1,10 +1,11 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher #include "settings/Settings.h" //The class under test. -#include //For std::numbers::pi. +#include #include //For shared_ptr. +#include #include diff --git a/tests/utils/AABB3DTest.cpp b/tests/utils/AABB3DTest.cpp index 711e20849c..31c12017dc 100644 --- a/tests/utils/AABB3DTest.cpp +++ b/tests/utils/AABB3DTest.cpp @@ -4,7 +4,7 @@ #include "utils/AABB3D.h" #include "utils/AABB.h" #include "utils/Coord_t.h" -#include "utils/polygon.h" +#include "geometry/Polygon.h" #include #include diff --git a/tests/utils/AABBTest.cpp b/tests/utils/AABBTest.cpp index bb39a91b4f..cc2968d3f1 100644 --- a/tests/utils/AABBTest.cpp +++ b/tests/utils/AABBTest.cpp @@ -2,10 +2,14 @@ // CuraEngine is released under the terms of the AGPLv3 or higher. #include "utils/AABB.h" -#include "utils/polygon.h" -#include + #include +#include + +#include "geometry/Polygon.h" +#include "geometry/Shape.h" + namespace cura { // NOLINTBEGIN(*-magic-numbers) @@ -34,15 +38,18 @@ TEST(AABBTest, TestConstructPoint) TEST(AABBTest, TestConstructPolygons) { - Polygons empty_polygon; + Shape empty_polygon; AABB polygons_box_a(empty_polygon); EXPECT_FALSE(polygons_box_a.contains(Point2LL(0, 0))) << "Box constructed from empty polygon shouldn't contain anything."; - Polygons polygons; - polygons.add(Polygon(ClipperLib::Path({ ClipperLib::IntPoint{ -10, -10 }, ClipperLib::IntPoint{ 10, -10 }, ClipperLib::IntPoint{ -5, -5 }, ClipperLib::IntPoint{ -10, 10 } }))); - polygons.add(Polygon(ClipperLib::Path({ ClipperLib::IntPoint{ 11, 11 }, ClipperLib::IntPoint{ -11, 11 }, ClipperLib::IntPoint{ 4, 4 }, ClipperLib::IntPoint{ 11, -11 } }))); - polygons.add(Polygon(ClipperLib::Path({ ClipperLib::IntPoint{ 2, 2 }, ClipperLib::IntPoint{ 2, 3 }, ClipperLib::IntPoint{ 3, 3 }, ClipperLib::IntPoint{ 3, 2 } }))); + Shape polygons; + polygons.push_back( + Polygon(ClipperLib::Path({ ClipperLib::IntPoint{ -10, -10 }, ClipperLib::IntPoint{ 10, -10 }, ClipperLib::IntPoint{ -5, -5 }, ClipperLib::IntPoint{ -10, 10 } }), false)); + polygons.push_back( + Polygon(ClipperLib::Path({ ClipperLib::IntPoint{ 11, 11 }, ClipperLib::IntPoint{ -11, 11 }, ClipperLib::IntPoint{ 4, 4 }, ClipperLib::IntPoint{ 11, -11 } }), false)); + polygons.push_back( + Polygon(ClipperLib::Path({ ClipperLib::IntPoint{ 2, 2 }, ClipperLib::IntPoint{ 2, 3 }, ClipperLib::IntPoint{ 3, 3 }, ClipperLib::IntPoint{ 3, 2 } }), false)); AABB polygons_box_b(polygons); @@ -161,4 +168,4 @@ TEST(AABBTest, TestToPolygon) } // NOLINTEND(*-magic-numbers) -} // namespace cura \ No newline at end of file +} // namespace cura diff --git a/tests/utils/IntPointTest.cpp b/tests/utils/IntPointTest.cpp index 1332e1b8db..742991be3f 100644 --- a/tests/utils/IntPointTest.cpp +++ b/tests/utils/IntPointTest.cpp @@ -1,9 +1,12 @@ // Copyright (c) 2022 Ultimaker B.V. // CuraEngine is released under the terms of the AGPLv3 or higher. -#include "utils/Point2LL.h" #include +#include "geometry/Point2LL.h" +#include "geometry/Point3Matrix.h" +#include "geometry/PointMatrix.h" + // NOLINTBEGIN(*-magic-numbers) namespace cura { diff --git a/tests/utils/LinearAlg2DTest.cpp b/tests/utils/LinearAlg2DTest.cpp index b9e4f41bb0..afa754b9fb 100644 --- a/tests/utils/LinearAlg2DTest.cpp +++ b/tests/utils/LinearAlg2DTest.cpp @@ -1,12 +1,15 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. #include "utils/linearAlg2D.h" #include +#include #include +#include "geometry/Point3Matrix.h" + // NOLINTBEGIN(*-magic-numbers) namespace cura { diff --git a/tests/utils/PolygonConnectorTest.cpp b/tests/utils/PolygonConnectorTest.cpp index b423d74d31..b2b08af3c0 100644 --- a/tests/utils/PolygonConnectorTest.cpp +++ b/tests/utils/PolygonConnectorTest.cpp @@ -2,11 +2,14 @@ // CuraEngine is released under the terms of the AGPLv3 or higher. #include "utils/PolygonConnector.h" // The class under test. -#include "utils/Coord_t.h" -#include "utils/polygon.h" // To create polygons to test with. -#include + #include +#include + +#include "geometry/Polygon.h" // To create polygons to test with. +#include "utils/Coord_t.h" + // NOLINTBEGIN(*-magic-numbers) namespace cura { @@ -21,10 +24,10 @@ class PolygonConnectorTest : public testing::Test Polygon test_triangle; Polygon test_circle; Polygon test_convex_shape; - Polygons test_shapes; // All above polygons! As well as an inset of 100 microns of them. + Shape test_shapes; // All above polygons! As well as an inset of 100 microns of them. PolygonConnector* pc; - Polygons connected_polygons; + Shape connected_polygons; std::vector connected_paths; virtual void SetUp() override @@ -70,7 +73,8 @@ TEST_F(PolygonConnectorTest, getBridgeNestedSquares) EXPECT_EQ(vSize(bridge->a_.from_point_ - bridge->a_.to_point_), 100) << "The polygons are 100 units spaced out concentrically, so this is the shortest possible bridge."; EXPECT_EQ(vSize(bridge->b_.from_point_ - bridge->b_.to_point_), 100) << "The second bridge should also be equally short in this case."; - EXPECT_EQ(LinearAlg2D::getDist2BetweenLineSegments(bridge->a_.from_point_, bridge->a_.to_point_, bridge->b_.from_point_, bridge->b_.to_point_), 100 * 100) << "The bridges should be spaced 1 line width (100 units) apart."; + EXPECT_EQ(LinearAlg2D::getDist2BetweenLineSegments(bridge->a_.from_point_, bridge->a_.to_point_, bridge->b_.from_point_, bridge->b_.to_point_), 100 * 100) + << "The bridges should be spaced 1 line width (100 units) apart."; EXPECT_LT(LinearAlg2D::pointIsLeftOfLine(bridge->b_.from_point_, bridge->a_.from_point_, bridge->a_.to_point_), 0) << "Connection B should be to the right of connection A."; EXPECT_LT(LinearAlg2D::pointIsLeftOfLine(bridge->b_.to_point_, bridge->a_.from_point_, bridge->a_.to_point_), 0) << "Connection B should be to the right of connection A."; } @@ -90,7 +94,8 @@ TEST_F(PolygonConnectorTest, getBridgeAdjacentSquares) EXPECT_EQ(vSize(bridge->a_.from_point_ - bridge->a_.to_point_), 100) << "The polygons are 100 units spaced apart, so this is the shortest possible bridge."; EXPECT_EQ(vSize(bridge->b_.from_point_ - bridge->b_.to_point_), 100) << "The second bridge should also be equally short in this case."; - EXPECT_EQ(LinearAlg2D::getDist2BetweenLineSegments(bridge->a_.from_point_, bridge->a_.to_point_, bridge->b_.from_point_, bridge->b_.to_point_), 100 * 100) << "The bridges should be spaced 1 line width (100 units) apart."; + EXPECT_EQ(LinearAlg2D::getDist2BetweenLineSegments(bridge->a_.from_point_, bridge->a_.to_point_, bridge->b_.from_point_, bridge->b_.to_point_), 100 * 100) + << "The bridges should be spaced 1 line width (100 units) apart."; EXPECT_LT(LinearAlg2D::pointIsLeftOfLine(bridge->b_.from_point_, bridge->a_.from_point_, bridge->a_.to_point_), 0) << "Connection B should be to the right of connection A."; EXPECT_LT(LinearAlg2D::pointIsLeftOfLine(bridge->b_.to_point_, bridge->a_.from_point_, bridge->a_.to_point_), 0) << "Connection B should be to the right of connection A."; } @@ -113,8 +118,10 @@ TEST_F(PolygonConnectorTest, getBridgeClosest) ASSERT_NE(bridge, std::nullopt) << "The two polygons are adjacent and spaced closely enough to bridge along their entire side, even with the slant."; - EXPECT_EQ(bridge->b_.from_point_, Point2LL(1000, 200)) << "The closest connection is [1000,200] -> [1100,200]. There is no space to the right of that, so bridge B should be there."; - EXPECT_EQ(bridge->b_.to_point_, Point2LL(1100, 200)) << "The closest connection is [1000,200] -> [1100,200]. There is no space to the right of that, so bridge B should be there."; + EXPECT_EQ(bridge->b_.from_point_, Point2LL(1000, 200)) + << "The closest connection is [1000,200] -> [1100,200]. There is no space to the right of that, so bridge B should be there."; + EXPECT_EQ(bridge->b_.to_point_, Point2LL(1100, 200)) + << "The closest connection is [1000,200] -> [1100,200]. There is no space to the right of that, so bridge B should be there."; EXPECT_GT(LinearAlg2D::pointIsLeftOfLine(bridge->a_.from_point_, bridge->b_.from_point_, bridge->b_.to_point_), 0) << "Connection A should be to the left of connection B."; EXPECT_GT(LinearAlg2D::pointIsLeftOfLine(bridge->a_.to_point_, bridge->b_.from_point_, bridge->b_.to_point_), 0) << "Connection A should be to the left of connection B."; } @@ -133,7 +140,8 @@ TEST_F(PolygonConnectorTest, getBridgeTooFar) std::optional> bridge = pc->getBridge(test_square, to_connect); - EXPECT_EQ(bridge, std::nullopt) << "The two polygons are 200 units apart where they are closest, which is more than 1.5 times the line width (100), so they can't be connected."; + EXPECT_EQ(bridge, std::nullopt) + << "The two polygons are 200 units apart where they are closest, which is more than 1.5 times the line width (100), so they can't be connected."; } /*! @@ -154,7 +162,8 @@ TEST_F(PolygonConnectorTest, getBridgeTooNarrow) std::optional> bridge = pc->getBridge(test_square, to_connect); - EXPECT_EQ(bridge, std::nullopt) << "Where the two polygons are adjacent is only 80 units wide. This is not enough to create a bridge with the connecting lines spaced 1 line width (100 units) apart."; + EXPECT_EQ(bridge, std::nullopt) + << "Where the two polygons are adjacent is only 80 units wide. This is not enough to create a bridge with the connecting lines spaced 1 line width (100 units) apart."; } /*! @@ -164,14 +173,14 @@ TEST_F(PolygonConnectorTest, getBridgeTooNarrow) */ TEST_F(PolygonConnectorTest, connectFourNested) { - Polygons connecting; - connecting.add(test_square_around); // 1200-wide square. - connecting.add(test_square); // 1000-wide square. - connecting.add(test_square.offset(-100)); // 800-wide square. - connecting.add(test_square.offset(-200)); // 600-wide square. + Shape connecting; + connecting.push_back(test_square_around); // 1200-wide square. + connecting.push_back(test_square); // 1000-wide square. + connecting.push_back(test_square.offset(-100)); // 800-wide square. + connecting.push_back(test_square.offset(-200)); // 600-wide square. pc->add(connecting); - Polygons output_polygons; + Shape output_polygons; std::vector output_paths; pc->connect(output_polygons, output_paths); diff --git a/tests/utils/PolygonTest.cpp b/tests/utils/PolygonTest.cpp index 7c0f5b566f..fec93cad6b 100644 --- a/tests/utils/PolygonTest.cpp +++ b/tests/utils/PolygonTest.cpp @@ -1,10 +1,14 @@ -// Copyright (c) 2022 Ultimaker B.V. +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher. -#include "utils/polygon.h" // The class under test. +#include "geometry/Polygon.h" // The class under test. + +#include #include +#include "geometry/OpenPolyline.h" +#include "geometry/SingleShape.h" #include "utils/Coord_t.h" #include "utils/SVG.h" // helper functions #include "utils/polygonUtils.h" // helper functions @@ -23,7 +27,7 @@ class PolygonTest : public testing::Test Polygon clipper_bug; Polygon clockwise_large; Polygon clockwise_small; - Polygons clockwise_donut; + Shape clockwise_donut; Polygon line; Polygon small_area; @@ -65,10 +69,10 @@ class PolygonTest : public testing::Test clockwise_small.emplace_back(50, 50); clockwise_small.emplace_back(50, -50); - Polygons outer; - Polygons inner; - outer.add(clockwise_large); - inner.add(clockwise_small); + Shape outer; + Shape inner; + outer.push_back(clockwise_large); + inner.push_back(clockwise_small); clockwise_donut = outer.difference(inner); line.emplace_back(0, 0); @@ -79,9 +83,9 @@ class PolygonTest : public testing::Test small_area.emplace_back(10, 10); small_area.emplace_back(0, 10); } - void twoPolygonsAreEqual(Polygons& polygon1, Polygons& polygon2) const + void twoPolygonsAreEqual(Shape& shape1, Shape& shape2) const { - auto poly_cmp = [](const ClipperLib::Path& a, const ClipperLib::Path& b) + auto poly_cmp = [](const Polygon& a, const Polygon& b) { return std::lexicographical_compare( a.begin(), @@ -93,15 +97,15 @@ class PolygonTest : public testing::Test return p1 < p2; }); }; - std::sort(polygon1.begin(), polygon1.end(), poly_cmp); - std::sort(polygon2.begin(), polygon2.end(), poly_cmp); + std::sort(shape1.begin(), shape1.end(), poly_cmp); + std::sort(shape2.begin(), shape2.end(), poly_cmp); - std::vector difference; - std::set_difference(polygon1.begin(), polygon1.end(), polygon2.begin(), polygon2.end(), std::back_inserter(difference), poly_cmp); + LinesSet difference; + std::set_difference(shape1.begin(), shape1.end(), shape2.begin(), shape2.end(), std::back_inserter(difference), poly_cmp); ASSERT_TRUE(difference.empty()) << "Paths in polygon1 not found in polygon2:" << difference; difference.clear(); - std::set_difference(polygon2.begin(), polygon2.end(), polygon1.begin(), polygon1.end(), std::back_inserter(difference), poly_cmp); + std::set_difference(shape2.begin(), shape2.end(), shape1.begin(), shape1.end(), std::back_inserter(difference), poly_cmp); ASSERT_TRUE(difference.empty()) << "Paths in polygon2 not found in polygon1:" << difference; } }; @@ -109,30 +113,30 @@ class PolygonTest : public testing::Test TEST_F(PolygonTest, polygonOffsetTest) { - Polygons test_squares; - test_squares.add(test_square); - const Polygons expanded = test_squares.offset(25); - const coord_t expanded_length = expanded.polygonLength(); + Shape test_squares; + test_squares.push_back(test_square); + const Shape expanded = test_squares.offset(25); + const coord_t expanded_length = expanded.length(); - Polygons square_hole; - PolygonRef square_inverted = square_hole.newPoly(); + Shape square_hole; + Polygon& square_inverted = square_hole.newLine(); for (int i = test_square.size() - 1; i >= 0; i--) { - square_inverted.add(test_square[i]); + square_inverted.push_back(test_square[i]); } - const Polygons contracted = square_hole.offset(25); - const coord_t contracted_length = contracted.polygonLength(); + const Shape contracted = square_hole.offset(25); + const coord_t contracted_length = contracted.length(); ASSERT_NEAR(expanded_length, contracted_length, 5) << "Offset on outside poly is different from offset on inverted poly!"; } TEST_F(PolygonTest, polygonOffsetBugTest) { - Polygons polys; - polys.add(clipper_bug); - const Polygons offsetted = polys.offset(-20); + Shape polys; + polys.push_back(clipper_bug); + const Shape offsetted = polys.offset(-20); - for (const ConstPolygonRef poly : offsetted) + for (const Polygon& poly : offsetted) { for (const Point2LL& p : poly) { @@ -143,8 +147,8 @@ TEST_F(PolygonTest, polygonOffsetBugTest) TEST_F(PolygonTest, isOutsideTest) { - Polygons test_triangle; - test_triangle.add(triangle); + Shape test_triangle; + test_triangle.push_back(triangle); EXPECT_FALSE(test_triangle.inside(Point2LL(0, 100))) << "Left point should be outside the triangle."; EXPECT_FALSE(test_triangle.inside(Point2LL(100, 100))) << "Middle left point should be outside the triangle."; @@ -156,28 +160,28 @@ TEST_F(PolygonTest, isOutsideTest) TEST_F(PolygonTest, isInsideTest) { - Polygons test_polys; - PolygonRef poly = test_polys.newPoly(); - poly.add(Point2LL(82124, 98235)); - poly.add(Point2LL(83179, 98691)); - poly.add(Point2LL(83434, 98950)); - poly.add(Point2LL(82751, 99026)); - poly.add(Point2LL(82528, 99019)); - poly.add(Point2LL(81605, 98854)); - poly.add(Point2LL(80401, 98686)); - poly.add(Point2LL(79191, 98595)); - poly.add(Point2LL(78191, 98441)); - poly.add(Point2LL(78998, 98299)); - poly.add(Point2LL(79747, 98179)); - poly.add(Point2LL(80960, 98095)); + Shape test_polys; + Polygon& poly = test_polys.newLine(); + poly.push_back(Point2LL(82124, 98235)); + poly.push_back(Point2LL(83179, 98691)); + poly.push_back(Point2LL(83434, 98950)); + poly.push_back(Point2LL(82751, 99026)); + poly.push_back(Point2LL(82528, 99019)); + poly.push_back(Point2LL(81605, 98854)); + poly.push_back(Point2LL(80401, 98686)); + poly.push_back(Point2LL(79191, 98595)); + poly.push_back(Point2LL(78191, 98441)); + poly.push_back(Point2LL(78998, 98299)); + poly.push_back(Point2LL(79747, 98179)); + poly.push_back(Point2LL(80960, 98095)); EXPECT_TRUE(test_polys.inside(Point2LL(78315, 98440))) << "Point should be inside the polygons!"; } TEST_F(PolygonTest, isOnBorderTest) { - Polygons test_triangle; - test_triangle.add(triangle); + Shape test_triangle; + test_triangle.push_back(triangle); EXPECT_FALSE(test_triangle.inside(Point2LL(200, 0), false)) << "Point is on the bottom edge of the triangle."; EXPECT_TRUE(test_triangle.inside(Point2LL(200, 0), true)) << "Point is on the bottom edge of the triangle."; @@ -187,8 +191,8 @@ TEST_F(PolygonTest, isOnBorderTest) TEST_F(PolygonTest, DISABLED_isInsideLineTest) // Disabled because this fails due to a bug in Clipper. { - Polygons polys; - polys.add(line); + Shape polys; + polys.push_back(line); EXPECT_FALSE(polys.inside(Point2LL(50, 0), false)) << "Should be outside since it is on the border and border is considered outside."; EXPECT_TRUE(polys.inside(Point2LL(50, 0), true)) << "Should be inside since it is on the border and border is considered inside."; @@ -196,25 +200,25 @@ TEST_F(PolygonTest, DISABLED_isInsideLineTest) // Disabled because this fails du TEST_F(PolygonTest, splitIntoPartsWithHoleTest) { - const std::vector parts = clockwise_donut.splitIntoParts(); + const std::vector parts = clockwise_donut.splitIntoParts(); - EXPECT_EQ(parts.size(), 1) << "Difference between two polygons should be one PolygonsPart!"; + EXPECT_EQ(parts.size(), 1) << "Difference between two polygons should be one SingleShape!"; } TEST_F(PolygonTest, differenceContainsOriginalPointTest) { - const PolygonsPart part = clockwise_donut.splitIntoParts()[0]; - const ConstPolygonRef outer = part.outerPolygon(); + const SingleShape part = clockwise_donut.splitIntoParts()[0]; + const Polygon& outer = part.outerPolygon(); EXPECT_NE(std::find(outer.begin(), outer.end(), clockwise_large[0]), outer.end()) << "Outer vertex must be in polygons difference."; - const ConstPolygonRef inner = part[1]; + const Polygon& inner = part[1]; EXPECT_NE(std::find(inner.begin(), inner.end(), clockwise_small[0]), inner.end()) << "Inner vertex must be in polygons difference."; } TEST_F(PolygonTest, differenceClockwiseTest) { - const PolygonsPart part = clockwise_donut.splitIntoParts()[0]; + const SingleShape part = clockwise_donut.splitIntoParts()[0]; - const ConstPolygonRef outer = part.outerPolygon(); + const Polygon& outer = part.outerPolygon(); // Apply the shoelace formula to determine surface area. If it's negative, the polygon is counterclockwise. coord_t area = 0; for (size_t point_index = 0; point_index < outer.size(); point_index++) @@ -226,7 +230,7 @@ TEST_F(PolygonTest, differenceClockwiseTest) } EXPECT_LT(area, 0) << "Outer polygon should be counter-clockwise."; - const ConstPolygonRef inner = part[1]; + const Polygon& inner = part[1]; area = 0; for (size_t point_index = 0; point_index < inner.size(); point_index++) { @@ -243,12 +247,12 @@ TEST_F(PolygonTest, differenceClockwiseTest) */ TEST_F(PolygonTest, convexTestCube) { - Polygons d_polygons; - PolygonRef d = d_polygons.newPoly(); - d.add(Point2LL(0, 0)); - d.add(Point2LL(10, 0)); - d.add(Point2LL(10, 10)); - d.add(Point2LL(0, 10)); + Shape d_polygons; + Polygon& d = d_polygons.newLine(); + d.push_back(Point2LL(0, 0)); + d.push_back(Point2LL(10, 0)); + d.push_back(Point2LL(10, 10)); + d.push_back(Point2LL(0, 10)); d_polygons.makeConvex(); @@ -264,8 +268,8 @@ TEST_F(PolygonTest, convexTestCube) */ TEST_F(PolygonTest, convexHullStar) { - Polygons d_polygons; - PolygonRef d = d_polygons.newPoly(); + Shape d_polygons; + Polygon& d = d_polygons.newLine(); const int num_points = 10; const int outer_radius = 20; @@ -275,11 +279,11 @@ TEST_F(PolygonTest, convexHullStar) { coord_t x_outer = -std::cos(angle_step * i) * outer_radius; coord_t y_outer = -std::sin(angle_step * i) * outer_radius; - d.add(Point2LL(x_outer, y_outer)); + d.push_back(Point2LL(x_outer, y_outer)); coord_t x_inner = -std::cos(angle_step * (i + 0.5)) * inner_radius; coord_t y_inner = -std::sin(angle_step * (i + 0.5)) * inner_radius; - d.add(Point2LL(x_inner, y_inner)); + d.push_back(Point2LL(x_inner, y_inner)); } d_polygons.makeConvex(); @@ -300,12 +304,12 @@ TEST_F(PolygonTest, convexHullStar) */ TEST_F(PolygonTest, convexHullMultipleMinX) { - Polygons d_polygons; - PolygonRef d = d_polygons.newPoly(); - d.add(Point2LL(0, 0)); - d.add(Point2LL(0, -10)); - d.add(Point2LL(10, 0)); - d.add(Point2LL(0, 10)); + Shape d_polygons; + Polygon& d = d_polygons.newLine(); + d.push_back(Point2LL(0, 0)); + d.push_back(Point2LL(0, -10)); + d.push_back(Point2LL(10, 0)); + d.push_back(Point2LL(0, 10)); /* * x\ x\ @@ -326,16 +330,16 @@ TEST_F(PolygonTest, convexHullMultipleMinX) */ TEST_F(PolygonTest, convexTestCubeColinear) { - Polygons d_polygons; - PolygonRef d = d_polygons.newPoly(); - d.add(Point2LL(0, 0)); - d.add(Point2LL(5, 0)); - d.add(Point2LL(10, 0)); - d.add(Point2LL(10, 5)); - d.add(Point2LL(10, 10)); - d.add(Point2LL(5, 10)); - d.add(Point2LL(0, 10)); - d.add(Point2LL(0, 5)); + Shape d_polygons; + Polygon& d = d_polygons.newLine(); + d.push_back(Point2LL(0, 0)); + d.push_back(Point2LL(5, 0)); + d.push_back(Point2LL(10, 0)); + d.push_back(Point2LL(10, 5)); + d.push_back(Point2LL(10, 10)); + d.push_back(Point2LL(5, 10)); + d.push_back(Point2LL(0, 10)); + d.push_back(Point2LL(0, 5)); d_polygons.makeConvex(); @@ -351,16 +355,16 @@ TEST_F(PolygonTest, convexTestCubeColinear) */ TEST_F(PolygonTest, convexHullRemoveDuplicatePoints) { - Polygons d_polygons; - PolygonRef d = d_polygons.newPoly(); - d.add(Point2LL(0, 0)); - d.add(Point2LL(0, 0)); - d.add(Point2LL(10, 0)); - d.add(Point2LL(10, 0)); - d.add(Point2LL(10, 10)); - d.add(Point2LL(10, 10)); - d.add(Point2LL(0, 10)); - d.add(Point2LL(0, 10)); + Shape d_polygons; + Polygon& d = d_polygons.newLine(); + d.push_back(Point2LL(0, 0)); + d.push_back(Point2LL(0, 0)); + d.push_back(Point2LL(10, 0)); + d.push_back(Point2LL(10, 0)); + d.push_back(Point2LL(10, 10)); + d.push_back(Point2LL(10, 10)); + d.push_back(Point2LL(0, 10)); + d.push_back(Point2LL(0, 10)); d_polygons.makeConvex(); @@ -380,10 +384,10 @@ TEST_F(PolygonTest, removeSmallAreas_simple) // basic set of polygons auto test_square_2 = test_square; test_square_2.translate(Point2LL(0, 500)); - auto d_polygons = Polygons{}; - d_polygons.add(test_square); - d_polygons.add(test_square_2); - d_polygons.add(triangle); + auto d_polygons = Shape{}; + d_polygons.push_back(test_square); + d_polygons.push_back(test_square_2); + d_polygons.push_back(triangle); // for the simple case there should be no change. auto act_polygons = d_polygons; @@ -411,16 +415,16 @@ TEST_F(PolygonTest, removeSmallAreas_small_area) triangle_1.translate(Point2LL(50, 0)); // add areas to polygons - auto d_polygons = Polygons{}; - d_polygons.add(small_area_1); - d_polygons.add(small_area_2); - d_polygons.add(test_square); // area = 10000 micron^2 = 1e-2 mm^2 - d_polygons.add(triangle_1); + auto d_polygons = Shape{}; + d_polygons.push_back(small_area_1); + d_polygons.push_back(small_area_2); + d_polygons.push_back(test_square); // area = 10000 micron^2 = 1e-2 mm^2 + d_polygons.push_back(triangle_1); - // make an expected Polygons - auto exp_polygons = Polygons{}; - exp_polygons.add(test_square); - exp_polygons.add(triangle_1); + // make an expected Shape + auto exp_polygons = Shape{}; + exp_polygons.push_back(test_square); + exp_polygons.push_back(triangle_1); // for remove_holes == false, 2 poly removed auto act_polygons = d_polygons; @@ -445,9 +449,9 @@ TEST_F(PolygonTest, removeSmallAreas_hole) small_hole_1.translate(Point2LL(10, 10)); // add areas to polygons - auto d_polygons = Polygons{}; - d_polygons.add(test_square); // area = 10000 micron^2 = 1e-2 mm^2 - d_polygons.add(small_hole_1); + auto d_polygons = Shape{}; + d_polygons.push_back(test_square); // area = 10000 micron^2 = 1e-2 mm^2 + d_polygons.push_back(small_hole_1); // for remove_holes == false there should be no change. @@ -456,9 +460,9 @@ TEST_F(PolygonTest, removeSmallAreas_hole) twoPolygonsAreEqual(act_polygons, d_polygons); // for remove_holes == true there should be one less poly. - // make an expected Polygons - auto exp_polygons = Polygons{}; - exp_polygons.add(test_square); + // make an expected Shape + auto exp_polygons = Shape{}; + exp_polygons.push_back(test_square); act_polygons = d_polygons; act_polygons.removeSmallAreas(1e-3, true); twoPolygonsAreEqual(act_polygons, exp_polygons); @@ -476,33 +480,33 @@ TEST_F(PolygonTest, removeSmallAreas_hole_2) small_hole_1.translate(Point2LL(10, 10)); small_hole_2.translate(Point2LL(160, 160)); auto med_square_1 = Polygon{}; // area = 2500 micron^2 = 2.5e-3 mm^2 - med_square_1.add(Point2LL(0, 0)); - med_square_1.add(Point2LL(50, 0)); - med_square_1.add(Point2LL(50, 50)); - med_square_1.add(Point2LL(0, 50)); + med_square_1.push_back(Point2LL(0, 0)); + med_square_1.push_back(Point2LL(50, 0)); + med_square_1.push_back(Point2LL(50, 50)); + med_square_1.push_back(Point2LL(0, 50)); med_square_1.translate(Point2LL(150, 150)); // add areas to polygons - auto d_polygons = Polygons{}; - d_polygons.add(test_square); // area = 10000 micron^2 = 1e-2 mm^2 - d_polygons.add(small_hole_1); - d_polygons.add(med_square_1); - d_polygons.add(small_hole_2); + auto d_polygons = Shape{}; + d_polygons.push_back(test_square); // area = 10000 micron^2 = 1e-2 mm^2 + d_polygons.push_back(small_hole_1); + d_polygons.push_back(med_square_1); + d_polygons.push_back(small_hole_2); // for remove_holes == false, two polygons removed. auto act_polygons = d_polygons; - // make an expected Polygons - auto exp_polygons = Polygons{}; - exp_polygons.add(test_square); - exp_polygons.add(small_hole_1); + // make an expected Shape + auto exp_polygons = Shape{}; + exp_polygons.push_back(test_square); + exp_polygons.push_back(small_hole_1); act_polygons.removeSmallAreas(3e-3, false); twoPolygonsAreEqual(act_polygons, exp_polygons); // for remove_holes == true, three polygons removed. act_polygons = d_polygons; // make an expected Polygons - exp_polygons = Polygons{}; - exp_polygons.add(test_square); + exp_polygons = Shape{}; + exp_polygons.push_back(test_square); act_polygons.removeSmallAreas(3e-3, true); twoPolygonsAreEqual(act_polygons, exp_polygons); } @@ -529,33 +533,60 @@ TEST_F(PolygonTest, removeSmallAreas_complex) triangle_1.translate(Point2LL(600, 0)); // add areas to polygons - auto d_polygons = Polygons{}; - d_polygons.add(small_area_1); - d_polygons.add(small_area_2); - d_polygons.add(test_square); // area = 10000 micron^2 = 1e-2 mm^2 - d_polygons.add(small_hole_1); - d_polygons.add(small_hole_2); - d_polygons.add(triangle_1); + auto d_polygons = Shape{}; + d_polygons.push_back(small_area_1); + d_polygons.push_back(small_area_2); + d_polygons.push_back(test_square); // area = 10000 micron^2 = 1e-2 mm^2 + d_polygons.push_back(small_hole_1); + d_polygons.push_back(small_hole_2); + d_polygons.push_back(triangle_1); // for remove_holes == false there should be 2 small areas removed. auto act_polygons = d_polygons; - // make an expected Polygons - auto exp_polygons = Polygons{}; - exp_polygons.add(test_square); // area = 10000 micron^2 = 1e-2 mm^2 - exp_polygons.add(small_hole_1); - exp_polygons.add(small_hole_2); - exp_polygons.add(triangle_1); + // make an expected Shape + auto exp_polygons = Shape{}; + exp_polygons.push_back(test_square); // area = 10000 micron^2 = 1e-2 mm^2 + exp_polygons.push_back(small_hole_1); + exp_polygons.push_back(small_hole_2); + exp_polygons.push_back(triangle_1); act_polygons.removeSmallAreas(1e-3, false); twoPolygonsAreEqual(act_polygons, exp_polygons); // for remove_holes == true there should be 2 small areas and 2 small holes removed. act_polygons = d_polygons; // make an expected Polygons - exp_polygons = Polygons{}; - exp_polygons.add(test_square); - exp_polygons.add(triangle_1); // area = 10000 micron^2 = 1e-2 mm^2 + exp_polygons = Shape{}; + exp_polygons.push_back(test_square); + exp_polygons.push_back(triangle_1); // area = 10000 micron^2 = 1e-2 mm^2 act_polygons.removeSmallAreas(1e-3, true); twoPolygonsAreEqual(act_polygons, exp_polygons); } + +/* + * Test that we can iterate over segments of open/closed polylines and convert them to each other + */ +TEST_F(PolygonTest, openCloseLines) +{ + // make some line + OpenPolyline open_polyline; + open_polyline.emplace_back(0, 0); + open_polyline.emplace_back(1000, 0); + open_polyline.emplace_back(1000, 1000); + open_polyline.emplace_back(0, 1000); + + // Make some casts and check that the results are consistent + EXPECT_EQ(open_polyline.length(), 3000); + + ClosedPolyline closed_polyline(open_polyline.getPoints(), false); + EXPECT_EQ(closed_polyline.length(), 4000); + + Polygon polygon(closed_polyline.getPoints(), false); + EXPECT_EQ(polygon.area(), 1000000); + + // Check advanced calculations on segment length + EXPECT_TRUE(open_polyline.shorterThan(3500)); + EXPECT_FALSE(closed_polyline.shorterThan(3500)); +} + } // namespace cura // NOLINTEND(*-magic-numbers) diff --git a/tests/utils/PolygonUtilsTest.cpp b/tests/utils/PolygonUtilsTest.cpp index f859a48c08..cc1907390e 100644 --- a/tests/utils/PolygonUtilsTest.cpp +++ b/tests/utils/PolygonUtilsTest.cpp @@ -5,9 +5,9 @@ #include +#include "geometry/Point2LL.h" // Creating and testing with points. +#include "geometry/Polygon.h" // Creating polygons to test with. #include "utils/Coord_t.h" -#include "utils/Point2LL.h" // Creating and testing with points. -#include "utils/polygon.h" // Creating polygons to test with. // NOLINTBEGIN(*-magic-numbers) namespace cura @@ -56,7 +56,7 @@ class MoveInsideTest : public testing::TestWithParam TEST_P(MoveInsideTest, MoveInside) { const MoveInsideParameters parameters = GetParam(); - const ClosestPolygonPoint cpp = PolygonUtils::findClosest(parameters.close_to, test_square); + const ClosestPointPolygon cpp = PolygonUtils::findClosest(parameters.close_to, test_square); Point2LL result = PolygonUtils::moveInside(cpp, parameters.distance); // FIXME: Clean-up message with ftm when CURA-8258 is implemented or when we use C++20 @@ -70,8 +70,8 @@ TEST_P(MoveInsideTest, MoveInside) TEST_P(MoveInsideTest, MoveInside2) { const MoveInsideParameters parameters = GetParam(); - Polygons polys; - polys.add(test_square); + Shape polys; + polys.push_back(test_square); Point2LL result = parameters.close_to; PolygonUtils::moveInside2(polys, result, parameters.distance); ASSERT_LE(vSize(result - parameters.supposed), 10) << parameters.close_to << " moved with " << parameters.distance << " micron inside to " << result << "rather than " @@ -99,7 +99,7 @@ TEST_F(MoveInsideTest, cornerEdgeTest) const Point2LL supposed1(80, 80); // Allow two possible values here, since the behaviour for this edge case is not specified. const Point2LL supposed2(72, 100); constexpr coord_t distance = 28; - const ClosestPolygonPoint cpp = PolygonUtils::findClosest(close_to, test_square); + const ClosestPointPolygon cpp = PolygonUtils::findClosest(close_to, test_square); const Point2LL result = PolygonUtils::moveInside(cpp, distance); constexpr coord_t maximum_error = 10; @@ -120,7 +120,7 @@ TEST_F(MoveInsideTest, middleTest) const Point2LL supposed3(20, 50); const Point2LL supposed4(50, 20); constexpr coord_t distance = 20; - const ClosestPolygonPoint cpp = PolygonUtils::findClosest(close_to, test_square); + const ClosestPointPolygon cpp = PolygonUtils::findClosest(close_to, test_square); const Point2LL result = PolygonUtils::moveInside(cpp, distance); constexpr coord_t maximum_error = 10; @@ -142,7 +142,7 @@ TEST_F(MoveInsideTest, middleTestPenalty) const Point2LL supposed(80, 50); const Point2LL preferred_dir(120, 60); constexpr coord_t distance = 20; - const ClosestPolygonPoint cpp = PolygonUtils::findClosest( + const ClosestPointPolygon cpp = PolygonUtils::findClosest( close_to, test_square, [preferred_dir](Point2LL candidate) @@ -164,8 +164,8 @@ TEST_F(MoveInsideTest, cornerEdgeTest2) const Point2LL supposed1(80, 80); // Allow two possible values here, since the behaviour for this edge case is not specified. const Point2LL supposed2(72, 100); constexpr coord_t distance = 28; - Polygons polys; - polys.add(test_square); + Shape polys; + polys.push_back(test_square); Point2LL result = close_to; PolygonUtils::moveInside2(polys, result, distance); @@ -178,9 +178,9 @@ TEST_F(MoveInsideTest, pointyCorner) { const Point2LL from(55, 100); // Above pointy bit. Point2LL result(from); - Polygons inside; - inside.add(pointy_square); - ClosestPolygonPoint cpp = PolygonUtils::ensureInsideOrOutside(inside, result, 10); + Shape inside; + inside.push_back(pointy_square); + ClosestPointPolygon cpp = PolygonUtils::ensureInsideOrOutside(inside, result, 10); ASSERT_NE(cpp.point_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; ASSERT_NE(cpp.poly_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; @@ -192,10 +192,10 @@ TEST_F(MoveInsideTest, pointyCornerFail) // Should fail with normal moveInside2 (and the like). const Point2LL from(55, 170); // Above pointy bit. Point2LL result(from); - Polygons inside; - inside.add(pointy_square); + Shape inside; + inside.push_back(pointy_square); - ClosestPolygonPoint cpp = PolygonUtils::moveInside2(inside, result, 10); + ClosestPointPolygon cpp = PolygonUtils::moveInside2(inside, result, 10); ASSERT_NE(cpp.point_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; ASSERT_NE(cpp.poly_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; ASSERT_FALSE(inside.inside(result)) << from << " could be moved inside, while it was designed to fail."; @@ -206,10 +206,10 @@ TEST_F(MoveInsideTest, outsidePointyCorner) const Point2LL from(60, 70); // Above pointy bit. Point2LL result(from); const Point2LL supposed(50, 70); // 10 below pointy bit. - Polygons inside; - inside.add(pointy_square); + Shape inside; + inside.push_back(pointy_square); - const ClosestPolygonPoint cpp = PolygonUtils::ensureInsideOrOutside(inside, result, -10); + const ClosestPointPolygon cpp = PolygonUtils::ensureInsideOrOutside(inside, result, -10); ASSERT_NE(cpp.point_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; ASSERT_NE(cpp.poly_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; ASSERT_TRUE(! inside.inside(result)) << from << " couldn't be moved outside."; @@ -221,10 +221,10 @@ TEST_F(MoveInsideTest, outsidePointyCornerFail) const Point2LL from(60, 70); // Above pointy bit. Point2LL result(from); const Point2LL supposed(50, 70); // 10 below pointy bit. - Polygons inside; - inside.add(pointy_square); + Shape inside; + inside.push_back(pointy_square); - const ClosestPolygonPoint cpp = PolygonUtils::moveInside2(inside, result, -10); + const ClosestPointPolygon cpp = PolygonUtils::moveInside2(inside, result, -10); ASSERT_NE(cpp.point_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; ASSERT_NE(cpp.poly_idx_, NO_INDEX) << "Couldn't ensure point inside close to " << from << "."; ASSERT_FALSE(! inside.inside(result)) << from << " could be moved outside to " << result << ", while it was designed to fail."; @@ -265,11 +265,11 @@ class FindCloseTest : public testing::TestWithParam TEST_P(FindCloseTest, FindClose) { const FindCloseParameters parameters = GetParam(); - Polygons polygons; - polygons.add(test_square); + Shape polygons; + polygons.push_back(test_square); auto loc_to_line = PolygonUtils::createLocToLineGrid(polygons, parameters.cell_size); - std::optional cpp; + std::optional cpp; if (parameters.penalty_function) { cpp = PolygonUtils::findClose(parameters.close_to, polygons, *loc_to_line, *parameters.penalty_function); @@ -312,9 +312,9 @@ INSTANTIATE_TEST_SUITE_P( class PolygonUtilsTest : public testing::Test { public: - Polygons test_squares; - Polygons test_line; - Polygons test_line_extra_vertices; // Line that has extra vertices along it that are technically unnecessary. + Shape test_squares; + Shape test_line; + Shape test_line_extra_vertices; // Line that has extra vertices along it that are technically unnecessary. PolygonUtilsTest() { @@ -323,30 +323,30 @@ class PolygonUtilsTest : public testing::Test test_square.emplace_back(100, 0); test_square.emplace_back(100, 100); test_square.emplace_back(0, 100); - test_squares.add(test_square); + test_squares.push_back(test_square); Polygon line; line.emplace_back(0, 0); line.emplace_back(100, 0); - test_line.add(line); + test_line.push_back(line); Polygon line_extra_vertices; line_extra_vertices.emplace_back(100, 0); line_extra_vertices.emplace_back(25, 0); line_extra_vertices.emplace_back(0, 0); line_extra_vertices.emplace_back(75, 0); - test_line_extra_vertices.add(line_extra_vertices); + test_line_extra_vertices.push_back(line_extra_vertices); } }; TEST_F(PolygonUtilsTest, spreadDotsSegment) { - std::vector supposed; - supposed.emplace_back(Point2LL(50, 0), 0, test_squares[0], 0); - supposed.emplace_back(Point2LL(100, 0), 1, test_squares[0], 0); - supposed.emplace_back(Point2LL(100, 50), 1, test_squares[0], 0); + std::vector supposed; + supposed.emplace_back(Point2LL(50, 0), 0, &test_squares[0], 0); + supposed.emplace_back(Point2LL(100, 0), 1, &test_squares[0], 0); + supposed.emplace_back(Point2LL(100, 50), 1, &test_squares[0], 0); - std::vector result; + std::vector result; PolygonUtils::spreadDots(PolygonsPointIndex(&test_squares, 0, 0), PolygonsPointIndex(&test_squares, 0, 2), 3, result); ASSERT_EQ(result.size(), supposed.size()); @@ -358,17 +358,17 @@ TEST_F(PolygonUtilsTest, spreadDotsSegment) TEST_F(PolygonUtilsTest, spreadDotsFull) { - std::vector supposed; - supposed.emplace_back(Point2LL(0, 0), 0, test_squares[0], 0); - supposed.emplace_back(Point2LL(50, 0), 0, test_squares[0], 0); - supposed.emplace_back(Point2LL(100, 0), 1, test_squares[0], 0); - supposed.emplace_back(Point2LL(100, 50), 1, test_squares[0], 0); - supposed.emplace_back(Point2LL(100, 100), 2, test_squares[0], 0); - supposed.emplace_back(Point2LL(50, 100), 2, test_squares[0], 0); - supposed.emplace_back(Point2LL(0, 100), 3, test_squares[0], 0); - supposed.emplace_back(Point2LL(0, 50), 3, test_squares[0], 0); - - std::vector result; + std::vector supposed; + supposed.emplace_back(Point2LL(0, 0), 0, &test_squares[0], 0); + supposed.emplace_back(Point2LL(50, 0), 0, &test_squares[0], 0); + supposed.emplace_back(Point2LL(100, 0), 1, &test_squares[0], 0); + supposed.emplace_back(Point2LL(100, 50), 1, &test_squares[0], 0); + supposed.emplace_back(Point2LL(100, 100), 2, &test_squares[0], 0); + supposed.emplace_back(Point2LL(50, 100), 2, &test_squares[0], 0); + supposed.emplace_back(Point2LL(0, 100), 3, &test_squares[0], 0); + supposed.emplace_back(Point2LL(0, 50), 3, &test_squares[0], 0); + + std::vector result; PolygonUtils::spreadDots(PolygonsPointIndex(&test_squares, 0, 0), PolygonsPointIndex(&test_squares, 0, 0), 8, result); ASSERT_EQ(result.size(), supposed.size()); @@ -399,7 +399,7 @@ struct GetNextParallelIntersectionParameters class GetNextParallelIntersectionTest : public testing::TestWithParam { public: - Polygons test_squares; + Shape test_squares; GetNextParallelIntersectionTest() { @@ -408,7 +408,7 @@ class GetNextParallelIntersectionTest : public testing::TestWithParam computed = PolygonUtils::getNextParallelIntersection(start, parameters.line_to, parameters.dist, parameters.forward); + const ClosestPointPolygon start = PolygonUtils::findClosest(parameters.start_point, test_squares); + std::optional computed = PolygonUtils::getNextParallelIntersection(start, parameters.line_to, parameters.dist, parameters.forward); ASSERT_EQ(bool(parameters.predicted), bool(computed)) << "An answer was predicted but not computed, or computed but not predicted."; if (parameters.predicted) @@ -448,7 +448,7 @@ TEST_F(PolygonUtilsTest, RelativeHammingSquaresOverlap) TEST_F(PolygonUtilsTest, RelativeHammingDisjunct) { - Polygons shifted_polys = test_squares; // Make a copy. + Shape shifted_polys = test_squares; // Make a copy. shifted_polys[0].translate(Point2LL(200, 0)); ASSERT_EQ(PolygonUtils::relativeHammingDistance(test_squares, shifted_polys), 1.0); @@ -456,7 +456,7 @@ TEST_F(PolygonUtilsTest, RelativeHammingDisjunct) TEST_F(PolygonUtilsTest, RelativeHammingHalfOverlap) { - Polygons shifted_polys = test_squares; // Make a copy. + Shape shifted_polys = test_squares; // Make a copy. shifted_polys[0].translate(Point2LL(50, 0)); ASSERT_EQ(PolygonUtils::relativeHammingDistance(test_squares, shifted_polys), 0.5); @@ -469,7 +469,7 @@ TEST_F(PolygonUtilsTest, RelativeHammingHalfOverlap) */ TEST_F(PolygonUtilsTest, RelativeHammingQuarterOverlap) { - Polygons shifted_polys = test_squares; // Make a copy. + Shape shifted_polys = test_squares; // Make a copy. shifted_polys[0].translate(Point2LL(50, 50)); ASSERT_EQ(PolygonUtils::relativeHammingDistance(test_squares, shifted_polys), 0.75); @@ -506,7 +506,7 @@ TEST_F(PolygonUtilsTest, TEST_F(PolygonUtilsTest, RelativeHammingLineLineDisjunct) { - Polygons shifted_line = test_line; // Make a copy. + Shape shifted_line = test_line; // Make a copy. shifted_line[0].translate(Point2LL(0, 1)); ASSERT_EQ(PolygonUtils::relativeHammingDistance(test_line, test_line), 1.0); diff --git a/tests/utils/SimplifyTest.cpp b/tests/utils/SimplifyTest.cpp index 3b80885267..00cb98f0b6 100644 --- a/tests/utils/SimplifyTest.cpp +++ b/tests/utils/SimplifyTest.cpp @@ -3,6 +3,8 @@ #include "utils/Simplify.h" // The unit under test. +#include + #include #include "utils/Coord_t.h" @@ -50,7 +52,7 @@ class SimplifyTest : public testing::Test const double increment = segment_length / radius; // Segments of 990 units. for (double angle = 0; angle < tau; angle += increment) { - circle.add(Point2LL(std::cos(angle) * radius, std::sin(angle) * radius)); + circle.push_back(Point2LL(std::cos(angle) * radius, std::sin(angle) * radius)); } square_collinear.clear(); @@ -66,16 +68,16 @@ class SimplifyTest : public testing::Test switch (side) { case 0: - square_collinear.add(Point2LL(longitude, latitude)); + square_collinear.push_back(Point2LL(longitude, latitude)); break; case 1: - square_collinear.add(Point2LL(width + latitude, longitude)); + square_collinear.push_back(Point2LL(width + latitude, longitude)); break; case 2: - square_collinear.add(Point2LL(width - longitude, width + latitude)); + square_collinear.push_back(Point2LL(width - longitude, width + latitude)); break; case 3: - square_collinear.add(Point2LL(latitude, width - longitude)); + square_collinear.push_back(Point2LL(latitude, width - longitude)); break; } } @@ -88,7 +90,7 @@ class SimplifyTest : public testing::Test constexpr size_t periods = 10; // How many waves of the sine to construct. for (double current_sine = 0; current_sine < std::numbers::pi * periods; current_sine += sine_step) { - sine.add(Point2LL(std::sin(current_sine) * amplitude, y_step * sine.size())); + sine.push_back(Point2LL(std::sin(current_sine) * amplitude, y_step * sine.size())); } spiral.clear(); @@ -99,7 +101,7 @@ class SimplifyTest : public testing::Test radius = 0; for (size_t i = 0; i < vertex_count; ++i) { - spiral.add(Point2LL(std::cos(angle) * radius, std::sin(angle) * radius)); + spiral.push_back(Point2LL(std::cos(angle) * radius, std::sin(angle) * radius)); angle += angle_step; radius += radius_step; } @@ -108,7 +110,7 @@ class SimplifyTest : public testing::Test constexpr coord_t invfreq = 30; for (size_t i = 0; i < vertex_count; ++i) { - zigzag.add(Point2LL(-amplitude + (i % 2) * 2 * amplitude, i * invfreq)); + zigzag.push_back(Point2LL(-amplitude + (i % 2) * 2 * amplitude, i * invfreq)); } } }; @@ -172,7 +174,7 @@ TEST_F(SimplifyTest, CircleMaxDeviation) TEST_F(SimplifyTest, Zigzag) { simplifier.max_resolution_ = 9999999; - Polygon simplified = simplifier.polyline(zigzag); + ClosedPolyline simplified = simplifier.polyline(zigzag); EXPECT_EQ(simplified.size(), 2) << "All zigzagged lines can be erased because they deviate less than the maximum deviation, leaving only the endpoints."; } @@ -197,7 +199,7 @@ TEST_F(SimplifyTest, LimitedLength) } } - Polygon simplified = simplifier.polyline(spiral); + ClosedPolyline simplified = simplifier.polyline(spiral); // Look backwards until the limit vertex is reached to verify that the polygon is unaltered there. for (size_t i = 0; i < simplified.size(); ++i) @@ -224,19 +226,19 @@ TEST_F(SimplifyTest, LimitedError) // Generate a zig-zag with gradually increasing deviation. Polygon increasing_zigzag; - increasing_zigzag.add(Point2LL(0, 0)); + increasing_zigzag.push_back(Point2LL(0, 0)); constexpr coord_t amplitude_step = 1; // Every 2 vertices, the amplitude increases by this much. constexpr coord_t y_step = 100; const coord_t amplitude_limit = simplifier.max_deviation_ * 2; // Increase amplitude up to this point. About half of the vertices should get removed. for (coord_t amplitude = 0; amplitude < amplitude_limit; amplitude += amplitude_step) { - increasing_zigzag.add(Point2LL(amplitude, increasing_zigzag.size() * y_step)); - increasing_zigzag.add(Point2LL(0, increasing_zigzag.size() * y_step)); + increasing_zigzag.push_back(Point2LL(amplitude, increasing_zigzag.size() * y_step)); + increasing_zigzag.push_back(Point2LL(0, increasing_zigzag.size() * y_step)); } size_t limit_vertex = 2 * simplifier.max_deviation_ / amplitude_step + 3; // 2 vertices per zag. Deviation/step zags. Add 3 since deviation equal to max +- epsilon is allowed. - Polygon simplified = simplifier.polyline(increasing_zigzag); + ClosedPolyline simplified = simplifier.polyline(increasing_zigzag); // Look backwards until the limit vertex is reached to verify that the polygon is unaltered there. for (size_t i = 0; i < simplified.size(); ++i) @@ -260,12 +262,12 @@ TEST_F(SimplifyTest, LimitedError) TEST_F(SimplifyTest, LongEdgesNotMoved) { Polygon polyline; - polyline.add(Point2LL(0, 0)); - polyline.add(Point2LL(10000, 10000)); // Long edge. - polyline.add(Point2LL(10010, 10000)); // Short edge. - polyline.add(Point2LL(21010, 0)); // Long edge. + polyline.push_back(Point2LL(0, 0)); + polyline.push_back(Point2LL(10000, 10000)); // Long edge. + polyline.push_back(Point2LL(10010, 10000)); // Short edge. + polyline.push_back(Point2LL(21010, 0)); // Long edge. - Polygon simplified = simplifier.polyline(polyline); + ClosedPolyline simplified = simplifier.polyline(polyline); // Verify that all small segments are removed. for (size_t i = 1; i < simplified.size(); ++i) @@ -297,12 +299,12 @@ TEST_F(SimplifyTest, LongEdgesNotMoved) TEST_F(SimplifyTest, LongEdgesButTooMuchDeviation) { Polygon polyline; - polyline.add(Point2LL(0, 0)); - polyline.add(Point2LL(0, 10000)); // Long edge. - polyline.add(Point2LL(10, 10000)); // Short edge. - polyline.add(Point2LL(20, 0)); // Long edge. Intersection with previous long edge is at 0,20000, which is too far. + polyline.push_back(Point2LL(0, 0)); + polyline.push_back(Point2LL(0, 10000)); // Long edge. + polyline.push_back(Point2LL(10, 10000)); // Short edge. + polyline.push_back(Point2LL(20, 0)); // Long edge. Intersection with previous long edge is at 0,20000, which is too far. - Polygon simplified = simplifier.polyline(polyline); + ClosedPolyline simplified = simplifier.polyline(polyline); // Verify that the polyline is unchanged. ASSERT_EQ(polyline.size(), simplified.size()) << "The polyline may not have been simplified because that would introduce vertices that deviate too much."; @@ -312,7 +314,7 @@ TEST_F(SimplifyTest, LongEdgesButTooMuchDeviation) } polyline.pop_back(); - polyline.add(Point2LL(10, 0)); // Replace last vertex with one that makes the two long edges exactly parallel. + polyline.push_back(Point2LL(10, 0)); // Replace last vertex with one that makes the two long edges exactly parallel. simplified = simplifier.polyline(polyline); @@ -335,7 +337,7 @@ TEST_F(SimplifyTest, LongEdgesButTooMuchDeviation) TEST_F(SimplifyTest, Sine) { simplifier.max_resolution_ = 9999999; - Polygon simplified = simplifier.polyline(sine); + ClosedPolyline simplified = simplifier.polyline(sine); EXPECT_EQ(simplified.size(), 2) << "All zigzagged lines can be erased because they deviate less than the maximum deviation, leaving only the endpoints."; } @@ -364,13 +366,13 @@ TEST_F(SimplifyTest, IdenticalVertices) switch (vertex) { case 0: - polygon.add(Point2LL(0, 0)); + polygon.push_back(Point2LL(0, 0)); break; case 1: - polygon.add(Point2LL(10000, 0)); + polygon.push_back(Point2LL(10000, 0)); break; case 2: - polygon.add(Point2LL(5000, 10000)); + polygon.push_back(Point2LL(5000, 10000)); break; } } @@ -387,17 +389,17 @@ TEST_F(SimplifyTest, ToDegenerate) { // Create a triangle where one of the vertices could be removed. Polygon triangle; - triangle.add(Point2LL(0, 0)); - triangle.add(Point2LL(1100, 0)); - triangle.add(Point2LL(550, 50)); // Deviates by 50, and both adjacent edges are just over 550 long. Could be removed. + triangle.push_back(Point2LL(0, 0)); + triangle.push_back(Point2LL(1100, 0)); + triangle.push_back(Point2LL(550, 50)); // Deviates by 50, and both adjacent edges are just over 550 long. Could be removed. triangle = simplifier.polygon(triangle); EXPECT_EQ(triangle.size(), 3) << "The triangle did not get simplified because that would reduce its vertices to less than 3, making it degenerate."; // Create a polyline that is shorter than the minimum resolution. - Polygon segment; - segment.add(Point2LL(0, 0)); - segment.add(Point2LL(4, 0)); // Less than 5 micron long, so vertices would always be removed. + ClosedPolyline segment; + segment.push_back(Point2LL(0, 0)); + segment.push_back(Point2LL(4, 0)); // Less than 5 micron long, so vertices would always be removed. segment = simplifier.polyline(segment); EXPECT_EQ(segment.size(), 0) << "The segment got removed entirely, because simplification would reduce its vertices to less than 2, making it degenerate."; diff --git a/tests/utils/SmoothTest.cpp b/tests/utils/SmoothTest.cpp index 1864e4b706..3fbcda11f1 100644 --- a/tests/utils/SmoothTest.cpp +++ b/tests/utils/SmoothTest.cpp @@ -1,15 +1,16 @@ -// Copyright (c) 2023 UltiMaker +// Copyright (c) 2024 UltiMaker // CuraEngine is released under the terms of the AGPLv3 or higher -#include +#include "utils/actions/smooth.h" -#include +#include #include -#include "utils/Point2LL.h" -#include "utils/actions/smooth.h" -#include "utils/polygon.h" +#include + +#include "geometry/Point2LL.h" +#include "geometry/Polygon.h" namespace cura { @@ -30,10 +31,10 @@ TEST(SmoothTest, TestSmooth) * */ - auto A = cura::Point2LL { 0, 0 }; - auto B = cura::Point2LL { 0, 100 }; - auto C = cura::Point2LL { 1, 100 }; - auto D = cura::Point2LL { 1, 200 }; + auto A = cura::Point2LL{ 0, 0 }; + auto B = cura::Point2LL{ 0, 100 }; + auto C = cura::Point2LL{ 1, 100 }; + auto D = cura::Point2LL{ 1, 200 }; const auto is_smooth = smooth.isSmooth(A, B, C, D, COS_FLUID_ANGLE); EXPECT_EQ(is_smooth, false); @@ -51,10 +52,10 @@ TEST(SmoothTest, TestSmooth) * D * */ - auto A = cura::Point2LL { 0, 0 }; - auto B = cura::Point2LL { 100, 0 }; - auto C = cura::Point2LL { 101, 1 }; - auto D = cura::Point2LL { 101, 101 }; + auto A = cura::Point2LL{ 0, 0 }; + auto B = cura::Point2LL{ 100, 0 }; + auto C = cura::Point2LL{ 101, 1 }; + auto D = cura::Point2LL{ 101, 101 }; const auto is_smooth = smooth.isSmooth(A, B, C, D, COS_FLUID_ANGLE); EXPECT_EQ(is_smooth, true); @@ -66,10 +67,10 @@ TEST(SmoothTest, TestSmooth) * A ----------- B - C -------------D * */ - auto A = cura::Point2LL { 0, 0 }; - auto B = cura::Point2LL { 100, 0 }; - auto C = cura::Point2LL { 101, 0 }; - auto D = cura::Point2LL { 201, 0 }; + auto A = cura::Point2LL{ 0, 0 }; + auto B = cura::Point2LL{ 100, 0 }; + auto C = cura::Point2LL{ 101, 0 }; + auto D = cura::Point2LL{ 201, 0 }; const auto is_smooth = smooth.isSmooth(A, B, C, D, COS_FLUID_ANGLE); EXPECT_EQ(is_smooth, true); @@ -81,10 +82,10 @@ TEST(SmoothTest, TestSmooth) * D ----------- C - B -------------A * */ - auto A = cura::Point2LL { 201, 0 }; - auto B = cura::Point2LL { 101, 0 }; - auto C = cura::Point2LL { 100, 0 }; - auto D = cura::Point2LL { 0, 0 }; + auto A = cura::Point2LL{ 201, 0 }; + auto B = cura::Point2LL{ 101, 0 }; + auto C = cura::Point2LL{ 100, 0 }; + auto D = cura::Point2LL{ 0, 0 }; const auto is_smooth = smooth.isSmooth(A, B, C, D, COS_FLUID_ANGLE); EXPECT_EQ(is_smooth, true); @@ -105,10 +106,10 @@ TEST(SmoothTest, TestSmooth) * D * */ - auto A = cura::Point2LL { 0, 0 }; - auto B = cura::Point2LL { 100, 0 }; - auto C = cura::Point2LL { 99, -1 }; - auto D = cura::Point2LL { 199, 99 }; + auto A = cura::Point2LL{ 0, 0 }; + auto B = cura::Point2LL{ 100, 0 }; + auto C = cura::Point2LL{ 99, -1 }; + auto D = cura::Point2LL{ 199, 99 }; const auto is_smooth = smooth.isSmooth(A, B, C, D, COS_FLUID_ANGLE); EXPECT_EQ(is_smooth, false); @@ -127,10 +128,10 @@ TEST(SmoothTest, TestSmooth) * D * */ - auto A = cura::Point2LL { 0, 0 }; - auto B = cura::Point2LL { 100, 0 }; - auto C = cura::Point2LL { 101, 1 }; - auto D = cura::Point2LL { 201, 101 }; + auto A = cura::Point2LL{ 0, 0 }; + auto B = cura::Point2LL{ 100, 0 }; + auto C = cura::Point2LL{ 101, 1 }; + auto D = cura::Point2LL{ 201, 101 }; const auto is_smooth = smooth.isSmooth(A, B, C, D, COS_FLUID_ANGLE); EXPECT_EQ(is_smooth, true); @@ -166,6 +167,5 @@ TEST(SmoothTest, TestSmooth) const auto is_smooth = smooth.isSmooth(A, B, C, D, COS_FLUID_ANGLE); EXPECT_EQ(is_smooth, false); }; - } -} // namespace cura \ No newline at end of file +} // namespace cura diff --git a/tests/utils/StringTest.cpp b/tests/utils/StringTest.cpp index 829c4c3da7..a0c172fbc8 100644 --- a/tests/utils/StringTest.cpp +++ b/tests/utils/StringTest.cpp @@ -2,7 +2,7 @@ // CuraEngine is released under the terms of the AGPLv3 or higher. #include "utils/string.h" // The file under test. -#include "utils/Point2LL.h" +#include "geometry/Point2LL.h" #include // NOLINTBEGIN(*-magic-numbers)