diff --git a/Modules/Applications/AppMathParser/app/otbBandMath.cxx b/Modules/Applications/AppMathParser/app/otbBandMath.cxx index f8ecc1bafb051cc231cf61b14c3d2ea7f0669fb1..37a9054ec90fca6c5a9cb629d6d5db56608459c1 100644 --- a/Modules/Applications/AppMathParser/app/otbBandMath.cxx +++ b/Modules/Applications/AppMathParser/app/otbBandMath.cxx @@ -66,22 +66,34 @@ private: SetDocLongDescription( "This application performs a mathematical operation on several multi-band " - "images and outputs the result into a monoband image. Evaluation of the " + "images and outputs the result into a monoband image. The given expression" + " is computed at each pixel position. Evaluation of the " "mathematical formula is done by the muParser libraries.\n\n" - "muParser version superior to 2.0, provides the '&&' and '||' logical " - "operators, and a ternary operator 'boolean_expression ? if_true : " - "if_false'.\n\n" - - "Older versions of muParser (prior to v-2.0) provides only the 'and' and " - "'or' logical operators, and a ternary operator 'if(; ; )'.\n\n" - - "The list of features and operators is available on the muParser website [1]." + "The formula can be written using:\n\n" + " * numerical values ( 2.3, -5, 3.1e4, ...)\n" + " * variables containing pixel values (e.g. : 'im2b3' is the pixel value" + " in 2nd image, 3rd band)\n" + " * binary operators:\n\n" + " * '+' addition, '-' subtraction, '*' multiplication, '/' division\n" + " * '^' raise x to the power of y\n" + " * '<' less than, '>' greater than, '<=' less or equal, '>=' greater or equal\n" + " * '==' equal, '!=' not equal\n" +#ifdef OTB_MUPARSER_HAS_CXX_LOGICAL_OPERATORS + " * '||' logical or, '&&' logical and\n" + " * if-then-else operator: '(condition ? value_true : value_false)'\n" +#else + " * 'or' logical or, 'and' logical and\n" + " * if-then-else operator: 'if(condition;value_true;value_false)'\n" +#endif + " * functions : exp(), log(), sin(), cos(), min(), max(), ...\n\n" + + "The full list of features and operators is available on the muParser website [1]." ); SetDocLimitations( "None" ); SetDocAuthors( "OTB-Team" ); - SetDocSeeAlso("[1] http://muparser.sourceforge.net/"); + SetDocSeeAlso("[1] http://beltoforion.de/article.php?a=muparser"); AddDocTag( "Miscellaneous" ); AddParameter( ParameterType_InputImageList, "il", "Input image-list" ); @@ -101,12 +113,7 @@ private: AddParameter( ParameterType_String, "exp", "Expression"); SetParameterDescription( "exp", - "The muParser mathematical expression to apply on input images.\n" - "Use im1b1 as first band of first image, im1b2 for the second band of " - "first image.\n" - "Use im2b1 as first band of second image, im2b2 for the second band of " - "second image.\n" - "etc." + "The muParser mathematical expression to apply on input images." ); // Doc example parameter settings diff --git a/Modules/Applications/AppMathParserX/app/otbBandMathX.cxx b/Modules/Applications/AppMathParserX/app/otbBandMathX.cxx index 7c56aa82da09b946c57ecc5225144e89768fb59a..90a7f245a6f3e1adf1976c27634ec5bbb12c28fd 100644 --- a/Modules/Applications/AppMathParserX/app/otbBandMathX.cxx +++ b/Modules/Applications/AppMathParserX/app/otbBandMathX.cxx @@ -82,44 +82,30 @@ private: "Fundamentals\n" "------------\n\n" - "The i-th input image is identified by the 'im<i+1>' (e.g. 'im1') prefix " - "(please, note the indexing from 1 to N, for N inputs).\n\n" - - "The following list summarizes the available variables of input #0:\n\n" - - "im1\n" - " a pixel from 1st input, made of n components (n bands).\n\n" - - "im1b2\n" - " the 2nd component of a pixel from 1st input (band index is 1-based).\n\n" - - "im1b2N3x4\n" - " a 3x4 pixels 'N'eighbourhood of a pixel the 2nd component of a pixel " - "from the 1st input.\n\n" - - "im1PhyX\n" - " horizontal (X-axis) spacing of the 1st input.\n\n" - - "im1PhyY\n" - " vertical spacing of the 1st input input.\n\n" - - "im1b2Mean\n" - " mean of the 2nd component of the 1st input (global statistics)\n\n" - - "im1b2Mini\n" - " minimum of the 2nd component of the 1st input (global statistics)\n\n" - - "im1b2Maxi\n" - " minimum of the 2nd component of the 1st input (global statistics)\n\n" - - "im1b2Sum\n" - " minimum of the 2nd component of the 1st input (global statistics)\n\n" - - "im1b2Var\n" - " minimum of the 2nd component of the 1st input (global statistics)\n\n" - - "idxX, idxY\n" - " indices of the current pixel (generic variables)\n\n" + "The formula can be written using:\n\n" + " * numerical values ( 2.3, -5, 3.1e4, ...)\n" + " * variables containing pixel values (please, note the indexing of " + "inputs from 1 to N). Examples for the first input image:\n\n" + " * 'im1' a pixel from 1st input, made of n components (n bands)\n" + " * 'im1b2' the 2nd component of a pixel from 1st input (band index is 1-based)\n" + " * 'im1b2N3x4' a 3x4 pixels 'N'eighbourhood of a pixel the 2nd " + "component of a pixel from the 1st input\n" + " * 'im1PhyX' horizontal (X-axis) spacing of the 1st input.\n" + " * 'im1PhyY' vertical spacing of the 1st input input.\n" + " * 'im1b2Mean' mean of the 2nd component of the 1st input (global statistics)\n" + " * 'im1b2Mini' minimum of the 2nd component of the 1st input (global statistics)\n" + " * 'im1b2Maxi' maximum of the 2nd component of the 1st input (global statistics)\n" + " * 'im1b2Sum' sum of the 2nd component of the 1st input (global statistics)\n" + " * 'im1b2Var' variance of the 2nd component of the 1st input (global statistics)\n" + " * 'idxX' and 'idxY' are the indices of the current pixel (generic variables)\n" + " * binary operators:\n\n" + " * '+' addition, '-' subtraction, '*' multiplication, '/' division\n" + " * '^' raise x to the power of y\n" + " * '<' less than, '>' greater than, '<=' less or equal, '>=' greater or equal\n" + " * '==' equal, '!=' not equal\n" + " * logical operators: 'or', 'and', 'xor'\n" + " * if-then-else operator: '(condition ? value_true : value_false)'\n" + " * functions : abs(), exp(), log(), sin(), cos(), min(), max(), ...\n\n" "Always keep in mind that this application only addresses mathematically " "well-defined formulas. For instance, it is not possible to add vectors of" @@ -130,16 +116,16 @@ private: "represented as a row vector.\n\n" "Example:\n" - " im1 + im2 (1)\n" + " im1 + im2\n" " represents the addition of pixels from the 1st and 2nd inputs. This " "expression is consistent only if both inputs have the same number of " "bands.\n\n" "Please, note that it is also possible to use the following expressions" - " to obtain the same result:\n" - " im1b1 + im2b1\n" - " im1b2 + im2b2 (2)\n" - " ...\n\n" + " to obtain the same result:\n\n" + " * im1b1 + im2b1\n" + " * im1b2 + im2b2\n" + " * ...\n\n" "Nevertheless, the first expression is by far much pleaseant. We call " "this new functionality the 'batch mode' (performing the same operation " @@ -150,41 +136,41 @@ private: "Another new feature is the possibility to perform operations that " "involve neighborhoods of pixels. Variables related to such neighborhoods " - "are always defined following the imIbJNKxP pattern, where:\n" - "- I is an number identifying the image input (remember, input #0 = im1, " + "are always defined following the imIbJNKxP pattern, where:\n\n" + " - I is an number identifying the image input (remember, input #0 = im1, " "and so on)\n" - "- J is an number identifying the band (remember, first band is indexed by" + " - J is an number identifying the band (remember, first band is indexed by" "1)\n" - "- KxP are two numbers that represent the size of the neighborhood (first " + " - KxP are two numbers that represent the size of the neighborhood (first " "one is related to the horizontal direction)\n\n" "NB: All neighborhood are centered, thus K and P must be odd numbers.\n\n" - "Many operators come with this new functionality:\n" - "- dotpr\n" - "- mean\n" - "- var\n" - "- median\n" - "- min\n" - "- max\n" - "- etc.\n\n" - - "For instance, if im1 represents the pixel of 3 bands image:\n" - " im1 - mean( im1b1N5x5, im1b2N5x5, im1b3N5x5 ) (3)\n\n" + "Many operators come with this new functionality:\n\n" + " - dotpr\n" + " - mean\n" + " - var\n" + " - median\n" + " - min\n" + " - max\n" + " - etc.\n\n" + + "For instance, if im1 represents the pixel of 3 bands image::\n\n" + " im1 - mean( im1b1N5x5, im1b2N5x5, im1b3N5x5 )\n\n" "could represent a high pass filter (note that by implying three " "neighborhoods, the operator mean returns a row vector of three components" ". It is a typical behaviour for many operators of this application).\n\n" - "In addition to the previous operators, other operators are available:\n" - "- existing operators/functions from muParserX, that were not originally " + "In addition to the previous operators, other operators are available:\n\n" + " - existing operators/functions from muParserX, that were not originally " "defined for vectors and matrices (e.g. cos, sin). These new " "operators/functions keep the original names to which we added the prefix " "'v' for vector (vcos, vsin, etc.)\n" - "- mult, div and pow operators, that perform element-wise multiplication, " + " - mult, div and pow operators, that perform element-wise multiplication, " "division or exponentiation of vector/matrices (e.g. im1 div im2).\n" - "- mlt, dv and pw operators, that perform multiplication, division or " + " - mlt, dv and pw operators, that perform multiplication, division or " "exponentiation of vector/matrices by a scalar (e.g. im1 dv 2.0).\n" - "- bands, which is a very useful operator. It allows selecting specific " + " - bands, which is a very useful operator. It allows selecting specific " "bands from an image, and/or to rearrange them in a new vector (e.g." "bands( im1, { 1, 2, 1, 1 } ) produces a vector of 4 components made of " "band 1, band 2, band 1 and band 1 values from the first input.\n\n" @@ -194,30 +180,19 @@ private: "The application itself\n" "----------------------\n\n" - "The application takes the following parameters:\n" - "-il Sets the list of inputs\n" - "-ext Sets the mathematical expression (see also limitations " - "section below).\n" - "-incontext Sets the text filename containing constants values (syntax: " - "'#type name value')\n\n" - - "An example of such a file is given below:\n" + "The application can use an expression supplied with the 'exp' parameter." + " It can also use an input context file, that defines variables and " + "expressions. An example of context file is given below::\n\n" " #F expo 1.1\n" " #M kernel1 { 0.1 , 0.2 , 0.3; 0.4 , 0.5 , 0.6; 0.7 , 0.8 , 0.9; 1 , 1.1" - ", 1.2; 1.3 , 1.4 , 1.5 }\n\n" + ", 1.2; 1.3 , 1.4 , 1.5 }\n" + " #E $dotpr( kernel1, im1b1N3x5 ); im2b1^expo$\n\n" "As we can see, #I/#F allows the definition of an integer/float constant, " "whereas #M allows the definition of a vector/matrix. In the latter case, " "elements of a row must be separated by commas, and rows must be separated" - " by semicolons.\n\n" - - "It is also possible to define expressions within the same txt file, with " - "#E <expr> (see limitations, below). For instance:\n" - " #E $dotpr( kernel1, im1b1N3x5 ); im2b1^expo$\n\n" - - "-outcontext Output usesr's constants and expressions (context).\n" - "-out Sets output image (multi-outputs is not implemented yet).\n" - "\n" + " by semicolons. It is also possible to define expressions within the same" + " txt file, with #E <expr> (see limitations, below).\n" "Finally, we strongly recommend to read the OTB Cookbook which can be " "found at: http://www.orfeo-toolbox.org/packages/OTBCookBook.pdf" diff --git a/Modules/Core/Common/include/otbStringToHTML.h b/Modules/Core/Common/include/otbStringToHTML.h new file mode 100644 index 0000000000000000000000000000000000000000..3eab95fd21c4210ab3f6317deb8eaa45bf97f100 --- /dev/null +++ b/Modules/Core/Common/include/otbStringToHTML.h @@ -0,0 +1,55 @@ +/* + * Copyright (C) 2005-2017 Centre National d'Etudes Spatiales (CNES) + * + * This file is part of Orfeo Toolbox + * + * https://www.orfeo-toolbox.org/ + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifndef otbStringToHTML_h +#define otbStringToHTML_h + +#include <string> +#include <iostream> + +#include "OTBCommonExport.h" + +namespace otb +{ + +/** + * \class StringToHTML + * + * Prepare a plain string for HTML encapsulation (protection of special + * characters) + */ +class OTBCommon_EXPORT StringToHTML +{ +public: + StringToHTML(const std::string & str); + + void Print(std::ostream& os) const; + +protected: + +private: + const std::string & m_Str; +}; + +extern OTBCommon_EXPORT std::ostream & operator<< (std::ostream& os, const otb::StringToHTML& str); + +} // end of namespace otb + +#endif diff --git a/Modules/Core/Common/src/CMakeLists.txt b/Modules/Core/Common/src/CMakeLists.txt index fdfd2e9f5cae6c4362a05d721c923074d966a2ab..02ff6250e794704244acbe370fc7c562a13524c3 100644 --- a/Modules/Core/Common/src/CMakeLists.txt +++ b/Modules/Core/Common/src/CMakeLists.txt @@ -28,6 +28,7 @@ set(OTBCommon_SRC otbStandardOneLineFilterWatcher.cxx otbWriterWatcherBase.cxx otbStopwatch.cxx + otbStringToHTML.cxx ) add_library(OTBCommon ${OTBCommon_SRC}) diff --git a/Modules/Core/Common/src/otbStringToHTML.cxx b/Modules/Core/Common/src/otbStringToHTML.cxx new file mode 100644 index 0000000000000000000000000000000000000000..064ec850c2b4f4767064c9e6f2ae01f7a1927771 --- /dev/null +++ b/Modules/Core/Common/src/otbStringToHTML.cxx @@ -0,0 +1,68 @@ +/* + * Copyright (C) 2005-2017 Centre National d'Etudes Spatiales (CNES) + * + * This file is part of Orfeo Toolbox + * + * https://www.orfeo-toolbox.org/ + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include "otbStringToHTML.h" + +namespace otb +{ + +StringToHTML::StringToHTML(const std::string & str) + : m_Str(str) +{ +} + +void +StringToHTML::Print(std::ostream& os) const +{ + // TODO : add a Tex to HTML translator (maybe try TtH) + bool formulaMode=false; + for(auto&& c : m_Str) + { + if (c=='`') + { + formulaMode = !formulaMode; + } + switch (c) + { + case '<': + os.write("<",4); + break; + case '>': + os.write(">",4); + break; + case '&': + os.write("&",5); + break; + case '\n': + os.write("<br/>",5); + break; + default: + os.put(c); + } + } +} + +std::ostream & operator<< (std::ostream& os, const StringToHTML& str) +{ + str.Print(os); + return os; +} + +} // end of namespace otb diff --git a/Modules/IO/IOGDAL/src/otbGDALOverviewsBuilder.cxx b/Modules/IO/IOGDAL/src/otbGDALOverviewsBuilder.cxx index ee3d3e72e8f4f28cc8a674ab7537cd90fff07553..7b6608c87e6db4af91dffa56e6d81006fa1ad5b5 100644 --- a/Modules/IO/IOGDAL/src/otbGDALOverviewsBuilder.cxx +++ b/Modules/IO/IOGDAL/src/otbGDALOverviewsBuilder.cxx @@ -30,6 +30,7 @@ #include "otbGDALImageIO.h" #include "otbSystem.h" +#include <cctype> namespace otb { diff --git a/Modules/IO/IORAD/src/otbRADImageIO.cxx b/Modules/IO/IORAD/src/otbRADImageIO.cxx index 9a870c9503c5f72640f1294a5465a90290596292..0f7c44a62c7956e160553993e1b46827902891d8 100644 --- a/Modules/IO/IORAD/src/otbRADImageIO.cxx +++ b/Modules/IO/IORAD/src/otbRADImageIO.cxx @@ -325,31 +325,31 @@ bool RADImageIO::InternalReadHeaderInformation(const std::string& file_name, std file >> lStrCodePix; lStrCodePix = itksys::SystemTools::UpperCase(lStrCodePix); - if (lStrCodePix == "OCT") + if (lStrCodePix == "OCT" || lStrCodePix == "BYT") { m_PixelType = SCALAR; SetComponentType(UCHAR); m_BytePerPixel = 1; } - if (lStrCodePix == "PHA") + else if (lStrCodePix == "PHA") { m_PixelType = SCALAR; SetComponentType(CHAR); m_BytePerPixel = 1; } - if (lStrCodePix == "I2") + else if (lStrCodePix == "I2") { m_PixelType = SCALAR; SetComponentType(SHORT); m_BytePerPixel = 2; } - if (lStrCodePix == "I4") + else if (lStrCodePix == "I4") { m_PixelType = SCALAR; SetComponentType(INT); m_BytePerPixel = 4; } - if (lStrCodePix == "R4") + else if (lStrCodePix == "R4") { m_PixelType = SCALAR; SetComponentType(FLOAT); diff --git a/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplicationHtmlDocGenerator.cxx b/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplicationHtmlDocGenerator.cxx index 02d4c8c5feebea537006812d7610670393de62a6..3e35f46436972f645c57e0bbc2a9d014775d4e51 100644 --- a/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplicationHtmlDocGenerator.cxx +++ b/Modules/Wrappers/ApplicationEngine/src/otbWrapperApplicationHtmlDocGenerator.cxx @@ -22,6 +22,7 @@ #include <stdio.h> #include "otbWrapperChoiceParameter.h" +#include "otbStringToHTML.h" namespace otb { @@ -55,12 +56,12 @@ namespace Wrapper #define otbDocHtmlParamMacro( type, param, fullKey, showKey ) \ oss << "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><span style=\" font-family:'Courier New, courier'; font-weight:600;\"; >"; \ - oss << param->GetName(); \ + oss << otb::StringToHTML(param->GetName()); \ if( showKey == true && param->GetKey()[0] != '\0' ) \ { \ if (!fullKey.empty()) \ { \ - oss << " ("<< fullKey<< "." << param->GetKey() << ")"; \ + oss << " ("<< fullKey<< "." << param->GetKey() << ")"; \ } \ else \ { \ @@ -70,7 +71,7 @@ else \ oss << ": </span>"; \ if( param->GetDescription()[0] != '\0' ) \ { \ - oss << param->GetDescription(); \ + oss << otb::StringToHTML(param->GetDescription()); \ } \ oss << "</p>"; @@ -93,10 +94,10 @@ ApplicationHtmlDocGenerator::GenerateDoc( const Application::Pointer app, std::s oss << "p, li { white-space: pre-wrap; }"; oss << "</style></head><body style=\" font-family:'Sans Serif'; font-size:9pt; font-weight:400; font-style:normal;\">"; - otbDocHtmlTitleMacro( app->GetDocName() ); + otbDocHtmlTitleMacro( otb::StringToHTML(app->GetDocName()) ); otbDocHtmlTitle1Macro( "Brief Description" ); - otbDocHtmlBodyMacro( app->GetDescription() ); + otbDocHtmlBodyMacro( otb::StringToHTML(app->GetDescription()) ); otbDocHtmlTitle1Macro( "Tags" ); std::string tagList; @@ -107,7 +108,7 @@ ApplicationHtmlDocGenerator::GenerateDoc( const Application::Pointer app, std::s tagList.append( app->GetDocTags()[i] ).append(", "); } tagList.append( app->GetDocTags()[app->GetDocTags().size() - 1]); - otbDocHtmlBodyMacro( tagList ); + otbDocHtmlBodyMacro( otb::StringToHTML(tagList) ); } else { @@ -115,7 +116,7 @@ ApplicationHtmlDocGenerator::GenerateDoc( const Application::Pointer app, std::s } otbDocHtmlTitle1Macro("Long Description"); - otbDocHtmlBodyMacro( app->GetDocLongDescription() ); + otbDocHtmlBodyMacro( otb::StringToHTML(app->GetDocLongDescription()) ); otbDocHtmlTitle1Macro("Parameters"); oss << "<ul>"; @@ -125,13 +126,13 @@ ApplicationHtmlDocGenerator::GenerateDoc( const Application::Pointer app, std::s oss<<"</ul>"; otbDocHtmlTitle1Macro( "Limitations"); - otbDocHtmlBodyMacro( app->GetDocLimitations() ); + otbDocHtmlBodyMacro( otb::StringToHTML(app->GetDocLimitations()) ); otbDocHtmlTitle1Macro( "Authors" ); - otbDocHtmlBodyMacro( app->GetDocAuthors() ); + otbDocHtmlBodyMacro( otb::StringToHTML(app->GetDocAuthors()) ); otbDocHtmlTitle1Macro( "See also" ); - otbDocHtmlBodyMacro( app->GetDocSeeAlso() ); + otbDocHtmlBodyMacro( otb::StringToHTML(app->GetDocSeeAlso()) ); otbDocHtmlTitle1Macro( "Example of use" ); if( showKey == true ) @@ -143,7 +144,7 @@ ApplicationHtmlDocGenerator::GenerateDoc( const Application::Pointer app, std::s if( showKey == true ) { otbDocHtmlBodyMacro( "<li>Command line to execute:</li>" ); - otbDocHtmlBodyCodeMacro( app->GetCLExample() ); + otbDocHtmlBodyCodeMacro( otb::StringToHTML(app->GetCLExample()) ); oss << "</ul>"; } @@ -151,8 +152,8 @@ ApplicationHtmlDocGenerator::GenerateDoc( const Application::Pointer app, std::s val = oss.str(); -// Replace "\n" string with <br/> - itksys::SystemTools::ReplaceString( val, "\n", "<br/>"); + // Replace ":\n\n" string with ":\n" (the extra LF is needed because of rst syntax + itksys::SystemTools::ReplaceString( val, ":<br/><br/>", ":<br/>"); } void diff --git a/Modules/Wrappers/ApplicationEngine/src/otbWrapperDocExampleStructure.cxx b/Modules/Wrappers/ApplicationEngine/src/otbWrapperDocExampleStructure.cxx index d88c5b9d51c611571a6518e0497f8b80ece774f9..59c9448ffd3264ae8cd3f1acbf73b3ca0553e201 100644 --- a/Modules/Wrappers/ApplicationEngine/src/otbWrapperDocExampleStructure.cxx +++ b/Modules/Wrappers/ApplicationEngine/src/otbWrapperDocExampleStructure.cxx @@ -19,6 +19,7 @@ */ #include "otbWrapperDocExampleStructure.h" +#include "otbStringToHTML.h" namespace otb { @@ -185,7 +186,7 @@ DocExampleStructure::GenerateHtmlExample( unsigned int exId ) { oss << "<li>"; oss << "<p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\">"; - oss << it->first << ": " << it->second; + oss << it->first << ": " << otb::StringToHTML(it->second); oss << "</p>"; oss << "</li>"; } @@ -211,7 +212,7 @@ DocExampleStructure::GenerateHtmlExample() if( m_NbOfExamples>1 ) oss << "<li>"; if( !m_ExampleCommentList[exId].empty() ) - oss << m_ExampleCommentList[exId]; + oss << otb::StringToHTML(m_ExampleCommentList[exId]); oss << this->GenerateHtmlExample( exId ); if( m_NbOfExamples>1 ) oss << "</li>";