QGIS API Documentation 4.0.0-Norrköping (1ddcee3d0e4)
Loading...
Searching...
No Matches
qgsblanksegmentutils.cpp
Go to the documentation of this file.
1/***************************************************************************
2 blanksegmentutils.cpp
3 ---------------------
4 begin : 2025/11/05
5 copyright : (C) 2025 by Julien Cabieces
6 email : julien dot cabieces at oslandia dot com
7 ***************************************************************************
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 ***************************************************************************/
15
17
18#include "qgsrendercontext.h"
19
20#include <QString>
21
22using namespace Qt::StringLiterals;
23
24QList<QList<QgsBlankSegmentUtils::BlankSegments>> QgsBlankSegmentUtils::parseBlankSegments( const QString &strBlankSegments, const QgsRenderContext &renderContext, Qgis::RenderUnit unit, QString &error )
25{
26 QString currentNumber;
27 QList<QList<BlankSegments>> blankSegments;
28 constexpr char internalError[] = "Internal error while processing blank segments";
29
30 auto appendLevel = [&blankSegments, &internalError]( int level ) -> void {
31 if ( level == 0 )
32 {
33 blankSegments.append( QList<BlankSegments>() );
34 }
35 else if ( level == 1 )
36 {
37 if ( blankSegments.isEmpty() )
38 throw std::runtime_error( internalError ); // should not happen
39 blankSegments.back().append( BlankSegments() );
40 }
41 else if ( level == 2 )
42 {
43 if ( blankSegments.isEmpty() || blankSegments.back().isEmpty() )
44 throw std::runtime_error( internalError ); // should not happen
45 blankSegments.back().back().append( QPair<double, double>( -1, -1 ) );
46 }
47 else
48 throw std::runtime_error( internalError ); // should not happen
49 };
50
51 auto addNumber = [&blankSegments, &internalError, &currentNumber]( const QChar &c ) -> void {
52 if ( blankSegments.isEmpty() || blankSegments.back().isEmpty() || blankSegments.back().back().isEmpty() )
53 {
54 throw std::runtime_error( internalError ); // should not happen
55 }
56
57 if ( ( c == ')' || c == ',' ) && blankSegments.back().back().back().first == -1 )
58 {
59 throw std::runtime_error( "Missing number" );
60 }
61
62 if ( blankSegments.back().back().back().second != -1 )
63 {
64 throw std::runtime_error( "Too many number" );
65 }
66
67 bool ok;
68 const double number = currentNumber.toDouble( &ok );
69 if ( !ok )
70 {
71 throw std::runtime_error( u"bad formatted number '%1'"_s.arg( currentNumber ).toStdString() );
72 }
73
74 BlankSegments &segments = blankSegments.back().back();
75 if ( segments.back().first == -1 )
76 {
77 if ( segments.count() > 1 && segments.at( segments.count() - 2 ).second > number )
78 throw std::runtime_error( u"Wrong blank segments distances, start (%1) < previous end (%2)"_s.arg( number ).arg( segments.at( segments.count() - 2 ).second ).toStdString() );
79
80 blankSegments.back().back().back().first = number;
81 }
82 else if ( blankSegments.back().back().back().first > number )
83 {
84 throw std::runtime_error( u"Wrong blank segments distances, start (%1) > end (%2)"_s.arg( blankSegments.back().back().back().first ).arg( number ).toStdString() );
85 }
86 else
87 {
88 blankSegments.back().back().back().second = number;
89 }
90 currentNumber.clear();
91 };
92
93 int level = -1;
94 int iChar = 0;
95 try
96 {
97 for ( const QChar &c : strBlankSegments )
98 {
99 if ( !currentNumber.isEmpty() && ( c.isSpace() || c == ')' || c == ',' ) )
100 {
101 if ( level < 2 )
102 throw std::runtime_error( "Missing '('" );
103 addNumber( c );
104 }
105
106 if ( c == '(' )
107 {
108 if ( level >= 2 )
109 throw std::runtime_error( "Extraneous '('" );
110 appendLevel( ++level );
111 }
112 else if ( c == ')' )
113 {
114 if ( level < 0 )
115 throw std::runtime_error( "Extraneous ')'" );
116 if ( level == 2 && !blankSegments.isEmpty() && !blankSegments.back().isEmpty() && blankSegments.back().back().count() == 1 && blankSegments.back().back().back() == QPair<double, double>( -1, -1 ) )
117 {
118 blankSegments.back().back().pop_back();
119 }
120 level--;
121 }
122 else if ( c == ',' )
123 {
124 if ( ( level == 0 && blankSegments.count() == 0 )
125 || ( level == 1 && !blankSegments.isEmpty() && blankSegments.back().count() == 0 )
126 || ( level == 2 && !blankSegments.isEmpty() && !blankSegments.back().isEmpty() && blankSegments.back().back().count() == 0 ) )
127 throw std::runtime_error( "No elements, Not expecting ','" );
128
129 appendLevel( level );
130 }
131 else if ( c.isNumber() || c == '.' )
132 {
133 currentNumber.append( c );
134 }
135 else if ( !c.isSpace() )
136 {
137 throw std::runtime_error( u"Invalid character '%1'"_s.arg( c ).toStdString() );
138 }
139 iChar++;
140 }
141 if ( level != -1 )
142 {
143 throw std::runtime_error( "Missing ')'" );
144 }
145 }
146 catch ( const std::exception &e )
147 {
148 blankSegments.clear();
149 error = u"%1 (column: %2)"_s.arg( e.what() ).arg( iChar );
150 }
151
152 // convert in pixels
153 std::for_each( blankSegments.begin(), blankSegments.end(), [&renderContext, &unit]( QList<BlankSegments> &rings ) {
154 std::for_each( rings.begin(), rings.end(), [&renderContext, &unit]( BlankSegments &blankSegments ) {
155 std::for_each( blankSegments.begin(), blankSegments.end(), [&renderContext, &unit]( QPair<double, double> &blankSegment ) {
156 blankSegment.first = renderContext.convertToPainterUnits( blankSegment.first, unit );
157 blankSegment.second = renderContext.convertToPainterUnits( blankSegment.second, unit );
158 } );
159 } );
160 } );
161
162 return blankSegments;
163}
RenderUnit
Rendering size units.
Definition qgis.h:5340
QList< QPair< double, double > > BlankSegments
static QList< QList< BlankSegments > > parseBlankSegments(const QString &strBlankSegments, const QgsRenderContext &renderContext, Qgis::RenderUnit unit, QString &error)
Parse blank segments string representation strBlankSegments.
Contains information about the context of a rendering operation.
As part of the API refactoring and improvements which landed in the Processing API was substantially reworked from the x version This was done in order to allow much of the underlying Processing framework to be ported into c