From 8799d7f1d72ceb891719e0bc15959e79a005be48 Mon Sep 17 00:00:00 2001 From: benjida Date: Tue, 20 Dec 2016 14:01:44 -0800 Subject: [PATCH] add scale by attribute option --- .gitignore | 2 + io_shapefile/op_import_shp.py | 1196 +++++++++++++++++---------------- 2 files changed, 630 insertions(+), 568 deletions(-) create mode 100644 .gitignore diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..f05075a8 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +.ipynb_checkpoints/ +*.ipynb \ No newline at end of file diff --git a/io_shapefile/op_import_shp.py b/io_shapefile/op_import_shp.py index db928d1f..2ccc743a 100644 --- a/io_shapefile/op_import_shp.py +++ b/io_shapefile/op_import_shp.py @@ -14,6 +14,8 @@ from ..utils.proj import Reproj from ..utils.bpu import adjust3Dview +import pdb + featureType={ 0:'Null', 1:'Point', @@ -34,591 +36,649 @@ """ dbf fields type: - C is ASCII characters - N is a double precision integer limited to around 18 characters in length - D is for dates in the YYYYMMDD format, with no spaces or hyphens between the sections - F is for floating point numbers with the same length limits as N - L is for logical data which is stored in the shapefile's attribute table as a short integer as a 1 (true) or a 0 (false). - The values it can receive are 1, 0, y, n, Y, N, T, F or the python builtins True and False + C is ASCII characters + N is a double precision integer limited to around 18 characters in length + D is for dates in the YYYYMMDD format, with no spaces or hyphens between the sections + F is for floating point numbers with the same length limits as N + L is for logical data which is stored in the shapefile's attribute table as a short integer as a 1 (true) or a 0 (false). + The values it can receive are 1, 0, y, n, Y, N, T, F or the python builtins True and False """ class IMPORT_SHP_FILE_DIALOG(Operator): - """Select shp file, loads the fields and start importgis.shapefile_props_dialog operator""" + """Select shp file, loads the fields and start importgis.shapefile_props_dialog operator""" - bl_idname = "importgis.shapefile_file_dialog" - bl_description = 'Import ESRI shapefile (.shp)' - bl_label = "Import SHP" - bl_options = {'INTERNAL'} + bl_idname = "importgis.shapefile_file_dialog" + bl_description = 'Import ESRI shapefile (.shp)' + bl_label = "Import SHP" + bl_options = {'INTERNAL'} - # Import dialog properties - filepath = StringProperty( - name="File Path", - description="Filepath used for importing the file", - maxlen=1024, - subtype='FILE_PATH' ) + # Import dialog properties + filepath = StringProperty( + name="File Path", + description="Filepath used for importing the file", + maxlen=1024, + subtype='FILE_PATH' ) - filename_ext = ".shp" + filename_ext = ".shp" - filter_glob = StringProperty( - default = "*.shp", - options = {'HIDDEN'} ) + filter_glob = StringProperty( + default = "*.shp", + options = {'HIDDEN'} ) - def invoke(self, context, event): - context.window_manager.fileselect_add(self) - return {'RUNNING_MODAL'} + def invoke(self, context, event): + context.window_manager.fileselect_add(self) + return {'RUNNING_MODAL'} - def draw(self, context): - layout = self.layout - layout.label("Options will be available") - layout.label("after selecting a file") + def draw(self, context): + layout = self.layout + layout.label("Options will be available") + layout.label("after selecting a file") - def execute(self, context): - if os.path.exists(self.filepath): - bpy.ops.importgis.shapefile_props_dialog('INVOKE_DEFAULT', filepath=self.filepath) - else: - self.report({'ERROR'}, "Invalid file") - return{'FINISHED'} + def execute(self, context): + if os.path.exists(self.filepath): + bpy.ops.importgis.shapefile_props_dialog('INVOKE_DEFAULT', filepath=self.filepath) + else: + self.report({'ERROR'}, "Invalid file") + return{'FINISHED'} class IMPORT_SHP_PROPS_DIALOG(Operator): - """Shapefile importer properties dialog""" - - bl_idname = "importgis.shapefile_props_dialog" - bl_description = 'Import ESRI shapefile (.shp)' - bl_label = "Import SHP" - bl_options = {"INTERNAL"} - - filepath = StringProperty() - - #special function to auto redraw an operator popup called through invoke_props_dialog - def check(self, context): - return True - - def listFields(self, context): - fieldsItems = [] - try: - shp = shpReader(self.filepath) - except: - self.report({'ERROR'}, "Unable to read shapefile") - return fieldsItems - fields = [field for field in shp.fields if field[0] != 'DeletionFlag'] #ignore default DeletionFlag field - for i, field in enumerate(fields): - #put each item in a tuple (key, label, tooltip) - fieldsItems.append( (field[0], field[0], '') ) - return fieldsItems - - - # Shapefile CRS definition - def listPredefCRS(self, context): - return PredefCRS.getEnumItems() - - shpCRS = EnumProperty( - name = "Shapefile CRS", - description = "Choose a Coordinate Reference System", - items = listPredefCRS ) - - # Elevation field - useFieldElev = BoolProperty( - name="Elevation from field", - description="Extract z elevation value from an attribute field", - default=False ) - fieldElevName = EnumProperty( - name = "Field", - description = "Choose field", - items = listFields ) - - #Extrusion field - useFieldExtrude = BoolProperty( - name="Extrusion from field", - description="Extract z extrusion value from an attribute field", - default=False ) - fieldExtrudeName = EnumProperty( - name = "Field", - description = "Choose field", - items = listFields ) - - #Extrusion axis - extrusionAxis = EnumProperty( - name="Extrude along", - description="Select extrusion axis", - items=[ ('Z', 'z axis', "Extrude along Z axis"), - ('NORMAL', 'Normal', "Extrude along normal")] ) - - #Create separate objects - separateObjects = BoolProperty( - name="Separate objects", - description="Import to separate objects instead one large object", - default=False ) - - #Name objects from field - useFieldName = BoolProperty( - name="Object name from field", - description="Extract name for created objects from an attribute field", - default=False ) - fieldObjName = EnumProperty( - name = "Field", - description = "Choose field", - items = listFields ) - - - def draw(self, context): - #Function used by blender to draw the panel. - scn = context.scene - layout = self.layout - - # - layout.prop(self, 'useFieldElev') - if self.useFieldElev: - layout.prop(self, 'fieldElevName') - # - layout.prop(self, 'useFieldExtrude') - if self.useFieldExtrude: - layout.prop(self, 'fieldExtrudeName') - layout.prop(self, 'extrusionAxis') - # - layout.prop(self, 'separateObjects') - if self.separateObjects: - layout.prop(self, 'useFieldName') - else: - self.useFieldName = False - if self.separateObjects and self.useFieldName: - layout.prop(self, 'fieldObjName') - # - #geoscnPrefs = context.user_preferences.addons['geoscene'].preferences - row = layout.row(align=True) - #row.prop(self, "shpCRS", text='CRS') - split = row.split(percentage=0.35, align=True) - split.label('CRS:') - split.prop(self, "shpCRS", text='') - row.operator("bgis.add_predef_crs", text='', icon='ZOOMIN') - # - geoscn = GeoScene() - if geoscn.isPartiallyGeoref: - georefManagerLayout(self, context) - - - def invoke(self, context, event): - return context.window_manager.invoke_props_dialog(self) - - def execute(self, context): - - elevField = self.fieldElevName if self.useFieldElev else "" - extrudField = self.fieldExtrudeName if self.useFieldExtrude else "" - nameField = self.fieldObjName if self.useFieldName else "" - - try: - bpy.ops.importgis.shapefile('INVOKE_DEFAULT', filepath=self.filepath, shpCRS=self.shpCRS, - fieldElevName=elevField, fieldExtrudeName=extrudField, fieldObjName=nameField, - extrusionAxis=self.extrusionAxis, separateObjects=self.separateObjects) - except Exception as e: - self.report({'ERROR'}, str(e)) - return {'FINISHED'} - - return{'FINISHED'} + """Shapefile importer properties dialog""" + + bl_idname = "importgis.shapefile_props_dialog" + bl_description = 'Import ESRI shapefile (.shp)' + bl_label = "Import SHP" + bl_options = {"INTERNAL"} + + filepath = StringProperty() + + #special function to auto redraw an operator popup called through invoke_props_dialog + def check(self, context): + return True + + def listFields(self, context): + fieldsItems = [] + try: + shp = shpReader(self.filepath) + except: + self.report({'ERROR'}, "Unable to read shapefile") + return fieldsItems + fields = [field for field in shp.fields if field[0] != 'DeletionFlag'] #ignore default DeletionFlag field + for i, field in enumerate(fields): + #put each item in a tuple (key, label, tooltip) + fieldsItems.append( (field[0], field[0], '') ) + return fieldsItems + + + # Shapefile CRS definition + def listPredefCRS(self, context): + return PredefCRS.getEnumItems() + + shpCRS = EnumProperty( + name = "Shapefile CRS", + description = "Choose a Coordinate Reference System", + items = listPredefCRS ) + + # Elevation field + useFieldElev = BoolProperty( + name="Elevation from field", + description="Extract z elevation value from an attribute field", + default=False ) + fieldElevName = EnumProperty( + name = "Field", + description = "Choose field", + items = listFields ) + + # Magnitude field ##BENJIDA + useFieldMag = BoolProperty( + name="Magnitude from field", + description="Scale symbol by certain attribute", + default=False ) + fieldMagName = EnumProperty( + name = "Field", + description = "Choose field", + items = listFields) + + #Extrusion field + useFieldExtrude = BoolProperty( + name="Extrusion from field", + description="Extract z extrusion value from an attribute field", + default=False ) + fieldExtrudeName = EnumProperty( + name = "Field", + description = "Choose field", + items = listFields ) + + #Extrusion axis + extrusionAxis = EnumProperty( + name="Extrude along", + description="Select extrusion axis", + items=[ ('Z', 'z axis', "Extrude along Z axis"), + ('NORMAL', 'Normal', "Extrude along normal")] ) + + #Create separate objects + separateObjects = BoolProperty( + name="Separate objects", + description="Import to separate objects instead one large object", + default=False ) + + #Name objects from field + useFieldName = BoolProperty( + name="Object name from field", + description="Extract name for created objects from an attribute field", + default=False ) + fieldObjName = EnumProperty( + name = "Field", + description = "Choose field", + items = listFields ) + + + def draw(self, context): + #Function used by blender to draw the panel. + scn = context.scene + layout = self.layout + + # + layout.prop(self, 'useFieldElev') + if self.useFieldElev: + layout.prop(self, 'fieldElevName') + + #BENJIDA + layout.prop(self, 'useFieldMag') + if self.useFieldMag: + layout.prop(self, 'fieldMagName') + + # + layout.prop(self, 'useFieldExtrude') + if self.useFieldExtrude: + layout.prop(self, 'fieldExtrudeName') + layout.prop(self, 'extrusionAxis') + # + layout.prop(self, 'separateObjects') + if self.separateObjects: + layout.prop(self, 'useFieldName') + else: + self.useFieldName = False + if self.separateObjects and self.useFieldName: + layout.prop(self, 'fieldObjName') + # + #geoscnPrefs = context.user_preferences.addons['geoscene'].preferences + row = layout.row(align=True) + #row.prop(self, "shpCRS", text='CRS') + split = row.split(percentage=0.35, align=True) + split.label('CRS:') + split.prop(self, "shpCRS", text='') + row.operator("bgis.add_predef_crs", text='', icon='ZOOMIN') + # + geoscn = GeoScene() + if geoscn.isPartiallyGeoref: + georefManagerLayout(self, context) + + + def invoke(self, context, event): + return context.window_manager.invoke_props_dialog(self) + + def execute(self, context): + + elevField = self.fieldElevName if self.useFieldElev else "" + #BENJIDA + magField = self.fieldMagName if self.useFieldMag else "" + extrudField = self.fieldExtrudeName if self.useFieldExtrude else "" + nameField = self.fieldObjName if self.useFieldName else "" + + try: #Benjida added fieldMagName + bpy.ops.importgis.shapefile('INVOKE_DEFAULT', filepath=self.filepath, shpCRS=self.shpCRS, + fieldElevName=elevField, fieldMagName=magField, fieldExtrudeName=extrudField, fieldObjName=nameField, + extrusionAxis=self.extrusionAxis, separateObjects=self.separateObjects) + except Exception as e: + self.report({'ERROR'}, str(e)) + return {'FINISHED'} + + return{'FINISHED'} class IMPORT_SHP(Operator): - """Import from ESRI shapefile file format (.shp)""" - - bl_idname = "importgis.shapefile" # important since its how bpy.ops.import.shapefile is constructed (allows calling operator from python console or another script) - #bl_idname rules: must contain one '.' (dot) charactere, no capital letters, no reserved words (like 'import') - bl_description = 'Import ESRI shapefile (.shp)' - bl_label = "Import SHP" - bl_options = {"UNDO"} - - filepath = StringProperty() - - shpCRS = StringProperty(name = "Shapefile CRS", description = "Coordinate Reference System") - - fieldElevName = StringProperty(name = "Field", description = "Field name") - fieldExtrudeName = StringProperty(name = "Field", description = "Field name") - fieldObjName = StringProperty(name = "Field", description = "Field name") - - #Extrusion axis - extrusionAxis = EnumProperty( - name="Extrude along", - description="Select extrusion axis", - items=[ ('Z', 'z axis', "Extrude along Z axis"), - ('NORMAL', 'Normal', "Extrude along normal")] - ) - #Create separate objects - separateObjects = BoolProperty( - name="Separate objects", - description="Import to separate objects instead one large object", - default=False - ) - - - - def execute(self, context): - - #Set cursor representation to 'loading' icon - w = context.window - w.cursor_set('WAIT') - t0 = time.clock() - - #Toogle object mode and deselect all - try: - bpy.ops.object.mode_set(mode='OBJECT') - except: - pass - - bpy.ops.object.select_all(action='DESELECT') - - #Path - shpName = os.path.basename(self.filepath)[:-4] - - #Get shp reader - print("Read shapefile...") - try: - shp = shpReader(self.filepath) - except: - self.report({'ERROR'}, "Unable to read shapefile") - return {'FINISHED'} - - #Check shape type - shpType = featureType[shp.shapeType] - print('Feature type : '+shpType) - if shpType not in ['Point','PolyLine','Polygon','PointZ','PolyLineZ','PolygonZ']: - self.report({'ERROR'}, "Cannot process multipoint, multipointZ, pointM, polylineM, polygonM and multipatch feature type") - return {'FINISHED'} - - #Get fields - fields = [field for field in shp.fields if field[0] != 'DeletionFlag'] #ignore default DeletionFlag field - fieldsNames = [field[0] for field in fields] - #print("DBF fields : "+str(fieldsNames)) - - if self.fieldElevName or (self.fieldObjName and self.separateObjects) or self.fieldExtrudeName: - self.useDbf = True - else: - self.useDbf = False - - if self.fieldObjName and self.separateObjects: - try: - nameFieldIdx = fieldsNames.index(self.fieldObjName) - except: - self.report({'ERROR'}, "Unable to find name field") - return {'FINISHED'} - - if self.fieldElevName: - try: - zFieldIdx = fieldsNames.index(self.fieldElevName) - except: - self.report({'ERROR'}, "Unable to find elevation field") - return {'FINISHED'} - - if fields[zFieldIdx][1] not in ['N', 'F', 'L'] : - self.report({'ERROR'}, "Elevation field do not contains numeric values") - return {'FINISHED'} - - if self.fieldExtrudeName: - try: - extrudeFieldIdx = fieldsNames.index(self.fieldExtrudeName) - except ValueError: - self.report({'ERROR'}, "Unable to find extrusion field") - return {'FINISHED'} - - if fields[extrudeFieldIdx][1] not in ['N', 'F', 'L'] : - self.report({'ERROR'}, "Extrusion field do not contains numeric values") - return {'FINISHED'} - - #Get shp and scene georef infos - shpCRS = self.shpCRS - geoscn = GeoScene() - if geoscn.isBroken: - self.report({'ERROR'}, "Scene georef is broken, please fix it beforehand") - return {'FINISHED'} - scale = geoscn.scale #TODO - if not geoscn.hasCRS: - try: - geoscn.crs = shpCRS - except Exception as e: - self.report({'ERROR'}, str(e)) - return {'FINISHED'} - - #Init reprojector class - if geoscn.crs != shpCRS: - print("Data will be reprojected from " + shpCRS + " to " + geoscn.crs) - try: - rprj = Reproj(shpCRS, geoscn.crs) - except Exception as e: - self.report({'ERROR'}, "Unable to reproject data. " + str(e)) - return {'FINISHED'} - if rprj.iproj == 'EPSGIO': - if shp.numRecords > 100: - self.report({'ERROR'}, "Reprojection through online epsg.io engine is limited to 100 features. \nPlease install GDAL or pyproj module.") - return {'FINISHED'} - - #Get bbox - bbox = BBOX(shp.bbox) - if geoscn.crs != shpCRS: - bbox = rprj.bbox(bbox) - - #Get or set georef dx, dy - if not geoscn.isGeoref: - dx, dy = bbox.center - geoscn.setOriginPrj(dx, dy) - else: - dx, dy = geoscn.getOriginPrj() - - #Tag if z will be extracted from shp geoms - if shpType[-1] == 'Z' and not self.fieldElevName: - self.useZGeom = True - else: - self.useZGeom = False - - #Get reader iterator (using iterator avoids loading all data in memory) - #warn, shp with zero field will return an empty shapeRecords() iterator - #to prevent this issue, iter only on shapes if there is no field required - if self.useDbf: - #Note: using shapeRecord solve the issue where number of shapes does not match number of table records - #because it iter only on features with geom and record - shpIter = shp.iterShapeRecords() - else: - shpIter = shp.iterShapes() - nbFeats = shp.numRecords - - #Create an empty BMesh - bm = bmesh.new() - #Extrusion is exponentially slow with large bmesh - #it's fastest to extrude a small bmesh and then join it to a final large bmesh - if not self.separateObjects and self.fieldExtrudeName: - finalBm = bmesh.new() - - progress = -1 - - #Main iteration over features - for i, feat in enumerate(shpIter): - - if self.useDbf: - shape = feat.shape - record = feat.record - else: - shape = feat - - #Progress infos - pourcent = round(((i+1)*100)/nbFeats) - if pourcent in list(range(0, 110, 10)) and pourcent != progress: - progress = pourcent - if pourcent == 100: - print(str(pourcent)+'%') - else: - print(str(pourcent), end="%, ") - sys.stdout.flush() #we need to flush or it won't print anything until after the loop has finished - - #Deal with multipart features - #If the shape record has multiple parts, the 'parts' attribute will contains the index of - #the first point of each part. If there is only one part then a list containing 0 is returned - if (shpType == 'PointZ' or shpType == 'Point'): #point layer has no attribute 'parts' - partsIdx = [0] - else: - try: #prevent "_shape object has no attribute parts" error - partsIdx = shape.parts - except: - partsIdx = [0] - nbParts = len(partsIdx) - - #Get list of shape's points - pts = shape.points - nbPts = len(pts) - - #Skip null geom - if nbPts == 0: - continue #go to next iteration of the loop - - #Reproj geom - if geoscn.crs != shpCRS: - pts = rprj.pts(pts) - - #Get extrusion offset - if self.fieldExtrudeName: - try: - offset = float(record[extrudeFieldIdx]) - except: - offset = 0 #null values will be set to zero - - #Iter over parts - for j in range(nbParts): - - # EXTRACT 3D GEOM - - geom = [] #will contains a list of 3d points - - #Find first and last part index - idx1 = partsIdx[j] - if j+1 == nbParts: - idx2 = nbPts - else: - idx2 = partsIdx[j+1] - - #Build 3d geom - for k, pt in enumerate(pts[idx1:idx2]): - if self.fieldElevName: - try: - z = float(record[zFieldIdx]) - except: - z = 0 #null values will be set to zero - elif self.useZGeom: - z = shape.z[idx1:idx2][k] - else: - z = 0 - geom.append((pt[0], pt[1], z)) - - #Shift coords - geom = [(pt[0]-dx, pt[1]-dy, pt[2]) for pt in geom] - - # BUILD BMESH - - # POINTS - if (shpType == 'PointZ' or shpType == 'Point'): - vert = [bm.verts.new(pt) for pt in geom] - #Extrusion - if self.fieldExtrudeName and offset > 0: - vect = (0, 0, offset) #along Z - result = bmesh.ops.extrude_vert_indiv(bm, verts=vert) - verts = result['verts'] - bmesh.ops.translate(bm, verts=verts, vec=vect) - - # LINES - if (shpType == 'PolyLine' or shpType == 'PolyLineZ'): - #Split polyline to lines - n = len(geom) - lines = [ (geom[i], geom[i+1]) for i in range(n) if i < n-1 ] - #Build edges - edges = [] - for line in lines: - verts = [bm.verts.new(pt) for pt in line] - edge = bm.edges.new(verts) - edges.append(edge) - #Extrusion - if self.fieldExtrudeName and offset > 0: - vect = (0, 0, offset) # along Z - result = bmesh.ops.extrude_edge_only(bm, edges=edges) - verts = [elem for elem in result['geom'] if isinstance(elem, bmesh.types.BMVert)] - bmesh.ops.translate(bm, verts=verts, vec=vect) - - # NGONS - if (shpType == 'Polygon' or shpType == 'PolygonZ'): - #According to the shapefile spec, polygons points are clockwise and polygon holes are counterclockwise - #in Blender face is up if points are in anticlockwise order - geom.reverse() #face up - geom.pop() #exlude last point because it's the same as first pt - if len(geom) >= 3: #needs 3 points to get a valid face - verts = [bm.verts.new(pt) for pt in geom] - face = bm.faces.new(verts) - #update normal to avoid null vector - face.normal_update() - if face.normal.z < 0: #this is a polygon hole, bmesh cannot handle polygon hole - pass #TODO - #Extrusion - if self.fieldExtrudeName and offset > 0: - #build translate vector - if self.extrusionAxis == 'NORMAL': - normal = face.normal - vect = normal * offset - elif self.extrusionAxis == 'Z': - vect = (0, 0, offset) - faces = bmesh.ops.extrude_discrete_faces(bm, faces=[face]) #return {'faces': [BMFace]} - verts = faces['faces'][0].verts - ##result = bmesh.ops.extrude_face_region(bm, geom=[face]) #return dict {"geom":[BMVert, BMEdge, BMFace]} - ##verts = [elem for elem in result['geom'] if isinstance(elem, bmesh.types.BMVert)] #geom type filter - bmesh.ops.translate(bm, verts=verts, vec=vect) - - - if self.separateObjects: - - if self.fieldObjName: - try: - name = record[nameFieldIdx] - except: - name = '' - # null values will return a bytes object containing a blank string of length equal to fields length definition - if isinstance(name, bytes): - name = '' - else: - name = str(name) - else: - name = shpName - - #Calc bmesh bbox - _bbox = BBOX.fromBmesh(bm) - - #Calc bmesh geometry origin and translate coords according to it - #then object location will be set to initial bmesh origin - #its a work around to bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY') - ox, oy, oz = _bbox.center - oz = _bbox.zmin - bmesh.ops.translate(bm, verts=bm.verts, vec=(-ox, -oy, -oz)) - - #Create new mesh from bmesh - mesh = bpy.data.meshes.new(name) - bm.to_mesh(mesh) - bm.clear() - - #Validate new mesh - mesh.validate(verbose=False) - - #Place obj - obj = bpy.data.objects.new(name, mesh) - context.scene.objects.link(obj) - context.scene.objects.active = obj - obj.select = True - obj.location = (ox, oy, oz) - - # bpy operators can be very cumbersome when scene contains lot of objects - # because it cause implicit scene updates calls - # so we must avoid using operators when created many objects with the 'separate objects' option) - ##bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY') - - #write attributes data - for i, field in enumerate(shp.fields): - fieldName = field[0] - if fieldName != 'DeletionFlag': - obj[fieldName] = record[i-1] - - - elif self.fieldExtrudeName: - #Join to final bmesh (use from_mesh method hack) - buff = bpy.data.meshes.new(".temp") - bm.to_mesh(buff) - finalBm.from_mesh(buff) - bpy.data.meshes.remove(buff) - bm.clear() - - #Write back the whole mesh - if not self.separateObjects: - - mesh = bpy.data.meshes.new(shpName) - - if self.fieldExtrudeName: - bm.free() - bm = finalBm - - bmesh.ops.remove_doubles(bm, verts=bm.verts, dist=0.0001) - bm.to_mesh(mesh) - - #Finish - #mesh.update(calc_edges=True) - mesh.validate(verbose=False) #return true if the mesh has been corrected - obj = bpy.data.objects.new(shpName, mesh) - context.scene.objects.link(obj) - context.scene.objects.active = obj - obj.select = True - bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY') - - #free the bmesh - bm.free() - - t = time.clock() - t0 - print('Build in %f seconds' % t) - - #Adjust grid size - bbox.shift(-dx, -dy) #convert shapefile bbox in 3d view space - adjust3Dview(context, bbox) - - - return {'FINISHED'} + """Import from ESRI shapefile file format (.shp)""" + + bl_idname = "importgis.shapefile" # important since its how bpy.ops.import.shapefile is constructed (allows calling operator from python console or another script) + #bl_idname rules: must contain one '.' (dot) charactere, no capital letters, no reserved words (like 'import') + bl_description = 'Import ESRI shapefile (.shp)' + bl_label = "Import SHP" + bl_options = {"UNDO"} + + filepath = StringProperty() + + shpCRS = StringProperty(name = "Shapefile CRS", description = "Coordinate Reference System") + + fieldElevName = StringProperty(name = "Field", description = "Field name") + #BENJIDA + fieldMagName = StringProperty(name = "Field", description = "Field name") + fieldExtrudeName = StringProperty(name = "Field", description = "Field name") + fieldObjName = StringProperty(name = "Field", description = "Field name") + + #Extrusion axis + extrusionAxis = EnumProperty( + name="Extrude along", + description="Select extrusion axis", + items=[ ('Z', 'z axis', "Extrude along Z axis"), + ('NORMAL', 'Normal', "Extrude along normal")] + ) + #Create separate objects + separateObjects = BoolProperty( + name="Separate objects", + description="Import to separate objects instead one large object", + default=False + ) + + + + def execute(self, context): + + #Set cursor representation to 'loading' icon + w = context.window + w.cursor_set('WAIT') + t0 = time.clock() + + #Toogle object mode and deselect all + try: + bpy.ops.object.mode_set(mode='OBJECT') + except: + pass + + bpy.ops.object.select_all(action='DESELECT') + + #Path + shpName = os.path.basename(self.filepath)[:-4] + + #Get shp reader + print("Read shapefile...") + try: + shp = shpReader(self.filepath) + except: + self.report({'ERROR'}, "Unable to read shapefile") + return {'FINISHED'} + + #Check shape type + shpType = featureType[shp.shapeType] + print('Feature type : '+shpType) + if shpType not in ['Point','PolyLine','Polygon','PointZ','PolyLineZ','PolygonZ']: + self.report({'ERROR'}, "Cannot process multipoint, multipointZ, pointM, polylineM, polygonM and multipatch feature type") + return {'FINISHED'} + + #Get fields + fields = [field for field in shp.fields if field[0] != 'DeletionFlag'] #ignore default DeletionFlag field + fieldsNames = [field[0] for field in fields] + print("DBF fields : "+str(fieldsNames)) + + #Benjida added fieldMagName + if self.fieldElevName or self.fieldMagName or (self.fieldObjName and self.separateObjects) or self.fieldExtrudeName: + self.useDbf = True + else: + self.useDbf = False + + if self.fieldObjName and self.separateObjects: + try: + nameFieldIdx = fieldsNames.index(self.fieldObjName) + except: + self.report({'ERROR'}, "Unable to find name field") + return {'FINISHED'} + + if self.fieldElevName: + try: + zFieldIdx = fieldsNames.index(self.fieldElevName) + except: + self.report({'ERROR'}, "Unable to find elevation field") + return {'FINISHED'} + + if fields[zFieldIdx][1] not in ['N', 'F', 'L'] : + self.report({'ERROR'}, "Elevation field do not contains numeric values") + return {'FINISHED'} + + #BENJIDA + if self.fieldMagName: + try: + magFieldIdx = fieldsNames.index(self.fieldMagName) + except: + self.report({'ERROR'}, "Unable to find magnitude field") + return {'FINISHED'} + + if fields[magFieldIdx][1] not in ['N', 'F', 'L'] : + self.report({'ERROR'}, "Magnitude field do not contains numeric values") + return {'FINISHED'} + + + if self.fieldExtrudeName: + try: + extrudeFieldIdx = fieldsNames.index(self.fieldExtrudeName) + except ValueError: + self.report({'ERROR'}, "Unable to find extrusion field") + return {'FINISHED'} + + if fields[extrudeFieldIdx][1] not in ['N', 'F', 'L'] : + self.report({'ERROR'}, "Extrusion field do not contains numeric values") + return {'FINISHED'} + + #Get shp and scene georef infos + shpCRS = self.shpCRS + geoscn = GeoScene() + if geoscn.isBroken: + self.report({'ERROR'}, "Scene georef is broken, please fix it beforehand") + return {'FINISHED'} + scale = geoscn.scale #TODO + if not geoscn.hasCRS: + try: + geoscn.crs = shpCRS + except Exception as e: + self.report({'ERROR'}, str(e)) + return {'FINISHED'} + + #Init reprojector class + if geoscn.crs != shpCRS: + print("Data will be reprojected from " + shpCRS + " to " + geoscn.crs) + try: + rprj = Reproj(shpCRS, geoscn.crs) + except Exception as e: + self.report({'ERROR'}, "Unable to reproject data. " + str(e)) + return {'FINISHED'} + if rprj.iproj == 'EPSGIO': + if shp.numRecords > 100: + self.report({'ERROR'}, "Reprojection through online epsg.io engine is limited to 100 features. \nPlease install GDAL or pyproj module.") + return {'FINISHED'} + + #Get bbox + bbox = BBOX(shp.bbox) + if geoscn.crs != shpCRS: + bbox = rprj.bbox(bbox) + + #Get or set georef dx, dy + if not geoscn.isGeoref: + dx, dy = bbox.center + geoscn.setOriginPrj(dx, dy) + else: + dx, dy = geoscn.getOriginPrj() + + #Tag if z will be extracted from shp geoms + if shpType[-1] == 'Z' and not self.fieldElevName: + self.useZGeom = True + else: + self.useZGeom = False + + #Get reader iterator (using iterator avoids loading all data in memory) + #warn, shp with zero field will return an empty shapeRecords() iterator + #to prevent this issue, iter only on shapes if there is no field required + if self.useDbf: + #Note: using shapeRecord solve the issue where number of shapes does not match number of table records + #because it iter only on features with geom and record + shpIter = shp.iterShapeRecords() + else: + shpIter = shp.iterShapes() + nbFeats = shp.numRecords + + #Create an empty BMesh + bm = bmesh.new() + #Extrusion is exponentially slow with large bmesh + #it's fastest to extrude a small bmesh and then join it to a final large bmesh + if not self.separateObjects and self.fieldExtrudeName: + finalBm = bmesh.new() + + progress = -1 + + #Main iteration over features + for i, feat in enumerate(shpIter): + + if self.useDbf: + shape = feat.shape + record = feat.record + else: + shape = feat + + #Progress infos + pourcent = round(((i+1)*100)/nbFeats) + if pourcent in list(range(0, 110, 10)) and pourcent != progress: + progress = pourcent + if pourcent == 100: + print(str(pourcent)+'%') + else: + print(str(pourcent), end="%, ") + sys.stdout.flush() #we need to flush or it won't print anything until after the loop has finished + + #Deal with multipart features + #If the shape record has multiple parts, the 'parts' attribute will contains the index of + #the first point of each part. If there is only one part then a list containing 0 is returned + if (shpType == 'PointZ' or shpType == 'Point'): #point layer has no attribute 'parts' + partsIdx = [0] + else: + try: #prevent "_shape object has no attribute parts" error + partsIdx = shape.parts + except: + partsIdx = [0] + nbParts = len(partsIdx) + + #Get list of shape's points + pts = shape.points + nbPts = len(pts) + + #Skip null geom + if nbPts == 0: + continue #go to next iteration of the loop + + #Reproj geom + if geoscn.crs != shpCRS: + pts = rprj.pts(pts) + + #Get extrusion offset + if self.fieldExtrudeName: + try: + offset = float(record[extrudeFieldIdx]) + except: + offset = 0 #null values will be set to zero + + #Iter over parts + for j in range(nbParts): + + # EXTRACT 3D GEOM + + geom = [] #will contains a list of 3d points + + #Find first and last part index + idx1 = partsIdx[j] + if j+1 == nbParts: + idx2 = nbPts + else: + idx2 = partsIdx[j+1] + + #Build 3d geom + for k, pt in enumerate(pts[idx1:idx2]): + if self.fieldElevName: + try: + z = float(record[zFieldIdx]) + except: + z = 0 #null values will be set to zero + elif self.useZGeom: + z = shape.z[idx1:idx2][k] + else: + z = 0 + + #BENJIDA, APPEND MAG TO GEOM ATTRIBUTE TABLE + if self.fieldMagName: + try: + mag = float(record[magFieldIdx]) + except: + mag = 0 #null values will be set to zero + elif self.useZGeom: + mag = shape.mag[idx1:idx2][k] + else: + mag = 0 + #BENJIDA added mag + geom.append((pt[0], pt[1], z, mag)) + + #Shift coords + #BENJIDA + mag = [pt[3] for pt in geom] + geom = [(pt[0]-dx, pt[1]-dy, pt[2]) for pt in geom] + # BUILD BMESH + + # POINTS + if (shpType == 'PointZ' or shpType == 'Point'): + vert = [bm.verts.new(pt) for pt in geom] + #Extrusion + if self.fieldExtrudeName and offset > 0: + vect = (0, 0, offset) #along Z + result = bmesh.ops.extrude_vert_indiv(bm, verts=vert) + verts = result['verts'] + bmesh.ops.translate(bm, verts=verts, vec=vect) + + #Magnitude #BENJIDA + if self.fieldMagName and self.fieldElevName: + loc = mathutils.Matrix.Translation(vert[0].co) + #rot = v.normal.to_track_quat('-Y','Z') #rotation of object + mat = mathutils.Matrix.Identity(3) + #mat.rotate(rot) #Applies rotation + mat = loc * mat.to_4x4() + + bmesh.ops.create_uvsphere(bm, u_segments=10, v_segments=10, diameter=mag[0]*50, matrix=mat) + + + # LINES + if (shpType == 'PolyLine' or shpType == 'PolyLineZ'): + #Split polyline to lines + n = len(geom) + lines = [ (geom[i], geom[i+1]) for i in range(n) if i < n-1 ] + #Build edges + edges = [] + for line in lines: + verts = [bm.verts.new(pt) for pt in line] + edge = bm.edges.new(verts) + edges.append(edge) + #Extrusion + if self.fieldExtrudeName and offset > 0: + vect = (0, 0, offset) # along Z + result = bmesh.ops.extrude_edge_only(bm, edges=edges) + verts = [elem for elem in result['geom'] if isinstance(elem, bmesh.types.BMVert)] + bmesh.ops.translate(bm, verts=verts, vec=vect) + + # NGONS + if (shpType == 'Polygon' or shpType == 'PolygonZ'): + #According to the shapefile spec, polygons points are clockwise and polygon holes are counterclockwise + #in Blender face is up if points are in anticlockwise order + geom.reverse() #face up + geom.pop() #exlude last point because it's the same as first pt + if len(geom) >= 3: #needs 3 points to get a valid face + verts = [bm.verts.new(pt) for pt in geom] + face = bm.faces.new(verts) + #update normal to avoid null vector + face.normal_update() + if face.normal.z < 0: #this is a polygon hole, bmesh cannot handle polygon hole + pass #TODO + #Extrusion + if self.fieldExtrudeName and offset > 0: + #build translate vector + if self.extrusionAxis == 'NORMAL': + normal = face.normal + vect = normal * offset + elif self.extrusionAxis == 'Z': + vect = (0, 0, offset) + faces = bmesh.ops.extrude_discrete_faces(bm, faces=[face]) #return {'faces': [BMFace]} + verts = faces['faces'][0].verts + ##result = bmesh.ops.extrude_face_region(bm, geom=[face]) #return dict {"geom":[BMVert, BMEdge, BMFace]} + ##verts = [elem for elem in result['geom'] if isinstance(elem, bmesh.types.BMVert)] #geom type filter + bmesh.ops.translate(bm, verts=verts, vec=vect) + + + if self.separateObjects: + + if self.fieldObjName: + try: + name = record[nameFieldIdx] + except: + name = '' + # null values will return a bytes object containing a blank string of length equal to fields length definition + if isinstance(name, bytes): + name = '' + else: + name = str(name) + else: + name = shpName + + #Calc bmesh bbox + _bbox = BBOX.fromBmesh(bm) + + #Calc bmesh geometry origin and translate coords according to it + #then object location will be set to initial bmesh origin + #its a work around to bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY') + ox, oy, oz = _bbox.center + oz = _bbox.zmin + bmesh.ops.translate(bm, verts=bm.verts, vec=(-ox, -oy, -oz)) + + #Create new mesh from bmesh + mesh = bpy.data.meshes.new(name) + bm.to_mesh(mesh) + bm.clear() + + #Validate new mesh + mesh.validate(verbose=False) + + #Place obj + obj = bpy.data.objects.new(name, mesh) + context.scene.objects.link(obj) + context.scene.objects.active = obj + obj.select = True + obj.location = (ox, oy, oz) + + # bpy operators can be very cumbersome when scene contains lot of objects + # because it cause implicit scene updates calls + # so we must avoid using operators when created many objects with the 'separate objects' option) + ##bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY') + + #write attributes data + for i, field in enumerate(shp.fields): + fieldName = field[0] + if fieldName != 'DeletionFlag': + obj[fieldName] = record[i-1] + + + elif self.fieldExtrudeName: + #Join to final bmesh (use from_mesh method hack) + buff = bpy.data.meshes.new(".temp") + bm.to_mesh(buff) + finalBm.from_mesh(buff) + bpy.data.meshes.remove(buff) + bm.clear() + + #Write back the whole mesh + if not self.separateObjects: + + mesh = bpy.data.meshes.new(shpName) + + if self.fieldExtrudeName: + bm.free() + bm = finalBm + + bmesh.ops.remove_doubles(bm, verts=bm.verts, dist=0.0001) + bm.to_mesh(mesh) + + #Finish + #mesh.update(calc_edges=True) + mesh.validate(verbose=False) #return true if the mesh has been corrected + obj = bpy.data.objects.new(shpName, mesh) + context.scene.objects.link(obj) + context.scene.objects.active = obj + obj.select = True + bpy.ops.object.origin_set(type='ORIGIN_GEOMETRY') + + #free the bmesh + bm.free() + + t = time.clock() - t0 + print('Build in %f seconds' % t) + + #Adjust grid size + bbox.shift(-dx, -dy) #convert shapefile bbox in 3d view space + adjust3Dview(context, bbox) + + + return {'FINISHED'}