Skip to content

Commit

Permalink
Merge pull request #369 from pklampros/fix_mif_mid_import
Browse files Browse the repository at this point in the history
Fix MapInfo .mif/.mid import problems
  • Loading branch information
pklampros authored Jul 31, 2020
2 parents 4e6685f + 924c1c2 commit b75df2f
Show file tree
Hide file tree
Showing 2 changed files with 86 additions and 3 deletions.
76 changes: 76 additions & 0 deletions salaTest/testmapinfodata.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,82 @@ TEST_CASE("MapInfo failing column attribute columns", "")
REQUIRE_FALSE(mapinfodata.readcolumnheaders(mifstream, columnheads));
}

TEST_CASE("MapInfo MID file with empty column", "")
{
const float EPSILON = 0.001;

// A typical MIF

std::string mifdata =
"Version 300\n" \
"Charset \"WindowsLatin1\"\n" \
"Delimiter \",\"\n" \
"Index 1,2,3,4\n" \
"CoordSys Earth Projection 8, 79, \"m\", -2, 49, 0.9996012717, 400000, -100000\n";

mifdata +=
"Columns 4\n" \
" ID Integer\n" \
" Length_m Float\n" \
" Height_m Float\n" \
" Width_m Float\n" \
"Data\n" \
"\n" \
"Line 534014.29 182533.33 535008.52 182764.11\n" \
" Pen (1,2,0)\n" \
"Line 533798.68 183094.69 534365.48 183159.01\n" \
" Pen (1,2,0)\n" \
"Point 534014.29 182533.33\n" \
" Symbol (34,0,12)";


// A MID with empty columns

std::string middata =
"1,1017.81,,\n" \
"2,568.795,,\n" \
"3,216.026,,";

ShapeMap shapeMap("MapInfoTest");
MapInfoData mapinfodata;

std::stringstream mifstream(mifdata);
std::stringstream midstream(middata);
REQUIRE(mapinfodata.import(mifstream, midstream, shapeMap) == MINFO_OK);

AttributeTable& att = shapeMap.getAttributeTable();

REQUIRE(att.getNumColumns() == 4);
REQUIRE(att.getColumn(0).getName() == "Id");
REQUIRE(att.getColumn(1).getName() == "Length_M");
REQUIRE(att.getColumn(2).getName() == "Height_M");
REQUIRE(att.getColumn(3).getName() == "Width_M");

std::map<int, SalaShape> shapes = shapeMap.getAllShapes();
auto shapeRef0 = depthmapX::getMapAtIndex(shapes, 0);
auto shapeRef1 = depthmapX::getMapAtIndex(shapes, 1);
auto shapeRef2 = depthmapX::getMapAtIndex(shapes, 2);

REQUIRE(att.getNumRows() == 3);
auto &row0 = att.getRow(AttributeKey(shapeRef0->first));
auto &row1 = att.getRow(AttributeKey(shapeRef1->first));
auto &row2 = att.getRow(AttributeKey(shapeRef2->first));

REQUIRE(row0.getValue("Id") == 1);
REQUIRE(row1.getValue("Id") == 2);
REQUIRE(row2.getValue("Id") == 3);
REQUIRE(row0.getValue("Length_M") == Approx(1017.81).epsilon(EPSILON));
REQUIRE(row1.getValue("Length_M") == Approx(568.795).epsilon(EPSILON));
REQUIRE(row2.getValue("Length_M") == Approx(216.026).epsilon(EPSILON));
REQUIRE(row0.getValue("Height_M") == -1);
REQUIRE(row1.getValue("Height_M") == -1);
REQUIRE(row2.getValue("Height_M") == -1);
REQUIRE(row0.getValue("Width_M") == -1);
REQUIRE(row1.getValue("Width_M") == -1);
REQUIRE(row2.getValue("Width_M") == -1);

}

TEST_CASE("Complete proper MapInfo file", "")
{
const float EPSILON = 0.001;
Expand Down
13 changes: 10 additions & 3 deletions salalib/parsers/mapinfodata.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -191,10 +191,17 @@ int MapInfoData::import(std::istream& miffile, std::istream& midfile, ShapeMap&
int length = (here < line.length()) ? here-first-1 : here-first;
std::string field = line.substr(first,length);
first = here;
if (reading == readable[nextreadable]) {
float val = stof(field);
if (length == 1 && field[0] == m_delimiter) {
// field is empty
row.setValue(colindexes[nextreadable], -1);
nextreadable++; // go to next column
} else if (reading == readable[nextreadable]) {
float val = -1;
if(!field.empty()) {
val = stof(field);
}
row.setValue(colindexes[nextreadable],val);
nextreadable++;
nextreadable++; // go to next column
}
reading++;
}
Expand Down

0 comments on commit b75df2f

Please sign in to comment.