LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/dxf - ogr_autocad_services.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 203 213 95.3 %
Date: 2025-01-18 02:53:07 Functions: 6 6 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  DXF and DWG Translators
       4             :  * Purpose:  Implements various generic services shared between autocad related
       5             :  *           drivers.
       6             :  * Author:   Frank Warmerdam, warmerdam@pobox.com
       7             :  *
       8             :  ******************************************************************************
       9             :  * Copyright (c) 2011, Frank Warmerdam <warmerdam@pobox.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  ****************************************************************************/
      13             : 
      14             : #include "ogr_autocad_services.h"
      15             : #include "cpl_conv.h"
      16             : 
      17             : /************************************************************************/
      18             : /*                           ACTextUnescape()                           */
      19             : /*                                                                      */
      20             : /*      Unexcape DXF/DWG style escape sequences such as %%d for the     */
      21             : /*      degree sign, and do the recoding to UTF8.  Set bIsMText to      */
      22             : /*      true to translate MTEXT-specific sequences like \P and \~.      */
      23             : /************************************************************************/
      24             : 
      25         147 : CPLString ACTextUnescape(const char *pszRawInput, const char *pszEncoding,
      26             :                          bool bIsMText)
      27             : 
      28             : {
      29         294 :     CPLString osResult;
      30         294 :     CPLString osInput = pszRawInput;
      31             : 
      32             :     /* -------------------------------------------------------------------- */
      33             :     /*      Translate text from Win-1252 to UTF8.  We approximate this      */
      34             :     /*      by treating Win-1252 as Latin-1.  Note that we likely ought     */
      35             :     /*      to be consulting the $DWGCODEPAGE header variable which         */
      36             :     /*      defaults to ANSI_1252 if not set.                               */
      37             :     /* -------------------------------------------------------------------- */
      38         147 :     osInput.Recode(pszEncoding, CPL_ENC_UTF8);
      39             : 
      40         147 :     const char *pszInput = osInput.c_str();
      41             : 
      42             :     /* -------------------------------------------------------------------- */
      43             :     /*      Now translate low-level escape sequences.  They are all plain   */
      44             :     /*      ASCII characters and won't have been affected by the UTF8       */
      45             :     /*      recoding.                                                       */
      46             :     /* -------------------------------------------------------------------- */
      47        3060 :     while (*pszInput != '\0')
      48             :     {
      49        2913 :         if (pszInput[0] == '^' && pszInput[1] != '\0')
      50             :         {
      51           5 :             if (pszInput[1] == ' ')
      52           3 :                 osResult += '^';
      53             :             else
      54           4 :                 osResult += static_cast<char>(CPLToupper(
      55           2 :                                 static_cast<unsigned char>(pszInput[1]))) ^
      56           2 :                             0x40;
      57           5 :             pszInput++;
      58             :         }
      59        2908 :         else if (STARTS_WITH_CI(pszInput, "%%c") ||
      60        2908 :                  STARTS_WITH_CI(pszInput, "%%d") ||
      61        2908 :                  STARTS_WITH_CI(pszInput, "%%p"))
      62             :         {
      63             :             wchar_t anWCharString[2];
      64             : 
      65           6 :             anWCharString[1] = 0;
      66             : 
      67             :             // These are special symbol representations for AutoCAD.
      68           6 :             if (STARTS_WITH_CI(pszInput, "%%c"))
      69           0 :                 anWCharString[0] =
      70             :                     0x2300;  // diameter (0x00F8 is a good approx)
      71           6 :             else if (STARTS_WITH_CI(pszInput, "%%d"))
      72           0 :                 anWCharString[0] = 0x00B0;  // degree
      73           6 :             else if (STARTS_WITH_CI(pszInput, "%%p"))
      74           6 :                 anWCharString[0] = 0x00B1;  // plus/minus
      75             : 
      76             :             char *pszUTF8Char =
      77           6 :                 CPLRecodeFromWChar(anWCharString, CPL_ENC_UCS2, CPL_ENC_UTF8);
      78             : 
      79           6 :             osResult += pszUTF8Char;
      80           6 :             CPLFree(pszUTF8Char);
      81             : 
      82           6 :             pszInput += 2;
      83             :         }
      84        2902 :         else if (!bIsMText && (STARTS_WITH_CI(pszInput, "%%u") ||
      85         512 :                                STARTS_WITH_CI(pszInput, "%%o") ||
      86         512 :                                STARTS_WITH_CI(pszInput, "%%k")))
      87             :         {
      88             :             // Underline, overline, and strikethrough markers.
      89             :             // These have no effect in MTEXT
      90           7 :             pszInput += 2;
      91             :         }
      92             :         else
      93             :         {
      94        2895 :             osResult += pszInput[0];
      95             :         }
      96             : 
      97        2913 :         pszInput++;
      98             :     }
      99             : 
     100         147 :     if (!bIsMText)
     101          62 :         return osResult;
     102             : 
     103             :     /* -------------------------------------------------------------------- */
     104             :     /*      If this is MTEXT, or something similar (e.g. DIMENSION text),   */
     105             :     /*      do a second pass to strip additional MTEXT format codes.        */
     106             :     /* -------------------------------------------------------------------- */
     107          85 :     pszInput = osResult.c_str();
     108         170 :     CPLString osMtextResult;
     109             : 
     110        2336 :     while (*pszInput != '\0')
     111             :     {
     112        2251 :         if (pszInput[0] == '\\' && pszInput[1] == 'P')
     113             :         {
     114          19 :             osMtextResult += '\n';
     115          19 :             pszInput++;
     116             :         }
     117        2232 :         else if (pszInput[0] == '\\' && pszInput[1] == '~')
     118             :         {
     119          16 :             osMtextResult += ' ';
     120          16 :             pszInput++;
     121             :         }
     122          23 :         else if (pszInput[0] == '\\' && pszInput[1] == 'U' &&
     123        2239 :                  pszInput[2] == '+' && CPLStrnlen(pszInput, 7) >= 7)
     124             :         {
     125          10 :             CPLString osHex;
     126          10 :             unsigned int iChar = 0;
     127             : 
     128          10 :             osHex.assign(pszInput + 3, 4);
     129          10 :             sscanf(osHex.c_str(), "%x", &iChar);
     130             : 
     131             :             wchar_t anWCharString[2];
     132          10 :             anWCharString[0] = (wchar_t)iChar;
     133          10 :             anWCharString[1] = 0;
     134             : 
     135             :             char *pszUTF8Char =
     136          10 :                 CPLRecodeFromWChar(anWCharString, CPL_ENC_UCS2, CPL_ENC_UTF8);
     137             : 
     138          10 :             osMtextResult += pszUTF8Char;
     139          10 :             CPLFree(pszUTF8Char);
     140             : 
     141          10 :             pszInput += 6;
     142             :         }
     143        2206 :         else if (pszInput[0] == '{' || pszInput[0] == '}')
     144             :         {
     145             :             // Skip braces, which are used for grouping
     146             :         }
     147        2203 :         else if (pszInput[0] == '\\' &&
     148          13 :                  strchr("WTAHFfCcQp", pszInput[1]) != nullptr)
     149             :         {
     150             :             // eg. \W1.073172x;\T1.099;Bonneuil de Verrines
     151             :             // See data/dwg/EP/42002.dwg
     152             :             // These are all inline formatting codes which take an argument
     153             :             // up to the first semicolon (\W for width, \f for font, etc)
     154             : 
     155          30 :             while (*pszInput != ';' && *pszInput != '\0')
     156          24 :                 pszInput++;
     157           6 :             if (*pszInput == '\0')
     158           0 :                 break;
     159             :         }
     160        2197 :         else if (pszInput[0] == '\\' &&
     161           7 :                  strchr("KkLlOo", pszInput[1]) != nullptr)
     162             :         {
     163             :             // Inline formatting codes that don't take an argument
     164             : 
     165           3 :             pszInput++;
     166             :         }
     167        2194 :         else if (pszInput[0] == '\\' && pszInput[1] == 'S')
     168             :         {
     169             :             // Stacked text. Normal escapes don't work inside a stack
     170             : 
     171           1 :             pszInput += 2;
     172          18 :             while (*pszInput != ';' && *pszInput != '\0')
     173             :             {
     174          17 :                 if (pszInput[0] == '\\' &&
     175           4 :                     strchr("^/#~", pszInput[1]) != nullptr)
     176             :                 {
     177           4 :                     osMtextResult += pszInput[1];
     178           4 :                     pszInput++;
     179           4 :                     if (pszInput[0] == '\0')
     180           0 :                         break;
     181             :                 }
     182          13 :                 else if (strchr("^/#~", pszInput[0]) == nullptr)
     183             :                 {
     184          12 :                     osMtextResult += pszInput[0];
     185             :                 }
     186          17 :                 pszInput++;
     187             :             }
     188           1 :             if (pszInput[0] == ';')
     189           1 :                 pszInput++;
     190           1 :             if (pszInput[0] == '\0')
     191           0 :                 break;
     192             :         }
     193        2193 :         else if (pszInput[0] == '\\' && strchr("\\{}", pszInput[1]) != nullptr)
     194             :         {
     195             :             // MTEXT character escapes
     196             : 
     197           3 :             osMtextResult += pszInput[1];
     198           3 :             pszInput++;
     199           3 :             if (pszInput[0] == '\0')
     200           0 :                 break;
     201             :         }
     202             :         else
     203             :         {
     204        2190 :             osMtextResult += *pszInput;
     205             :         }
     206             : 
     207        2251 :         pszInput++;
     208             :     }
     209             : 
     210          85 :     return osMtextResult;
     211             : }
     212             : 
     213             : /************************************************************************/
     214             : /*                          ACGetColorTable()                           */
     215             : /************************************************************************/
     216             : 
     217        1260 : const unsigned char *ACGetColorTable()
     218             : 
     219             : {
     220             :     static const unsigned char abyDXFColors[768] = {
     221             :         0,   0,   0,    // 0
     222             :         255, 0,   0,    // 1
     223             :         255, 255, 0,    // 2
     224             :         0,   255, 0,    // 3
     225             :         0,   255, 255,  // 4
     226             :         0,   0,   255,  // 5
     227             :         255, 0,   255,  // 6
     228             :         0,   0,   0,    // 7 - it should be white, but that plots poorly
     229             :         127, 127, 127,  // 8
     230             :         191, 191, 191,  // 9
     231             :         255, 0,   0,    // 10
     232             :         255, 127, 127,  // 11
     233             :         165, 0,   0,    // 12
     234             :         165, 82,  82,   // 13
     235             :         127, 0,   0,    // 14
     236             :         127, 63,  63,   // 15
     237             :         76,  0,   0,    // 16
     238             :         76,  38,  38,   // 17
     239             :         38,  0,   0,    // 18
     240             :         38,  19,  19,   // 19
     241             :         255, 63,  0,    // 20
     242             :         255, 159, 127,  // 21
     243             :         165, 41,  0,    // 22
     244             :         165, 103, 82,   // 23
     245             :         127, 31,  0,    // 24
     246             :         127, 79,  63,   // 25
     247             :         76,  19,  0,    // 26
     248             :         76,  47,  38,   // 27
     249             :         38,  9,   0,    // 28
     250             :         38,  23,  19,   // 29
     251             :         255, 127, 0,    // 30
     252             :         255, 191, 127,  // 31
     253             :         165, 82,  0,    // 32
     254             :         165, 124, 82,   // 33
     255             :         127, 63,  0,    // 34
     256             :         127, 95,  63,   // 35
     257             :         76,  38,  0,    // 36
     258             :         76,  57,  38,   // 37
     259             :         38,  19,  0,    // 38
     260             :         38,  28,  19,   // 39
     261             :         255, 191, 0,    // 40
     262             :         255, 223, 127,  // 41
     263             :         165, 124, 0,    // 42
     264             :         165, 145, 82,   // 43
     265             :         127, 95,  0,    // 44
     266             :         127, 111, 63,   // 45
     267             :         76,  57,  0,    // 46
     268             :         76,  66,  38,   // 47
     269             :         38,  28,  0,    // 48
     270             :         38,  33,  19,   // 49
     271             :         255, 255, 0,    // 50
     272             :         255, 255, 127,  // 51
     273             :         165, 165, 0,    // 52
     274             :         165, 165, 82,   // 53
     275             :         127, 127, 0,    // 54
     276             :         127, 127, 63,   // 55
     277             :         76,  76,  0,    // 56
     278             :         76,  76,  38,   // 57
     279             :         38,  38,  0,    // 58
     280             :         38,  38,  19,   // 59
     281             :         191, 255, 0,    // 60
     282             :         223, 255, 127,  // 61
     283             :         124, 165, 0,    // 62
     284             :         145, 165, 82,   // 63
     285             :         95,  127, 0,    // 64
     286             :         111, 127, 63,   // 65
     287             :         57,  76,  0,    // 66
     288             :         66,  76,  38,   // 67
     289             :         28,  38,  0,    // 68
     290             :         33,  38,  19,   // 69
     291             :         127, 255, 0,    // 70
     292             :         191, 255, 127,  // 71
     293             :         82,  165, 0,    // 72
     294             :         124, 165, 82,   // 73
     295             :         63,  127, 0,    // 74
     296             :         95,  127, 63,   // 75
     297             :         38,  76,  0,    // 76
     298             :         57,  76,  38,   // 77
     299             :         19,  38,  0,    // 78
     300             :         28,  38,  19,   // 79
     301             :         63,  255, 0,    // 80
     302             :         159, 255, 127,  // 81
     303             :         41,  165, 0,    // 82
     304             :         103, 165, 82,   // 83
     305             :         31,  127, 0,    // 84
     306             :         79,  127, 63,   // 85
     307             :         19,  76,  0,    // 86
     308             :         47,  76,  38,   // 87
     309             :         9,   38,  0,    // 88
     310             :         23,  38,  19,   // 89
     311             :         0,   255, 0,    // 90
     312             :         127, 255, 127,  // 91
     313             :         0,   165, 0,    // 92
     314             :         82,  165, 82,   // 93
     315             :         0,   127, 0,    // 94
     316             :         63,  127, 63,   // 95
     317             :         0,   76,  0,    // 96
     318             :         38,  76,  38,   // 97
     319             :         0,   38,  0,    // 98
     320             :         19,  38,  19,   // 99
     321             :         0,   255, 63,   // 100
     322             :         127, 255, 159,  // 101
     323             :         0,   165, 41,   // 102
     324             :         82,  165, 103,  // 103
     325             :         0,   127, 31,   // 104
     326             :         63,  127, 79,   // 105
     327             :         0,   76,  19,   // 106
     328             :         38,  76,  47,   // 107
     329             :         0,   38,  9,    // 108
     330             :         19,  38,  23,   // 109
     331             :         0,   255, 127,  // 110
     332             :         127, 255, 191,  // 111
     333             :         0,   165, 82,   // 112
     334             :         82,  165, 124,  // 113
     335             :         0,   127, 63,   // 114
     336             :         63,  127, 95,   // 115
     337             :         0,   76,  38,   // 116
     338             :         38,  76,  57,   // 117
     339             :         0,   38,  19,   // 118
     340             :         19,  38,  28,   // 119
     341             :         0,   255, 191,  // 120
     342             :         127, 255, 223,  // 121
     343             :         0,   165, 124,  // 122
     344             :         82,  165, 145,  // 123
     345             :         0,   127, 95,   // 124
     346             :         63,  127, 111,  // 125
     347             :         0,   76,  57,   // 126
     348             :         38,  76,  66,   // 127
     349             :         0,   38,  28,   // 128
     350             :         19,  38,  33,   // 129
     351             :         0,   255, 255,  // 130
     352             :         127, 255, 255,  // 131
     353             :         0,   165, 165,  // 132
     354             :         82,  165, 165,  // 133
     355             :         0,   127, 127,  // 134
     356             :         63,  127, 127,  // 135
     357             :         0,   76,  76,   // 136
     358             :         38,  76,  76,   // 137
     359             :         0,   38,  38,   // 138
     360             :         19,  38,  38,   // 139
     361             :         0,   191, 255,  // 140
     362             :         127, 223, 255,  // 141
     363             :         0,   124, 165,  // 142
     364             :         82,  145, 165,  // 143
     365             :         0,   95,  127,  // 144
     366             :         63,  111, 127,  // 145
     367             :         0,   57,  76,   // 146
     368             :         38,  66,  76,   // 147
     369             :         0,   28,  38,   // 148
     370             :         19,  33,  38,   // 149
     371             :         0,   127, 255,  // 150
     372             :         127, 191, 255,  // 151
     373             :         0,   82,  165,  // 152
     374             :         82,  124, 165,  // 153
     375             :         0,   63,  127,  // 154
     376             :         63,  95,  127,  // 155
     377             :         0,   38,  76,   // 156
     378             :         38,  57,  76,   // 157
     379             :         0,   19,  38,   // 158
     380             :         19,  28,  38,   // 159
     381             :         0,   63,  255,  // 160
     382             :         127, 159, 255,  // 161
     383             :         0,   41,  165,  // 162
     384             :         82,  103, 165,  // 163
     385             :         0,   31,  127,  // 164
     386             :         63,  79,  127,  // 165
     387             :         0,   19,  76,   // 166
     388             :         38,  47,  76,   // 167
     389             :         0,   9,   38,   // 168
     390             :         19,  23,  38,   // 169
     391             :         0,   0,   255,  // 170
     392             :         127, 127, 255,  // 171
     393             :         0,   0,   165,  // 172
     394             :         82,  82,  165,  // 173
     395             :         0,   0,   127,  // 174
     396             :         63,  63,  127,  // 175
     397             :         0,   0,   76,   // 176
     398             :         38,  38,  76,   // 177
     399             :         0,   0,   38,   // 178
     400             :         19,  19,  38,   // 179
     401             :         63,  0,   255,  // 180
     402             :         159, 127, 255,  // 181
     403             :         41,  0,   165,  // 182
     404             :         103, 82,  165,  // 183
     405             :         31,  0,   127,  // 184
     406             :         79,  63,  127,  // 185
     407             :         19,  0,   76,   // 186
     408             :         47,  38,  76,   // 187
     409             :         9,   0,   38,   // 188
     410             :         23,  19,  38,   // 189
     411             :         127, 0,   255,  // 190
     412             :         191, 127, 255,  // 191
     413             :         82,  0,   165,  // 192
     414             :         124, 82,  165,  // 193
     415             :         63,  0,   127,  // 194
     416             :         95,  63,  127,  // 195
     417             :         38,  0,   76,   // 196
     418             :         57,  38,  76,   // 197
     419             :         19,  0,   38,   // 198
     420             :         28,  19,  38,   // 199
     421             :         191, 0,   255,  // 200
     422             :         223, 127, 255,  // 201
     423             :         124, 0,   165,  // 202
     424             :         145, 82,  165,  // 203
     425             :         95,  0,   127,  // 204
     426             :         111, 63,  127,  // 205
     427             :         57,  0,   76,   // 206
     428             :         66,  38,  76,   // 207
     429             :         28,  0,   38,   // 208
     430             :         33,  19,  38,   // 209
     431             :         255, 0,   255,  // 210
     432             :         255, 127, 255,  // 211
     433             :         165, 0,   165,  // 212
     434             :         165, 82,  165,  // 213
     435             :         127, 0,   127,  // 214
     436             :         127, 63,  127,  // 215
     437             :         76,  0,   76,   // 216
     438             :         76,  38,  76,   // 217
     439             :         38,  0,   38,   // 218
     440             :         38,  19,  38,   // 219
     441             :         255, 0,   191,  // 220
     442             :         255, 127, 223,  // 221
     443             :         165, 0,   124,  // 222
     444             :         165, 82,  145,  // 223
     445             :         127, 0,   95,   // 224
     446             :         127, 63,  111,  // 225
     447             :         76,  0,   57,   // 226
     448             :         76,  38,  66,   // 227
     449             :         38,  0,   28,   // 228
     450             :         38,  19,  33,   // 229
     451             :         255, 0,   127,  // 230
     452             :         255, 127, 191,  // 231
     453             :         165, 0,   82,   // 232
     454             :         165, 82,  124,  // 233
     455             :         127, 0,   63,   // 234
     456             :         127, 63,  95,   // 235
     457             :         76,  0,   38,   // 236
     458             :         76,  38,  57,   // 237
     459             :         38,  0,   19,   // 238
     460             :         38,  19,  28,   // 239
     461             :         255, 0,   63,   // 240
     462             :         255, 127, 159,  // 241
     463             :         165, 0,   41,   // 242
     464             :         165, 82,  103,  // 243
     465             :         127, 0,   31,   // 244
     466             :         127, 63,  79,   // 245
     467             :         76,  0,   19,   // 246
     468             :         76,  38,  47,   // 247
     469             :         38,  0,   9,    // 248
     470             :         38,  19,  23,   // 249
     471             :         84,  84,  84,   // 250
     472             :         118, 118, 118,  // 251
     473             :         152, 152, 152,  // 252
     474             :         186, 186, 186,  // 253
     475             :         220, 220, 220,  // 254
     476             :         255, 255, 255   // 255
     477             :     };
     478             : 
     479        1260 :     return abyDXFColors;
     480             : }
     481             : 
     482             : /************************************************************************/
     483             : /*                       ACGetKnownDimStyleCodes()                      */
     484             : /*                                                                      */
     485             : /*      Gets a list of the DIMSTYLE codes that we care about. Array     */
     486             : /*      terminates with a zero value.                                   */
     487             : /************************************************************************/
     488             : 
     489         216 : const int *ACGetKnownDimStyleCodes()
     490             : {
     491             :     static const int aiKnownCodes[] = {40,  41,  42,  44,  75,  76,  77,
     492             :                                        140, 147, 176, 178, 271, 341, 0};
     493             : 
     494         216 :     return aiKnownCodes;
     495             : }
     496             : 
     497             : /************************************************************************/
     498             : /*                      ACGetDimStylePropertyName()                     */
     499             : /************************************************************************/
     500             : 
     501        4397 : const char *ACGetDimStylePropertyName(const int iDimStyleCode)
     502             : 
     503             : {
     504             :     // We are only interested in properties required by the DIMENSION
     505             :     // and LEADER code. Return NULL for other properties.
     506        4397 :     switch (iDimStyleCode)
     507             :     {
     508         241 :         case 40:
     509         241 :             return "DIMSCALE";
     510         256 :         case 41:
     511         256 :             return "DIMASZ";
     512         237 :         case 42:
     513         237 :             return "DIMEXO";
     514         237 :         case 44:
     515         237 :             return "DIMEXE";
     516         222 :         case 75:
     517         222 :             return "DIMSE1";
     518         222 :         case 76:
     519         222 :             return "DIMSE2";
     520         239 :         case 77:
     521         239 :             return "DIMTAD";
     522         250 :         case 140:
     523         250 :             return "DIMTXT";
     524         235 :         case 147:
     525         235 :             return "DIMGAP";
     526         238 :         case 176:
     527         238 :             return "DIMCLRD";
     528         222 :         case 178:
     529         222 :             return "DIMCLRT";
     530         244 :         case 271:
     531         244 :             return "DIMDEC";
     532         224 :         case 341:
     533         224 :             return "DIMLDRBLK";
     534        1330 :         default:
     535        1330 :             return nullptr;
     536             :     }
     537             : }
     538             : 
     539             : /************************************************************************/
     540             : /*                    ACGetDimStylePropertyDefault()                    */
     541             : /************************************************************************/
     542             : 
     543        2808 : const char *ACGetDimStylePropertyDefault(const int iDimStyleCode)
     544             : 
     545             : {
     546             :     // We are only interested in properties required by the DIMENSION
     547             :     // and LEADER code. Return "0" for other, unknown properties.
     548             :     // These defaults were obtained from the Express\defaults.scr file
     549             :     // in an AutoCAD installation.
     550        2808 :     switch (iDimStyleCode)
     551             :     {
     552         216 :         case 40:
     553         216 :             return "1.0";
     554         216 :         case 41:
     555         216 :             return "0.18";
     556         216 :         case 42:
     557         216 :             return "0.0625";
     558         216 :         case 44:
     559         216 :             return "0.18";
     560         216 :         case 75:
     561         216 :             return "0";
     562         216 :         case 76:
     563         216 :             return "0";
     564         216 :         case 77:
     565         216 :             return "0";
     566         216 :         case 140:
     567         216 :             return "0.18";
     568         216 :         case 147:
     569         216 :             return "0.09";
     570         216 :         case 176:
     571         216 :             return "0";
     572         216 :         case 178:
     573         216 :             return "0";
     574         216 :         case 271:
     575         216 :             return "4";
     576         216 :         case 341:
     577         216 :             return "";
     578           0 :         default:
     579           0 :             return "0";
     580             :     }
     581             : }
     582             : 
     583             : /************************************************************************/
     584             : /*                            ACAdjustText()                            */
     585             : /*                                                                      */
     586             : /*      Rotate and scale text features by the designated amount by      */
     587             : /*      adjusting the style string.                                     */
     588             : /************************************************************************/
     589             : 
     590         568 : void ACAdjustText(const double dfAngle, const double dfScaleX,
     591             :                   const double dfScaleY, OGRFeature *const poFeature)
     592             : 
     593             : {
     594             :     /* -------------------------------------------------------------------- */
     595             :     /*      We only try to alter text elements (LABEL styles).              */
     596             :     /* -------------------------------------------------------------------- */
     597         568 :     if (poFeature->GetStyleString() == nullptr)
     598         504 :         return;
     599             : 
     600         568 :     CPLString osOldStyle = poFeature->GetStyleString();
     601             : 
     602         568 :     if (!STARTS_WITH(osOldStyle, "LABEL("))
     603         504 :         return;
     604             : 
     605             :     // Split the style string up into its parts
     606          64 :     osOldStyle.erase(0, 6);
     607          64 :     osOldStyle.erase(osOldStyle.size() - 1);
     608          64 :     char **papszTokens = CSLTokenizeString2(
     609             :         osOldStyle, ",",
     610             :         CSLT_HONOURSTRINGS | CSLT_PRESERVEQUOTES | CSLT_PRESERVEESCAPES);
     611             : 
     612             :     /* -------------------------------------------------------------------- */
     613             :     /*      Update the text angle.                                          */
     614             :     /* -------------------------------------------------------------------- */
     615             :     char szBuffer[64];
     616             : 
     617          64 :     if (dfAngle != 0.0)
     618             :     {
     619           4 :         double dfOldAngle = 0.0;
     620             : 
     621           4 :         const char *pszAngle = CSLFetchNameValue(papszTokens, "a");
     622           4 :         if (pszAngle)
     623           0 :             dfOldAngle = CPLAtof(pszAngle);
     624             : 
     625           4 :         CPLsnprintf(szBuffer, sizeof(szBuffer), "%.3g", dfOldAngle + dfAngle);
     626           4 :         papszTokens = CSLSetNameValue(papszTokens, "a", szBuffer);
     627             :     }
     628             : 
     629             :     /* -------------------------------------------------------------------- */
     630             :     /*      Update the text width and height.                               */
     631             :     /* -------------------------------------------------------------------- */
     632             : 
     633          64 :     if (dfScaleY != 1.0)
     634             :     {
     635           4 :         const char *pszHeight = CSLFetchNameValue(papszTokens, "s");
     636           4 :         if (pszHeight)
     637             :         {
     638           4 :             const double dfOldHeight = CPLAtof(pszHeight);
     639             : 
     640           4 :             CPLsnprintf(szBuffer, sizeof(szBuffer), "%.3gg",
     641             :                         dfOldHeight * dfScaleY);
     642           4 :             papszTokens = CSLSetNameValue(papszTokens, "s", szBuffer);
     643             :         }
     644             :     }
     645             : 
     646          64 :     if (dfScaleX != dfScaleY && dfScaleY != 0.0)
     647             :     {
     648           4 :         const double dfWidthFactor = dfScaleX / dfScaleY;
     649           4 :         double dfOldWidth = 100.0;
     650             : 
     651           4 :         const char *pszWidth = CSLFetchNameValue(papszTokens, "w");
     652           4 :         if (pszWidth)
     653           3 :             dfOldWidth = CPLAtof(pszWidth);
     654             : 
     655           4 :         CPLsnprintf(szBuffer, sizeof(szBuffer), "%.4g",
     656             :                     dfOldWidth * dfWidthFactor);
     657           4 :         papszTokens = CSLSetNameValue(papszTokens, "w", szBuffer);
     658             :     }
     659             : 
     660             :     /* -------------------------------------------------------------------- */
     661             :     /*      Update the text offsets.                                        */
     662             :     /* -------------------------------------------------------------------- */
     663             : 
     664          64 :     if (dfScaleX != 1.0 || dfScaleY != 1.0 || dfAngle != 0.0)
     665             :     {
     666           6 :         double dfOldDx = 0.0;
     667           6 :         double dfOldDy = 0.0;
     668             : 
     669           6 :         const char *pszDx = CSLFetchNameValue(papszTokens, "dx");
     670           6 :         if (pszDx)
     671           2 :             dfOldDx = CPLAtof(pszDx);
     672           6 :         const char *pszDy = CSLFetchNameValue(papszTokens, "dy");
     673           6 :         if (pszDy)
     674           0 :             dfOldDy = CPLAtof(pszDy);
     675             : 
     676           6 :         if (dfOldDx != 0.0 || dfOldDy != 0.0)
     677             :         {
     678           2 :             const double dfAngleRadians = dfAngle * M_PI / 180.0;
     679             : 
     680           2 :             CPLsnprintf(szBuffer, sizeof(szBuffer), "%.6gg",
     681           2 :                         dfScaleX * dfOldDx * cos(dfAngleRadians) +
     682           2 :                             dfScaleY * dfOldDy * -sin(dfAngleRadians));
     683           2 :             papszTokens = CSLSetNameValue(papszTokens, "dx", szBuffer);
     684             : 
     685           2 :             CPLsnprintf(szBuffer, sizeof(szBuffer), "%.6gg",
     686           2 :                         dfScaleX * dfOldDx * sin(dfAngleRadians) +
     687           2 :                             dfScaleY * dfOldDy * cos(dfAngleRadians));
     688           2 :             papszTokens = CSLSetNameValue(papszTokens, "dy", szBuffer);
     689             :         }
     690             :     }
     691             : 
     692          64 :     CSLSetNameValueSeparator(papszTokens, ":");
     693             : 
     694         128 :     CPLString osNewStyle = "LABEL(";
     695          64 :     int iIndex = 0;
     696         497 :     while (papszTokens[iIndex])
     697             :     {
     698         433 :         if (iIndex > 0)
     699         369 :             osNewStyle += ",";
     700         433 :         osNewStyle += papszTokens[iIndex++];
     701             :     }
     702          64 :     osNewStyle += ")";
     703             : 
     704          64 :     poFeature->SetStyleString(osNewStyle);
     705             : 
     706          64 :     CSLDestroy(papszTokens);
     707             : }

Generated by: LCOV version 1.14