From d2abbba33301e8edab6622c72df1f799f2f071c5 Mon Sep 17 00:00:00 2001
From: Timo Koch <timo.koch@iws.uni-stuttgart.de>
Date: Mon, 13 May 2019 11:28:37 +0200
Subject: [PATCH] [vtkreader] Enable reading polylines with multiple points

---
 dumux/io/vtk/vtkreader.hh | 91 +++++++++++++++++++++++++++++++++------
 1 file changed, 79 insertions(+), 12 deletions(-)

diff --git a/dumux/io/vtk/vtkreader.hh b/dumux/io/vtk/vtkreader.hh
index f7d48e448d..8cb862863b 100644
--- a/dumux/io/vtk/vtkreader.hh
+++ b/dumux/io/vtk/vtkreader.hh
@@ -258,17 +258,17 @@ private:
             const auto connectivity = parseDataArray_<std::vector<unsigned int>>(connectivityNode);
             const auto offsets = parseDataArray_<std::vector<unsigned int>>(offsetsNode);
 
-            if (verbose) std::cout << "Found " << offsets.size() << " element." << std::endl;
+            if (verbose) std::cout << "Found " << offsets.size() << " polylines." << std::endl;
 
             unsigned int lastOffset = 0;
             for (unsigned int i = 0; i < offsets.size(); ++i)
             {
-                const auto geomType = Dune::GeometryTypes::line;
+                // a polyline can have many points in the VTK format
+                // split the line in segments with two points
                 unsigned int offset = offsets[i];
-                std::vector<unsigned int> corners; corners.resize(offset-lastOffset);
-                for (unsigned int j = 0; j < offset-lastOffset; ++j)
-                    corners[Dune::VTK::renumber(geomType, j)] = connectivity[lastOffset+j];
-                factory.insertElement(geomType, std::move(corners));
+                for (unsigned int j = 0; j < offset-lastOffset-1; ++j)
+                    factory.insertElement(Dune::GeometryTypes::line,
+                        std::vector<unsigned int>({ connectivity[lastOffset+j], connectivity[lastOffset+j+1] }));
                 lastOffset = offset;
             }
         }
@@ -293,16 +293,83 @@ private:
         const XMLElement* cellDataNode = getDataNode_(pieceNode, DataType::cellData);
         if (cellDataNode != nullptr)
         {
-            const XMLElement* dataArray = cellDataNode->FirstChildElement("DataArray");
-            for (; dataArray != nullptr; dataArray = dataArray->NextSiblingElement("DataArray"))
+            const XMLElement* cellsNode = pieceNode->FirstChildElement("Cells");
+            const XMLElement* linesNode = pieceNode->FirstChildElement("Lines");
+
+            if (cellsNode)
             {
-                const char *attributeText = dataArray->Attribute("Name");
+                const XMLElement* dataArray = cellDataNode->FirstChildElement("DataArray");
+                for (; dataArray != nullptr; dataArray = dataArray->NextSiblingElement("DataArray"))
+                {
+                    const char *attributeText = dataArray->Attribute("Name");
 
-                if (attributeText == nullptr)
-                    DUNE_THROW(Dune::IOError, "Couldn't get Name attribute of a cell data array.");
+                    if (attributeText == nullptr)
+                        DUNE_THROW(Dune::IOError, "Couldn't get Name attribute of a cell data array.");
 
-                cellData[std::string(attributeText)] = parseDataArray_<std::vector<double>>(dataArray);
+                    cellData[std::string(attributeText)] = parseDataArray_<std::vector<double>>(dataArray);
+                }
+            }
+            // for poly data
+            else if (linesNode)
+            {
+                // first parse all the cell data (each cell in this sense can be a polyline)
+                const XMLElement* dataArray = cellDataNode->FirstChildElement("DataArray");
+                if (dataArray)
+                {
+                    Data polyLineCellData;
+                    for (; dataArray != nullptr; dataArray = dataArray->NextSiblingElement("DataArray"))
+                    {
+                        const char *attributeText = dataArray->Attribute("Name");
+
+                        if (attributeText == nullptr)
+                            DUNE_THROW(Dune::IOError, "Couldn't get Name attribute of a cell data array.");
+
+                        polyLineCellData[std::string(attributeText)] = parseDataArray_<std::vector<double>>(dataArray);
+                    }
+
+                    // a polyline can have many points in the VTK format
+                    // we split the line in segments with two points
+                    // so we also need to duplicate the cell data to fit the increased line number
+                    const XMLElement* offsetsNode = findDataArray_(linesNode, "offsets");
+                    const auto offsets = parseDataArray_<std::vector<unsigned int>>(offsetsNode);
+
+                    if (offsets.size() != polyLineCellData.begin()->second.size())
+                        DUNE_THROW(Dune::IOError, "Expected the same number of cell data entries (is "
+                                                   << polyLineCellData.begin()->second.size()
+                                                   << ") as polylines (" << offsets.size() << ")!");
+
+                    // count the number of Dune cells to be able to resize the data vectors
+                    unsigned int lastOffset = 0;
+                    std::size_t numCells = 0;
+                    for (unsigned int i = 0; i < offsets.size(); ++i)
+                    {
+                        unsigned int offset = offsets[i];
+                        for (unsigned int j = 0; j < offset-lastOffset-1; ++j)
+                            ++numCells;
+                        lastOffset = offset;
+                    }
+
+                    // create the data arrays
+                    for (const auto& dArray : polyLineCellData)
+                    {
+                        cellData[dArray.first] = std::vector<double>(numCells);
+                        auto& cd = cellData[dArray.first];
+                        const auto& pd = dArray.second;
+
+                        lastOffset = 0;
+                        std::size_t cellIdx = 0;
+                        for (unsigned int i = 0; i < offsets.size(); ++i)
+                        {
+                            unsigned int offset = offsets[i];
+                            for (unsigned int j = 0; j < offset-lastOffset-1; ++j)
+                                cd[cellIdx++] = pd[i];
+                            lastOffset = offset;
+                        }
+                    }
+                }
             }
+            else
+                DUNE_THROW(Dune::IOError, "No Cells or Lines element found in " << fileName_);
         }
 
         const XMLElement* pointDataNode = getDataNode_(pieceNode, DataType::pointData);
-- 
GitLab