ExportRasterByMask.py 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. """
  2. ***************************************************************************
  3. ExportRasterByMask.py
  4. ---------------------
  5. Date : September 2013
  6. Copyright : (C) 2013 by Alexander Bruy
  7. Email : alexander bruy at gmail dot com
  8. ***************************************************************************
  9. * *
  10. * This program is free software; you can redistribute it and/or modify *
  11. * it under the terms of the GNU General Public License as published by *
  12. * the Free Software Foundation; either version 2 of the License, or *
  13. * (at your option) any later version. *
  14. * *
  15. ***************************************************************************
  16. """
  17. __author__ = 'Alexander Bruy'
  18. __date__ = 'September 2013'
  19. __copyright__ = '(C) 2013, Alexander Bruy'
  20. import os
  21. from qgis.PyQt.QtGui import QIcon
  22. from qgis.core import (QgsRasterFileWriter,
  23. QgsProcessing,
  24. QgsProcessingException,
  25. QgsProcessingParameterDefinition,
  26. QgsProcessingParameterFeatureSource,
  27. QgsProcessingParameterRasterLayer,
  28. QgsProcessingParameterCrs,
  29. QgsProcessingParameterEnum,
  30. QgsProcessingParameterExtent,
  31. QgsProcessingParameterString,
  32. QgsProcessingParameterNumber,
  33. QgsProcessingParameterBoolean,
  34. QgsProcessingParameterRasterDestination)
  35. from processing.algs.gdal.GdalAlgorithm import GdalAlgorithm
  36. from processing.algs.gdal.GdalUtils import GdalUtils
  37. from processing.tools.StringUtils import (get_download_dir, dict_to_str_with_equals, sourceToDBConfig, getUUID, get_temp_dir)
  38. pluginPath = os.path.split(os.path.split(os.path.dirname(__file__))[0])[0]
  39. class ExportRasterByMask(GdalAlgorithm):
  40. INPUT = 'INPUT'
  41. MASK = 'MASK'
  42. SOURCE_CRS = 'SOURCE_CRS'
  43. TARGET_CRS = 'TARGET_CRS'
  44. EXTENT = 'TARGET_EXTENT'
  45. NODATA = 'NODATA'
  46. ALPHA_BAND = 'ALPHA_BAND'
  47. CROP_TO_CUTLINE = 'CROP_TO_CUTLINE'
  48. KEEP_RESOLUTION = 'KEEP_RESOLUTION'
  49. SET_RESOLUTION = 'SET_RESOLUTION'
  50. X_RESOLUTION = 'X_RESOLUTION'
  51. Y_RESOLUTION = 'Y_RESOLUTION'
  52. OPTIONS = 'OPTIONS'
  53. DATA_TYPE = 'DATA_TYPE'
  54. MULTITHREADING = 'MULTITHREADING'
  55. EXTRA = 'EXTRA'
  56. OUTPUT = 'OUTPUT'
  57. def __init__(self):
  58. super().__init__()
  59. def initAlgorithm(self, config=None):
  60. self.TYPES = [self.tr('Use Input Layer Data Type'), 'Byte', 'Int16', 'UInt16', 'UInt32', 'Int32', 'Float32',
  61. 'Float64', 'CInt16', 'CInt32', 'CFloat32', 'CFloat64', 'Int8']
  62. self.addParameter(QgsProcessingParameterRasterLayer(self.INPUT,
  63. self.tr('数据源')))
  64. self.addParameter(QgsProcessingParameterFeatureSource(self.MASK,
  65. self.tr('导出的数据范围'),
  66. [QgsProcessing.TypeVectorPolygon]))
  67. self.addParameter(QgsProcessingParameterNumber(self.NODATA,
  68. self.tr('无效值'),
  69. type=QgsProcessingParameterNumber.Double,
  70. defaultValue=None,
  71. optional=True))
  72. # self.addParameter(QgsProcessingParameterBoolean(self.CROP_TO_CUTLINE,
  73. # self.tr('按范围进行导出'),
  74. # defaultValue=True))
  75. options_param = QgsProcessingParameterString(self.OPTIONS,
  76. self.tr('高级设置'),
  77. defaultValue='',
  78. optional=True)
  79. options_param.setFlags(options_param.flags() | QgsProcessingParameterDefinition.FlagAdvanced)
  80. options_param.setMetadata({
  81. 'widget_wrapper': {
  82. 'class': 'processing.algs.gdal.ui.RasterOptionsWidget.RasterOptionsWidgetWrapper'}})
  83. self.addParameter(options_param)
  84. dataType_param = QgsProcessingParameterEnum(self.DATA_TYPE,
  85. self.tr('输出栅格类型'),
  86. self.TYPES,
  87. allowMultiple=False,
  88. defaultValue=0)
  89. dataType_param.setFlags(dataType_param.flags() | QgsProcessingParameterDefinition.FlagAdvanced)
  90. self.addParameter(dataType_param)
  91. self.addParameter(QgsProcessingParameterRasterDestination(self.OUTPUT,
  92. self.tr('导出位置')))
  93. def name(self):
  94. return 'rasterexport'
  95. def displayName(self):
  96. return self.tr('栅格数据分发')
  97. def icon(self):
  98. return QIcon(os.path.join(pluginPath, 'images', 'dbms', 'export-img.png'))
  99. def group(self):
  100. return self.tr('数据分发')
  101. def groupId(self):
  102. return 'dataexport'
  103. def commandName(self):
  104. return 'gdalwarp'
  105. def getConsoleCommands(self, parameters, context, feedback, executing=True):
  106. inLayer = self.parameterAsRasterLayer(parameters, self.INPUT, context)
  107. if inLayer is None:
  108. raise QgsProcessingException(self.invalidRasterError(parameters, self.INPUT))
  109. maskLayer, maskLayerName = self.getOgrCompatibleSource(self.MASK, parameters, context, feedback, executing)
  110. # sourceCrs = self.parameterAsCrs(parameters, self.SOURCE_CRS, context)
  111. # targetCrs = self.parameterAsCrs(parameters, self.TARGET_CRS, context)
  112. # bbox = self.parameterAsExtent(parameters, self.EXTENT, context)
  113. # bboxCrs = self.parameterAsExtentCrs(parameters, self.EXTENT, context)
  114. if self.NODATA in parameters and parameters[self.NODATA] is not None:
  115. nodata = self.parameterAsDouble(parameters, self.NODATA, context)
  116. else:
  117. nodata = None
  118. options = self.parameterAsString(parameters, self.OPTIONS, context)
  119. out = self.parameterAsOutputLayer(parameters, self.OUTPUT, context)
  120. self.setOutputValue(self.OUTPUT, out)
  121. arguments = ['-overwrite']
  122. data_type = self.parameterAsEnum(parameters, self.DATA_TYPE, context)
  123. if data_type:
  124. if self.TYPES[data_type] == 'Int8' and GdalUtils.version() < 3070000:
  125. raise QgsProcessingException(self.tr('Int8 data type requires GDAL version 3.7 or later'))
  126. arguments.append('-ot ' + self.TYPES[data_type])
  127. output_format = QgsRasterFileWriter.driverForExtension(os.path.splitext(out)[1])
  128. if not output_format:
  129. raise QgsProcessingException(self.tr('Output format is invalid'))
  130. arguments.append('-of')
  131. arguments.append(output_format)
  132. arguments.append('-cutline')
  133. arguments.append(maskLayer)
  134. arguments.append('-cl')
  135. arguments.append(maskLayerName)
  136. # if self.parameterAsBoolean(parameters, self.CROP_TO_CUTLINE, context):
  137. # arguments.append('-crop_to_cutline')
  138. if nodata is not None:
  139. arguments.append(f'-dstnodata {nodata}')
  140. if options:
  141. arguments.extend(GdalUtils.parseCreationOptions(options))
  142. source = inLayer.source()
  143. if source.__contains__("dbname"):
  144. uid = getUUID() + ".tif"
  145. tempdir = get_temp_dir()
  146. tempfile = f'{tempdir}/{uid}'
  147. arguments.append(tempfile)
  148. arguments.append(out)
  149. ####### 开始处理gdal导出postgis栅格数据 ########
  150. real = source.replace("\"\"\"", "").replace(" (rast)", "")
  151. re = sourceToDBConfig(real)
  152. st = dict_to_str_with_equals(re)
  153. source = f"PG:{st}"
  154. print(source)
  155. cmd = f'gdal_translate -of GTiff "{source}" {tempfile}'
  156. return [cmd, self.commandName(), GdalUtils.escapeAndJoin(arguments)]
  157. else:
  158. arguments.append(source)
  159. arguments.append(out)
  160. return [self.commandName(), GdalUtils.escapeAndJoin(arguments)]