通过UTM坐标系进行偏移

This commit is contained in:
weixin_46229132 2025-02-14 20:14:06 +08:00
parent 971517c145
commit f6d6f112c9
2 changed files with 107 additions and 46 deletions

View File

@ -1,6 +1,6 @@
import argparse import argparse
from datetime import timedelta from datetime import timedelta
from odm_preprocess import PreprocessConfig, ImagePreprocessor from odm_preprocess_fast import PreprocessConfig, ImagePreprocessor
def parse_args(): def parse_args():
parser = argparse.ArgumentParser(description='ODM预处理工具') parser = argparse.ArgumentParser(description='ODM预处理工具')

View File

@ -3,19 +3,47 @@ import subprocess
import json import json
import shutil import shutil
import logging import logging
from pyproj import Transformer
class ConvertOBJ: class ConvertOBJ:
def __init__(self, output_dir: str, center_lat: float, center_lon: float): def __init__(self, output_dir: str, center_lat: float, center_lon: float):
self.output_dir = output_dir self.output_dir = output_dir
self.center_lat = center_lat self.center_lat = center_lat
self.center_lon = center_lon self.center_lon = center_lon
# 用于存储所有grid的UTM范围
self.min_east = float('inf')
self.min_north = float('inf')
self.max_east = float('-inf')
self.max_north = float('-inf')
# 初始化UTM到WGS84的转换器
self.transformer = Transformer.from_crs(
"EPSG:32649", "EPSG:4326", always_xy=True)
self.logger = logging.getLogger('UAV_Preprocess.ConvertOBJ') self.logger = logging.getLogger('UAV_Preprocess.ConvertOBJ')
def convert_grid_obj(self, grid_points): def convert_grid_obj(self, grid_points):
"""转换每个网格的OBJ文件为OSGB格式""" """转换每个网格的OBJ文件为OSGB格式"""
os.makedirs(os.path.join(self.output_dir, "osgb", "Data"), exist_ok=True) os.makedirs(os.path.join(self.output_dir,
tile_infos = [] "osgb", "Data"), exist_ok=True)
# 第一次遍历获取所有grid的UTM范围
for grid_id, points in grid_points.items():
base_dir = os.path.join(
self.output_dir,
f"grid_{grid_id[0]}_{grid_id[1]}",
"project"
)
log_file = os.path.join(
base_dir, "odm_orthophoto", "odm_orthophoto_log.txt")
east_offset, north_offset = self.read_utm_offset(log_file)
# 更新UTM范围
self.min_east = min(self.min_east, east_offset)
self.min_north = min(self.min_north, north_offset)
self.max_east = max(self.max_east, east_offset)
self.max_north = max(self.max_north, north_offset)
tile_infos = []
for grid_id in grid_points.keys(): for grid_id in grid_points.keys():
try: try:
tile_info = self._convert_single_grid(grid_id, grid_points) tile_info = self._convert_single_grid(grid_id, grid_points)
@ -32,54 +60,46 @@ class ConvertOBJ:
project_dir = os.path.join(self.output_dir, grid_name, "project") project_dir = os.path.join(self.output_dir, grid_name, "project")
texturing_dir = os.path.join(project_dir, "odm_texturing") texturing_dir = os.path.join(project_dir, "odm_texturing")
opensfm_dir = os.path.join(project_dir, "opensfm") opensfm_dir = os.path.join(project_dir, "opensfm")
# 检查输入文件是否存在
obj_file = os.path.join(texturing_dir, "odm_textured_model_geo.obj") obj_file = os.path.join(texturing_dir, "odm_textured_model_geo.obj")
log_file = os.path.join(
project_dir, "odm_orthophoto", "odm_orthophoto_log.txt")
if not os.path.exists(obj_file): if not os.path.exists(obj_file):
raise FileNotFoundError(f"找不到OBJ文件: {obj_file}") raise FileNotFoundError(f"找不到OBJ文件: {obj_file}")
# 2. 执行格式转换 # 2. 读取UTM偏移量修改obj文件顶点坐标
utm_offset = self.read_utm_offset(log_file)
modified_obj = self.modify_obj_coordinates(
obj_file, utm_offset)
# 3. 执行格式转换
self.logger.info(f"开始转换网格 {grid_id} 的OBJ文件") self.logger.info(f"开始转换网格 {grid_id} 的OBJ文件")
output_osgb = os.path.join(texturing_dir, "Tile.osgb") output_osgb = os.path.join(texturing_dir, "Tile.osgb")
# 计算当前网格相对于中心点的偏移
grid_data = grid_points[grid_id]
lats = [point['lat'] for point in grid_data]
lons = [point['lon'] for point in grid_data]
min_lat = min(lats)
min_lon = min(lons)
# 计算偏移量(米)
offset_x = self._calculate_distance(self.center_lat, self.center_lon, self.center_lat, min_lon)
offset_y = self._calculate_distance(self.center_lat, self.center_lon, min_lat, self.center_lon)
# 修改转换命令,使用正确的参数格式
cmd = ( cmd = (
f"osgconv {obj_file} {output_osgb} " f"osgconv {obj_file} {output_osgb} "
f"--compressed --smooth --fix-transparency " f"--compressed --smooth --fix-transparency "
f"-t {offset_x},{offset_y},0 " # 使用 -t 参数进行平移
f"-o 0,1,0-0,0,-1" f"-o 0,1,0-0,0,-1"
) )
self.logger.info(f"执行osgconv命令{cmd}")
try: try:
subprocess.run(cmd, shell=True, check=True, cwd=texturing_dir) subprocess.run(cmd, shell=True, check=True, cwd=texturing_dir)
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
raise RuntimeError(f"OSGB转换失败: {str(e)}") raise RuntimeError(f"OSGB转换失败: {str(e)}")
# 3. 读取地理信息 # 4. 读取地理信息
ref_lla_file = os.path.join(opensfm_dir, "reference_lla.json") ref_lla_file = os.path.join(opensfm_dir, "reference_lla.json")
with open(ref_lla_file, 'r') as f: with open(ref_lla_file, 'r') as f:
ref_lla = json.load(f) ref_lla = json.load(f)
# 4. 创建OSGB目录结构 # 5. 创建OSGB目录结构
osgb_base_dir = os.path.join(self.output_dir, "osgb") osgb_base_dir = os.path.join(self.output_dir, "osgb")
data_dir = os.path.join(osgb_base_dir, "Data") data_dir = os.path.join(osgb_base_dir, "Data")
tile_dir = os.path.join(data_dir, f"Tile_{grid_id[0]}_{grid_id[1]}") tile_dir = os.path.join(data_dir, f"Tile_{grid_id[0]}_{grid_id[1]}")
os.makedirs(tile_dir, exist_ok=True) os.makedirs(tile_dir, exist_ok=True)
# 5. 复制OSGB文件 # 5. 复制OSGB文件
target_osgb = os.path.join(tile_dir, f"Tile_{grid_id[0]}_{grid_id[1]}.osgb") target_osgb = os.path.join(
tile_dir, f"Tile_{grid_id[0]}_{grid_id[1]}.osgb")
shutil.copy2(output_osgb, target_osgb) shutil.copy2(output_osgb, target_osgb)
# 计算当前网格的边界框 # 计算当前网格的边界框
@ -93,10 +113,6 @@ class ConvertOBJ:
min_lon = min(lons) min_lon = min(lons)
max_lon = max(lons) max_lon = max(lons)
# 计算相对于中心点的偏移
offset_x = self._calculate_distance(self.center_lat, self.center_lon, self.center_lat, min_lon)
offset_y = self._calculate_distance(self.center_lat, self.center_lon, min_lat, self.center_lon)
tile_info = { tile_info = {
'id': f"{grid_id[0]}_{grid_id[1]}", 'id': f"{grid_id[0]}_{grid_id[1]}",
'bounds': { 'bounds': {
@ -105,7 +121,6 @@ class ConvertOBJ:
'min_lon': min_lon, 'min_lon': min_lon,
'max_lon': max_lon 'max_lon': max_lon
}, },
'offset': (offset_x, offset_y)
} }
return tile_info return tile_info
@ -141,7 +156,6 @@ class ConvertOBJ:
for tile in tile_infos: for tile in tile_infos:
metadata_content += f""" metadata_content += f"""
<Tile id="{tile['id']}"> <Tile id="{tile['id']}">
<Offset>{tile['offset'][0]},{tile['offset'][1]},0</Offset>
<BoundingBox> <BoundingBox>
<MinLat>{tile['bounds']['min_lat']}</MinLat> <MinLat>{tile['bounds']['min_lat']}</MinLat>
<MaxLat>{tile['bounds']['max_lat']}</MaxLat> <MaxLat>{tile['bounds']['max_lat']}</MaxLat>
@ -162,3 +176,50 @@ class ConvertOBJ:
metadata_file = os.path.join(self.output_dir, "osgb", "metadata.xml") metadata_file = os.path.join(self.output_dir, "osgb", "metadata.xml")
with open(metadata_file, 'w', encoding='utf-8') as f: with open(metadata_file, 'w', encoding='utf-8') as f:
f.write(metadata_content) f.write(metadata_content)
def read_utm_offset(self, log_file: str) -> tuple:
"""读取UTM偏移量"""
try:
east_offset = None
north_offset = None
with open(log_file, 'r') as f:
lines = f.readlines()
for i, line in enumerate(lines):
if 'utm_north_offset' in line and i + 1 < len(lines):
north_offset = float(lines[i + 1].strip())
elif 'utm_east_offset' in line and i + 1 < len(lines):
east_offset = float(lines[i + 1].strip())
if east_offset is None or north_offset is None:
raise ValueError("未找到UTM偏移量")
return east_offset, north_offset
except Exception as e:
self.logger.error(f"读取UTM偏移量时发生错误: {str(e)}")
raise
def modify_obj_coordinates(self, obj_file: str, utm_offset: tuple) -> str:
"""修改obj文件中的顶点坐标使用相对坐标系"""
east_offset, north_offset = utm_offset
output_obj = obj_file.replace('.obj', '_utm.obj')
try:
with open(obj_file, 'r') as f_in, open(output_obj, 'w') as f_out:
for line in f_in:
if line.startswith('v '):
# 处理顶点坐标行
parts = line.strip().split()
# 使用相对于整体最小UTM坐标的偏移
x = float(parts[1]) + (east_offset - self.min_east)
y = float(parts[2]) + (north_offset - self.min_north)
z = float(parts[3])
f_out.write(f'v {x:.6f} {y:.6f} {z:.6f}\n')
else:
# 其他行直接写入
f_out.write(line)
return output_obj
except Exception as e:
self.logger.error(f"修改obj坐标时发生错误: {str(e)}")
raise