EVOLUTION-MANAGER
Edit File: vfkdatablock.cpp
/****************************************************************************** * $Id: vfkdatablock.cpp 27044 2014-03-16 23:41:27Z rouault $ * * Project: VFK Reader - Data block definition * Purpose: Implements VFKDataBlock class. * Author: Martin Landa, landa.martin gmail.com * ****************************************************************************** * Copyright (c) 2009-2013, Martin Landa <landa.martin gmail.com> * Copyright (c) 2012-2013, Even Rouault <even dot rouault at mines-paris dot org> * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation * files (the "Software"), to deal in the Software without * restriction, including without limitation the rights to use, copy, * modify, merge, publish, distribute, sublicense, and/or sell copies * of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. ****************************************************************************/ #include <ctime> #include "vfkreader.h" #include "vfkreaderp.h" #include "cpl_conv.h" #include "cpl_error.h" /*! \brief VFK Data Block constructor \param pszName data block name */ IVFKDataBlock::IVFKDataBlock(const char *pszName, const IVFKReader *poReader) { m_pszName = CPLStrdup(pszName); m_nPropertyCount = 0; m_papoProperty = NULL; m_nFeatureCount = -1; /* load data on first request */ m_papoFeature = NULL; m_iNextFeature = -1; m_nGeometryType = wkbUnknown; m_bGeometry = FALSE; /* geometry is not loaded by default */ m_bGeometryPerBlock = TRUE; /* load geometry per block/feature */ m_poReader = (IVFKReader *) poReader; m_nRecordCount[RecordValid] = 0L; /* number of valid records */ m_nRecordCount[RecordSkipped] = 0L; /* number of skipped (invalid) records */ m_nRecordCount[RecordDuplicated] = 0L; /* number of duplicated records */ } /*! \brief VFKDataBlock destructor */ IVFKDataBlock::~IVFKDataBlock() { CPLFree(m_pszName); for (int i = 0; i < m_nPropertyCount; i++) { if (m_papoProperty[i]) delete m_papoProperty[i]; } CPLFree(m_papoProperty); for (int i = 0; i < m_nFeatureCount; i++) { if (m_papoFeature[i]) delete m_papoFeature[i]; } CPLFree(m_papoFeature); } /*! \brief Get property definition \param iIndex property index \return pointer to VFKPropertyDefn definition or NULL on failure */ VFKPropertyDefn *IVFKDataBlock::GetProperty(int iIndex) const { if(iIndex < 0 || iIndex >= m_nPropertyCount) return NULL; return m_papoProperty[iIndex]; } /*! \brief Set properties \param poLine pointer to line */ void IVFKDataBlock::SetProperties(const char *poLine) { const char *poChar, *poProp; char *pszName, *pszType; int nLength; pszName = pszType = NULL; /* skip data block name */ for (poChar = poLine; *poChar != '0' && *poChar != ';'; poChar++) ; if (*poChar == '\0') return; poChar++; /* read property name/type */ poProp = poChar; nLength = 0; while(*poChar != '\0') { if (*poChar == ' ') { pszName = (char *) CPLRealloc(pszName, nLength + 1); strncpy(pszName, poProp, nLength); pszName[nLength] = '\0'; poProp = ++poChar; nLength = 0; } else if (*poChar == ';') { pszType = (char *) CPLRealloc(pszType, nLength + 1); strncpy(pszType, poProp, nLength); pszType[nLength] = '\0'; /* add property */ if (pszName && *pszName != '\0' && pszType && *pszType != '\0') AddProperty(pszName, pszType); poProp = ++poChar; nLength = 0; } poChar++; nLength++; } pszType = (char *) CPLRealloc(pszType, nLength + 1); strncpy(pszType, poProp, nLength); pszType[nLength] = '\0'; /* add property */ if (pszName && *pszName != '\0' && pszType && *pszType != '\0') AddProperty(pszName, pszType); CPLFree(pszName); CPLFree(pszType); } /*! \brief Add data block property \param pszName property name \param pszType property type \return number of properties */ int IVFKDataBlock::AddProperty(const char *pszName, const char *pszType) { VFKPropertyDefn *poNewProperty = new VFKPropertyDefn(pszName, pszType, m_poReader->IsLatin2()); m_nPropertyCount++; m_papoProperty = (VFKPropertyDefn **) CPLRealloc(m_papoProperty, sizeof (VFKPropertyDefn *) * m_nPropertyCount); m_papoProperty[m_nPropertyCount-1] = poNewProperty; return m_nPropertyCount; } int IVFKDataBlock::GetFeatureCount() { if (m_nFeatureCount < 0) { m_poReader->ReadDataRecords(this); /* read VFK data records */ if (m_bGeometryPerBlock && !m_bGeometry) { LoadGeometry(); /* get real number of features */ } } return m_nFeatureCount; } /*! \brief Set number of features per data block \param nNewCount number of features \param bIncrement increment current value */ void IVFKDataBlock::SetFeatureCount(int nNewCount, bool bIncrement) { if (bIncrement) { m_nFeatureCount += nNewCount; } else { m_nFeatureCount = nNewCount; } } /*! \brief Reset reading \param iIdx force index */ void IVFKDataBlock::ResetReading(int iIdx) { if (iIdx > -1) { m_iNextFeature = iIdx; } else { m_iNextFeature = 0; } } /*! \brief Get next feature \return pointer to VFKFeature instance or NULL on error */ IVFKFeature *IVFKDataBlock::GetNextFeature() { if (m_nFeatureCount < 0) { m_poReader->ReadDataRecords(this); } if (m_bGeometryPerBlock && !m_bGeometry) { LoadGeometry(); } if (m_iNextFeature < 0) ResetReading(); if (m_iNextFeature < 0 || m_iNextFeature >= m_nFeatureCount) return NULL; return m_papoFeature[m_iNextFeature++]; } /*! \brief Get previous feature \return pointer to VFKFeature instance or NULL on error */ IVFKFeature *IVFKDataBlock::GetPreviousFeature() { if (m_nFeatureCount < 0) { m_poReader->ReadDataRecords(this); } if (m_bGeometryPerBlock && !m_bGeometry) { LoadGeometry(); } if (m_iNextFeature < 0) ResetReading(); if (m_iNextFeature < 0 || m_iNextFeature >= m_nFeatureCount) return NULL; return m_papoFeature[m_iNextFeature--]; } /*! \brief Get first feature \return pointer to VFKFeature instance or NULL on error */ IVFKFeature *IVFKDataBlock::GetFirstFeature() { if (m_nFeatureCount < 0) { m_poReader->ReadDataRecords(this); } if (m_bGeometryPerBlock && !m_bGeometry) { LoadGeometry(); } if (m_nFeatureCount < 1) return NULL; return m_papoFeature[0]; } /*! \brief Get last feature \return pointer to VFKFeature instance or NULL on error */ IVFKFeature *IVFKDataBlock::GetLastFeature() { if (m_nFeatureCount < 0) { m_poReader->ReadDataRecords(this); } if (m_bGeometryPerBlock && !m_bGeometry) { LoadGeometry(); } if (m_nFeatureCount < 1) return NULL; return m_papoFeature[m_nFeatureCount-1]; } /*! \brief Get property index by name \param pszName property name \return property index or -1 on error (property name not found) */ int IVFKDataBlock::GetPropertyIndex(const char *pszName) const { for (int i = 0; i < m_nPropertyCount; i++) if (EQUAL(pszName,m_papoProperty[i]->GetName())) return i; return -1; } /*! \brief Set geometry type (point, linestring, polygon) \return geometry type */ OGRwkbGeometryType IVFKDataBlock::SetGeometryType() { m_nGeometryType = wkbNone; /* pure attribute records */ if (EQUAL (m_pszName, "SOBR") || EQUAL (m_pszName, "OBBP") || EQUAL (m_pszName, "SPOL") || EQUAL (m_pszName, "OB") || EQUAL (m_pszName, "OP") || EQUAL (m_pszName, "OBPEJ")) m_nGeometryType = wkbPoint; else if (EQUAL (m_pszName, "SBP") || EQUAL (m_pszName, "HP") || EQUAL (m_pszName, "DPM")) m_nGeometryType = wkbLineString; else if (EQUAL (m_pszName, "PAR") || EQUAL (m_pszName, "BUD")) m_nGeometryType = wkbPolygon; return m_nGeometryType; } /*! \brief Get geometry type \return geometry type */ OGRwkbGeometryType IVFKDataBlock::GetGeometryType() const { return m_nGeometryType; } /*! \brief Get feature by index \param iIndex feature index \return pointer to feature definition or NULL on failure */ IVFKFeature *IVFKDataBlock::GetFeatureByIndex(int iIndex) const { if(iIndex < 0 || iIndex >= m_nFeatureCount) return NULL; return m_papoFeature[iIndex]; } /*! \brief Get feature by FID Modifies next feature id. \param nFID feature id \return pointer to feature definition or NULL on failure (not found) */ IVFKFeature *IVFKDataBlock::GetFeature(long nFID) { if (m_nFeatureCount < 0) { m_poReader->ReadDataRecords(this); } if (nFID < 1 || nFID > m_nFeatureCount) return NULL; if (m_bGeometryPerBlock && !m_bGeometry) { LoadGeometry(); } return GetFeatureByIndex(int (nFID) - 1); /* zero-based index */ } /*! \brief Load geometry Print warning when some invalid features are detected. \return number of invalid features or -1 on failure */ int IVFKDataBlock::LoadGeometry() { int nInvalid; #ifdef DEBUG_TIMING clock_t start, end; #endif if (m_bGeometry) return 0; nInvalid = 0; m_bGeometry = TRUE; #ifdef DEBUG_TIMING start = clock(); #endif if (m_nFeatureCount < 0) { m_poReader->ReadDataRecords(this); } if (EQUAL (m_pszName, "SOBR") || EQUAL (m_pszName, "SPOL") || EQUAL (m_pszName, "OP") || EQUAL (m_pszName, "OBPEJ") || EQUAL (m_pszName, "OB") || EQUAL (m_pszName, "OBBP")) { /* -> wkbPoint */ nInvalid = LoadGeometryPoint(); } else if (EQUAL (m_pszName, "SBP")) { /* -> wkbLineString */ nInvalid = LoadGeometryLineStringSBP(); } else if (EQUAL (m_pszName, "HP") || EQUAL (m_pszName, "DPM")) { /* -> wkbLineString */ nInvalid = LoadGeometryLineStringHP(); } else if (EQUAL (m_pszName, "PAR") || EQUAL (m_pszName, "BUD")) { /* -> wkbPolygon */ nInvalid = LoadGeometryPolygon(); } #ifdef DEBUG_TIMING end = clock(); #endif if (nInvalid > 0) { CPLError(CE_Warning, CPLE_AppDefined, "%s: %d features with invalid or empty geometry found", m_pszName, nInvalid); } #ifdef DEBUG_TIMING CPLDebug("OGR-VFK", "VFKDataBlock::LoadGeometry(): name=%s time=%ld sec", m_pszName, (long)((end - start) / CLOCKS_PER_SEC)); #endif return nInvalid; } /*! \brief Add linestring to a ring (private) \param[in,out] papoRing list of rings \param poLine pointer to linestring to be added to a ring \param bNewRing create new ring \param bBackword allow backward direction \return TRUE on success or FALSE on failure */ bool IVFKDataBlock::AppendLineToRing(PointListArray *papoRing, const OGRLineString *poLine, bool bNewRing, bool bBackward) { OGRPoint *poFirst, *poLast; OGRPoint *poFirstNew, *poLastNew; OGRPoint pt; PointList poList; /* OGRLineString -> PointList */ for (int i = 0; i < poLine->getNumPoints(); i++) { poLine->getPoint(i, &pt); poList.push_back(pt); } /* create new ring */ if (bNewRing) { papoRing->push_back(new PointList(poList)); return TRUE; } poFirstNew = &(poList.front()); poLastNew = &(poList.back()); for (PointListArray::const_iterator i = papoRing->begin(), e = papoRing->end(); i != e; ++i) { PointList *ring = (*i); poFirst = &(ring->front()); poLast = &(ring->back()); if (!poFirst || !poLast || poLine->getNumPoints() < 2) return FALSE; if (poFirstNew->Equals(poLast)) { /* forward, skip first point */ ring->insert(ring->end(), poList.begin()+1, poList.end()); return TRUE; } if (bBackward && poFirstNew->Equals(poFirst)) { /* backward, skip last point */ ring->insert(ring->begin(), poList.rbegin(), poList.rend()-1); return TRUE; } if (poLastNew->Equals(poLast)) { /* backward, skip first point */ ring->insert(ring->end(), poList.rbegin()+1, poList.rend()); return TRUE; } if (bBackward && poLastNew->Equals(poFirst)) { /* forward, skip last point */ ring->insert(ring->begin(), poList.begin(), poList.end()-1); return TRUE; } } return FALSE; } /*! \brief Set next feature \param poFeature pointer to current feature \return index of current feature or -1 on failure */ int IVFKDataBlock::SetNextFeature(const IVFKFeature *poFeature) { for (int i = 0; i < m_nFeatureCount; i++) { if (m_papoFeature[i] == poFeature) { m_iNextFeature = i + 1; return i; } } return -1; } /*! \brief Add feature \param poNewFeature pointer to VFKFeature instance */ void IVFKDataBlock::AddFeature(IVFKFeature *poNewFeature) { m_nFeatureCount++; m_papoFeature = (IVFKFeature **) CPLRealloc(m_papoFeature, sizeof (IVFKFeature *) * m_nFeatureCount); m_papoFeature[m_nFeatureCount-1] = poNewFeature; } /*! \brief Get number of records \param iRec record type (valid, skipped, duplicated) \return number of records */ int IVFKDataBlock::GetRecordCount(RecordType iRec) const { return m_nRecordCount[iRec]; } /*! \brief Increment number of records \param iRec record type (valid, skipped, duplicated) */ void IVFKDataBlock::SetIncRecordCount(RecordType iRec) { m_nRecordCount[iRec]++; } /*! \brief Get first found feature based on it's properties Note: modifies next feature. \param idx property index \param value property value \param poList list of features (NULL to loop all features) \return pointer to feature definition or NULL on failure (not found) */ VFKFeature *VFKDataBlock::GetFeature(int idx, GUIntBig value, VFKFeatureList *poList) { GUIntBig iPropertyValue; VFKFeature *poVfkFeature; if (poList) { for (VFKFeatureList::iterator i = poList->begin(), e = poList->end(); i != e; ++i) { poVfkFeature = *i; iPropertyValue = strtoul(poVfkFeature->GetProperty(idx)->GetValueS(), NULL, 0); if (iPropertyValue == value) { poList->erase(i); /* ??? */ return poVfkFeature; } } } else { for (int i = 0; i < m_nFeatureCount; i++) { poVfkFeature = (VFKFeature *) GetFeatureByIndex(i); iPropertyValue = strtoul(poVfkFeature->GetProperty(idx)->GetValueS(), NULL, 0); if (iPropertyValue == value) { m_iNextFeature = i + 1; return poVfkFeature; } } } return NULL; } /*! \brief Get features based on properties \param idx property index \param value property value \return list of features */ VFKFeatureList VFKDataBlock::GetFeatures(int idx, GUIntBig value) { GUIntBig iPropertyValue; VFKFeature *poVfkFeature; std::vector<VFKFeature *> poResult; for (int i = 0; i < m_nFeatureCount; i++) { poVfkFeature = (VFKFeature *) GetFeatureByIndex(i); iPropertyValue = strtoul(poVfkFeature->GetProperty(idx)->GetValueS(), NULL, 0); if (iPropertyValue == value) { poResult.push_back(poVfkFeature); } } return poResult; } /*! \brief Get features based on properties \param idx1 property index \param idx2 property index \param value property value \return list of features */ VFKFeatureList VFKDataBlock::GetFeatures(int idx1, int idx2, GUIntBig value) { GUIntBig iPropertyValue1, iPropertyValue2; VFKFeature *poVfkFeature; std::vector<VFKFeature *> poResult; for (int i = 0; i < m_nFeatureCount; i++) { poVfkFeature = (VFKFeature *) GetFeatureByIndex(i); iPropertyValue1 = strtoul(poVfkFeature->GetProperty(idx1)->GetValueS(), NULL, 0); if (idx2 < 0) { if (iPropertyValue1 == value) { poResult.push_back(poVfkFeature); } } else { iPropertyValue2 = strtoul(poVfkFeature->GetProperty(idx2)->GetValueS(), NULL, 0); if (iPropertyValue1 == value || iPropertyValue2 == value) { poResult.push_back(poVfkFeature); } } } return poResult; } /*! \brief Get feature count based on property value \param pszName property name \param pszValue property value \return number of features or -1 on error */ int VFKDataBlock::GetFeatureCount(const char *pszName, const char *pszValue) { int nfeatures, propIdx; VFKFeature *poVFKFeature; propIdx = GetPropertyIndex(pszName); if (propIdx < 0) return -1; nfeatures = 0; for (int i = 0; i < ((IVFKDataBlock *) this)->GetFeatureCount(); i++) { poVFKFeature = (VFKFeature *) ((IVFKDataBlock *) this)->GetFeature(i); if (!poVFKFeature) return -1; if (EQUAL (poVFKFeature->GetProperty(propIdx)->GetValueS(), pszValue)) nfeatures++; } return nfeatures; } /*! \brief Load geometry (point layers) \return number of invalid features */ int VFKDataBlock::LoadGeometryPoint() { /* -> wkbPoint */ long nInvalid; double x, y; int i_idxX, i_idxY; VFKFeature *poFeature; nInvalid = 0; i_idxY = GetPropertyIndex("SOURADNICE_Y"); i_idxX = GetPropertyIndex("SOURADNICE_X"); if (i_idxY < 0 || i_idxX < 0) { CPLError(CE_Failure, CPLE_NotSupported, "Corrupted data (%s).\n", m_pszName); return nInvalid; } for (int j = 0; j < ((IVFKDataBlock *) this)->GetFeatureCount(); j++) { poFeature = (VFKFeature *) GetFeatureByIndex(j); x = -1.0 * poFeature->GetProperty(i_idxY)->GetValueD(); y = -1.0 * poFeature->GetProperty(i_idxX)->GetValueD(); OGRPoint pt(x, y); if (!poFeature->SetGeometry(&pt)) nInvalid++; } return nInvalid; } /*! \brief Load geometry (linestring SBP layer) \return number of invalid features */ int VFKDataBlock::LoadGeometryLineStringSBP() { int idxId, idxBp_Id, idxPCB; GUIntBig id, ipcb; int nInvalid; VFKDataBlock *poDataBlockPoints; VFKFeature *poFeature, *poPoint, *poLine; OGRLineString oOGRLine; nInvalid = 0; poLine = NULL; poDataBlockPoints = (VFKDataBlock *) m_poReader->GetDataBlock("SOBR"); if (NULL == poDataBlockPoints) { CPLError(CE_Failure, CPLE_NotSupported, "Data block %s not found.\n", m_pszName); return nInvalid; } poDataBlockPoints->LoadGeometry(); idxId = poDataBlockPoints->GetPropertyIndex("ID"); idxBp_Id = GetPropertyIndex("BP_ID"); idxPCB = GetPropertyIndex("PORADOVE_CISLO_BODU"); if (idxId < 0 || idxBp_Id < 0 || idxPCB < 0) { CPLError(CE_Failure, CPLE_NotSupported, "Corrupted data (%s).\n", m_pszName); return nInvalid; } for (int j = 0; j < ((IVFKDataBlock *) this)->GetFeatureCount(); j++) { poFeature = (VFKFeature *) GetFeatureByIndex(j); CPLAssert(NULL != poFeature); poFeature->SetGeometry(NULL); id = strtoul(poFeature->GetProperty(idxBp_Id)->GetValueS(), NULL, 0); ipcb = strtoul(poFeature->GetProperty(idxPCB)->GetValueS(), NULL, 0); if (ipcb == 1) { if (!oOGRLine.IsEmpty()) { oOGRLine.setCoordinateDimension(2); /* force 2D */ if (!poLine->SetGeometry(&oOGRLine)) nInvalid++; oOGRLine.empty(); /* restore line */ } poLine = poFeature; } else { poFeature->SetGeometryType(wkbUnknown); } poPoint = poDataBlockPoints->GetFeature(idxId, id); if (!poPoint) continue; OGRPoint *pt = (OGRPoint *) poPoint->GetGeometry(); oOGRLine.addPoint(pt); } /* add last line */ oOGRLine.setCoordinateDimension(2); /* force 2D */ if (poLine) { if (!poLine->SetGeometry(&oOGRLine)) nInvalid++; } poDataBlockPoints->ResetReading(); return nInvalid; } /*! \brief Load geometry (linestring HP/DPM layer) \return number of invalid features */ int VFKDataBlock::LoadGeometryLineStringHP() { long nInvalid; int idxId, idxMy_Id, idxPCB; GUIntBig id; VFKDataBlock *poDataBlockLines; VFKFeature *poFeature, *poLine; VFKFeatureList poLineList; nInvalid = 0; poDataBlockLines = (VFKDataBlock *) m_poReader->GetDataBlock("SBP"); if (NULL == poDataBlockLines) { CPLError(CE_Failure, CPLE_NotSupported, "Data block %s not found.\n", m_pszName); return nInvalid; } poDataBlockLines->LoadGeometry(); idxId = GetPropertyIndex("ID"); if (EQUAL (m_pszName, "HP")) idxMy_Id = poDataBlockLines->GetPropertyIndex("HP_ID"); else idxMy_Id = poDataBlockLines->GetPropertyIndex("DPM_ID"); idxPCB = poDataBlockLines->GetPropertyIndex("PORADOVE_CISLO_BODU"); if (idxId < 0 || idxMy_Id < 0 || idxPCB < 0) { CPLError(CE_Failure, CPLE_NotSupported, "Corrupted data (%s).\n", m_pszName); return nInvalid; } poLineList = poDataBlockLines->GetFeatures(idxPCB, 1); /* reduce to first segment */ for (int i = 0; i < ((IVFKDataBlock *) this)->GetFeatureCount(); i++) { poFeature = (VFKFeature *) GetFeatureByIndex(i); CPLAssert(NULL != poFeature); id = strtoul(poFeature->GetProperty(idxId)->GetValueS(), NULL, 0); poLine = poDataBlockLines->GetFeature(idxMy_Id, id, &poLineList); if (!poLine || !poLine->GetGeometry()) continue; if (!poFeature->SetGeometry(poLine->GetGeometry())) nInvalid++; } poDataBlockLines->ResetReading(); return nInvalid; } /*! \brief Load geometry (polygon BUD/PAR layers) \return number of invalid features */ int VFKDataBlock::LoadGeometryPolygon() { long nInvalid; bool bIsPar, bNewRing, bFound; GUIntBig id, idOb; int nCount, nCountMax; int idxId, idxPar1, idxPar2, idxBud, idxOb, idxIdOb; VFKFeature *poFeature; VFKDataBlock *poDataBlockLines1, *poDataBlockLines2; VFKFeatureList poLineList; PointListArray poRingList; /* first is to be considered as exterior */ OGRLinearRing ogrRing; OGRPolygon ogrPolygon; idxPar1 = idxPar2 = idxBud = idxOb = idxIdOb = 0; nInvalid = 0; if (EQUAL (m_pszName, "PAR")) { poDataBlockLines1 = (VFKDataBlock *) m_poReader->GetDataBlock("HP"); poDataBlockLines2 = poDataBlockLines1; bIsPar = TRUE; } else { poDataBlockLines1 = (VFKDataBlock *) m_poReader->GetDataBlock("OB"); poDataBlockLines2 = (VFKDataBlock *) m_poReader->GetDataBlock("SBP"); bIsPar = FALSE; } if (NULL == poDataBlockLines1 || NULL == poDataBlockLines2) { CPLError(CE_Failure, CPLE_NotSupported, "Data block %s not found.\n", m_pszName); return nInvalid; } poDataBlockLines1->LoadGeometry(); poDataBlockLines2->LoadGeometry(); idxId = GetPropertyIndex("ID"); if (idxId < 0) { CPLError(CE_Failure, CPLE_NotSupported, "Corrupted data (%s).\n", m_pszName); return nInvalid; } if (bIsPar) { idxPar1 = poDataBlockLines1->GetPropertyIndex("PAR_ID_1"); idxPar2 = poDataBlockLines1->GetPropertyIndex("PAR_ID_2"); if (idxPar1 < 0 || idxPar2 < 0) { CPLError(CE_Failure, CPLE_NotSupported, "Corrupted data (%s).\n", m_pszName); return nInvalid; } } else { /* BUD */ idxIdOb = poDataBlockLines1->GetPropertyIndex("ID"); idxBud = poDataBlockLines1->GetPropertyIndex("BUD_ID"); idxOb = poDataBlockLines2->GetPropertyIndex("OB_ID"); if (idxIdOb < 0 || idxBud < 0 || idxOb < 0) { CPLError(CE_Failure, CPLE_NotSupported, "Corrupted data (%s).\n", m_pszName); return nInvalid; } } for (int i = 0; i < ((IVFKDataBlock *) this)->GetFeatureCount(); i++) { poFeature = (VFKFeature *) GetFeatureByIndex(i); CPLAssert(NULL != poFeature); id = strtoul(poFeature->GetProperty(idxId)->GetValueS(), NULL, 0); if (bIsPar) { poLineList = poDataBlockLines1->GetFeatures(idxPar1, idxPar2, id); } else { VFKFeature *poLineOb, *poLineSbp; std::vector<VFKFeature *> poLineListOb; poLineListOb = poDataBlockLines1->GetFeatures(idxBud, id); for (std::vector<VFKFeature *>::const_iterator iOb = poLineListOb.begin(), eOb = poLineListOb.end(); iOb != eOb; ++iOb) { poLineOb = (*iOb); idOb = strtoul(poLineOb->GetProperty(idxIdOb)->GetValueS(), NULL, 0); poLineSbp = poDataBlockLines2->GetFeature(idxOb, idOb); if (poLineSbp) poLineList.push_back(poLineSbp); } } if (poLineList.size() < 1) continue; /* clear */ ogrPolygon.empty(); poRingList.clear(); /* collect rings (points) */ bFound = FALSE; nCount = 0; nCountMax = poLineList.size() * 2; while (poLineList.size() > 0 && nCount < nCountMax) { bNewRing = !bFound ? TRUE : FALSE; bFound = FALSE; for (VFKFeatureList::iterator iHp = poLineList.begin(), eHp = poLineList.end(); iHp != eHp; ++iHp) { const OGRLineString *pLine = (OGRLineString *) (*iHp)->GetGeometry(); if (pLine && AppendLineToRing(&poRingList, pLine, bNewRing)) { bFound = TRUE; poLineList.erase(iHp); break; } } nCount++; } /* create rings */ for (PointListArray::const_iterator iRing = poRingList.begin(), eRing = poRingList.end(); iRing != eRing; ++iRing) { PointList *poList = *iRing; ogrRing.empty(); for (PointList::iterator iPoint = poList->begin(), ePoint = poList->end(); iPoint != ePoint; ++iPoint) { ogrRing.addPoint(&(*iPoint)); } ogrPolygon.addRing(&ogrRing); } /* set polygon */ ogrPolygon.setCoordinateDimension(2); /* force 2D */ if (!poFeature->SetGeometry(&ogrPolygon)) nInvalid++; } /* free ring list */ for (PointListArray::iterator iRing = poRingList.begin(), eRing = poRingList.end(); iRing != eRing; ++iRing) { delete (*iRing); *iRing = NULL; } poDataBlockLines1->ResetReading(); poDataBlockLines2->ResetReading(); return nInvalid; }