百度地图切片源码

开发工具:lazarus

算法仍有问题。


unit unit_main;

{$mode objfpc}{$H+}

interface

uses
  Classes, SysUtils, FileUtil, Forms, Controls, Graphics, Dialogs, StdCtrls,
  LCLType, LCLIntf, ExtCtrls, IntfGraphics, GraphType, ComCtrls, regexpr,
  Math,BGRABitmap, BGRABitmapTypes, BCImageButton;

type

  { TFormMain }

  TFormMain = class(TForm)
    Button2: TButton;
    ButtonBrowsePic: TButton;
    Button4: TButton;
    BtnCut: TButton;
    Button6: TButton;
    BtnPreview: TButton;
    CombxSrcPic: TComboBox;
    CombxMin: TComboBox;
    CombxMax: TComboBox;
    EditPmzbY: TEdit;
    EditPmzbX: TEdit;
    EditClipBoard: TEdit;
    EditJD: TEdit;
    EditWD: TEdit;
    EditSrcPic: TEdit;
    EditDstPath: TEdit;
    GroupBox1: TGroupBox;
    Image1: TImage;
    Image2: TImage;
    Image3: TImage;
    Image0: TImage;
    ImageAll: TImage;
    Label1: TLabel;
    Label2: TLabel;
    Label3: TLabel;
    Label4: TLabel;
    Label5: TLabel;
    Label6: TLabel;
    Label7: TLabel;
    Label8: TLabel;
    LabelLink: TLabel;
    Memo1: TMemo;
    MemoMap: TMemo;
    OpenDialog1: TOpenDialog;
    Panel1: TPanel;
    pbar: TProgressBar;
    rbPng: TRadioButton;
    rbJpg: TRadioButton;
    SDirDiog: TSelectDirectoryDialog;
    procedure ButtonBrowsePicClick(Sender: TObject);
    procedure Button4Click(Sender: TObject);
    procedure BtnCutClick(Sender: TObject);
    procedure Button6Click(Sender: TObject);
    procedure BtnPreviewClick(Sender: TObject);
    procedure FormCreate(Sender: TObject);
    procedure FormDestroy(Sender: TObject);
    procedure LabelLinkClick(Sender: TObject);
  private
    { private declarations }

    //AImage: TLazIntfImage;
    //lRawImage: TRawImage;
    BGRABmpAll, BGRAImgText: TBGRABitmap;
    procedure SaveToPng(bmp: TBitmap; PngFileName: String);
    // 加载图片文件
    procedure loadPic;
  public
    tukuaiCenterX, tukuaiCenterY: integer; // 中心图块坐标
    procedure ShowForm(FormClass: TFormClass);
    { public declarations }
  end;


var
  FormMain: TFormMain;

implementation

uses unit_map, Unit_PicShow;

{$R *.lfm}

{ TFormMain }

procedure TFormMain.ShowForm(FormClass: TFormClass);
begin
  with FormClass.Create(self) do
    try
      ShowModal;
    finally
      Free;
    end;
end;

procedure TFormMain.SaveToPng(bmp: TBitmap; PngFileName: String);
var
  png : TPortableNetworkGraphic;
begin
  png := TPortableNetworkGraphic.Create;
  try
    png.Assign(bmp);
    png.SaveToFile(PngFileName);
  finally
    png.Free;
  end;
end;

// 加载图片文件
procedure TFormMain.loadPic;
var
  BGRABmpPart, BGRAStretch: TBGRABitmap;
  i,j: Integer;
  fn,fnJpg,fnPng: string;
  Rect:TRect;
begin
  // 加载完整图片
  BGRABmpAll := TBGRABitmap.Create(utf8ToSys(EditSrcPic.Text));
  ImageAll.Picture.LoadFromFile((EditSrcPic.Text));

  // 显示打开图片的宽度及高度
  memo1.Lines.Add('图片宽度:' + intToStr(BGRABmpAll.Width) + '; 图片高度:' + intToStr(BGRABmpAll.Height));

  // 完整预览选择的图片
  BGRAStretch := BGRABmpAll.Resample(Panel1.Width, Panel1.Height) as TBGRABitmap;


  // 显示左上角图片 ---------
  //Rect.TopLeft:=Point(0,0);
  //Rect.BottomRight:=Point(256, 256);
  //BGRABmpAll.DrawPart(Rect, Image0.Canvas, 0, 0, true);
  // 显示左上角图片 =========

  //
  BGRABmpPart := TBGRABitmap.Create(256, 256);
  for i := 0 to 1 do
    for j := 0 to 1 do
      begin
        Rect.TopLeft:=Point(256 * i, 256 * j);
        Rect.BottomRight:=Point(256 * (i + 1), 256 * (j + 1));
        //BGRABmpPart.FillRect(0, 0, 256, 256, BGRA(0,0,0,0), dmset);
        BGRABmpAll.DrawPart(Rect, BGRABmpPart.Canvas, 0, 0, true);
        if (i = 0) and (j = 0) then
        begin
          BGRABmpAll.DrawPart(Rect, Image0.Canvas, 0, 0, true);
          memo1.Lines.Add('第1样张图片显示完毕,图片加黑边为正常。');
        end;

        fnPng := extractFilePath(paramStr(0)) + intToStr(i) + '_' + intToStr(j) + '.png';
        BGRABmpPart.SaveToFile(fnPng);
        memo1.Lines.Add('测试图片存储为:' + sysToUtf8(fnPng));

        fnJpg := extractFilePath(paramStr(0)) + intToStr(i) + '_' + intToStr(j) + '.jpg';
        BGRABmpPart.SaveToFile(fnJpg);
        memo1.Lines.Add('测试图片存储为:' + sysToUtf8(fnJpg));

      end;

  memo1.Lines.Add('测试图片处理完毕。');

  BGRABmpPart.Free;
  BGRAStretch.Free;


end;

// 源图片定位按钮
procedure TFormMain.ButtonBrowsePicClick(Sender: TObject);
var
  fn: string;
begin
  memo1.Clear;

  OpenDialog1.Filter:='jpeg文件|*.jpg|png文件|*.png';
  if OpenDialog1.Execute then
    EditSrcPic.Text := OpenDialog1.FileName;

  fn := trim(EditSrcPic.Text);
  if fn = '' then
    exit;

  loadPic;

  BtnCut.Enabled := true;
  BtnPreview.Enabled := true;
end;

// 目标路径
procedure TFormMain.Button4Click(Sender: TObject);
begin
  if SDirDiog.Execute then
  begin
    EditDstPath.Text := SDirDiog.FileName;
  end;
end;

// 开始切图按钮
procedure TFormMain.BtnCutClick(Sender: TObject);
var
  iGradeMin, iGradeMax, iGradeDef, xiangsuX, xiangsuY: integer;
  picSrc, destPath, destFn, f: string;
  srcPicStrechW, srcPicStrechH, rectBotmRighX, rectBotmRighY, srcTileWidth,
  tilCoordX, tilCoordY, iGradeCur, i, j, Vtimes, Htimes: integer;
  strList: TStringList;
  Rect:TRect;
  BGRAStretch, BGRABmpPart: TBGRABitmap;
  clrText: TBGRAPixel;
begin
  memo1.Clear;


  picSrc := trim(EditSrcPic.Text);
  if picSrc = '' then
  begin
    showmessage('请定位待处理图片。');
    exit;
  end;

  destPath := EditDstPath.Text;
  if destPath = '' then
  begin
    showmessage('请定义输出路径');
    exit;
  end;

  // 创建 tiles 目录 ----
  destPath := trim(EditDstPath.Text) + '\tiles';
  if not DirectoryExists(destPath) then
     CreateDir(destPath);
  // 创建 tiles 目录 ====

  iGradeMin := strToInt(CombxMin.Text); // 最小级别
  iGradeMax := strToInt(CombxMax.Text);  // 最大级别
  iGradeDef := strToInt(CombxSrcPic.Text);  // 原图所在的级别

  pbar.Min := 0;

  //生成 map.html ------------------------------------
  strList:= TStringList.Create;
  strList.LoadFromFile(ExtractFilePath(ParamStr(0)) + '\t.html');

  strList.Text := stringReplace(strList.Text, '#minZoom#', intToStr(iGradeMin), [rfReplaceAll]);
  strList.Text := stringReplace(strList.Text, '#maxZoom#', intToStr(iGradeMax), [rfReplaceAll]);

  strList.Text := stringReplace(strList.Text, '#center_x#', EditJD.Text, [rfReplaceAll]);
  strList.Text := stringReplace(strList.Text, '#center_y#', EditWD.Text, [rfReplaceAll]);

  strList.Text := stringReplace(strList.Text, '#defaultZoom#', intToStr(iGradeDef), [rfReplaceAll]);

  f := trim(EditDstPath.Text) + '\map.html';
  DeleteFile(f);
  strList.SaveToFile(f);
  strList.Free;
  //生成 map.html ====================================

  for iGradeCur := iGradeMin to iGradeMax do
  //for l := 6 to 8 do
  begin
    // 计算中心像素坐标
    // 像素坐标 = |平面坐标 × 2 iGradeCur - 18| (iGradeCur - 18是2的指数)
    xiangsuX := trunc(abs(strToFloat(EditPmzbX.Text) * power(2, iGradeCur - 18)));
    xiangsuY := trunc(abs(strToFloat(EditPmzbY.Text) * power(2, iGradeCur - 18)));

    memo1.Lines.Add('---------------------------------------');
    memo1.Lines.Add('处理级别:' + intToStr(iGradeCur));
    memo1.Lines.Add('中心像素坐标:' + intToStr(xiangsuX) + '-' + intToStr(xiangsuY));

    // 计算中心图块坐标
    // 图块坐标 = |像素坐标 ÷ 256|
    tukuaiCenterX := trunc(xiangsuX / 256);
    tukuaiCenterY := trunc(xiangsuY / 256);
    memo1.Lines.Add('中心图块坐标:' + intToStr(tukuaiCenterX) + '-' + intToStr(tukuaiCenterY));

    // 生成当前级别目录
    destPath := trim(EditDstPath.Text) + '\tiles\' + intToStr(iGradeCur);
    if not DirectoryExists(destPath) then
      CreateDir(destPath);

    // widthSingle := 256 div iGradeCur * gradeDef;   // 源图切割后,单个图片的边长

    // srcPicStrechW := trunc(abs(BGRABmpAll.Width * power(2, iGradeDef - iGradeCur))); // 计算某级别下,变形后源图宽度
    // srcPicStrechH := trunc(abs(BGRABmpAll.Height * power(2, iGradeDef - iGradeCur))); // 计算某级别下,变形后源图高度
    //memo1.Lines.Add('变形w:' + intToStr(srcPicStrechW));
    //memo1.Lines.Add('变形H:' + intToStr(srcPicStrechH));

    srcTileWidth := trunc(abs(256 * power(2, iGradeDef - iGradeCur))); // 计算源瓦片宽度

    memo1.Lines.Add('单个源瓦片宽度:' + intToStr(srcTileWidth));
    if srcTileWidth <=0 then
    begin
      memo1.Lines.Add('瓦片宽度不可小于1。');
      exit;
    end;

    // 创建宽度为  srcTileWidth 的 BGRABitmap,临时存储未变形的切片
    // BGRABmpPart := TBGRABitmap.Create(srcTileWidth, srcTileWidth);
    pbar.Max := iGradeCur - 1;

    // 横向分割块数
    if BGRABmpAll.Width mod srcTileWidth = 0 then
      Htimes := BGRABmpAll.Width div srcTileWidth
    else
      Htimes := BGRABmpAll.Width div srcTileWidth + 1;

    // 纵向分割块数
    if BGRABmpAll.Height mod srcTileWidth = 0 then
      Vtimes := BGRABmpAll.Height div srcTileWidth
    else
      Vtimes := BGRABmpAll.Height div srcTileWidth + 1;

    memo1.Lines.Add('Htimes:' + intToStr(Htimes));
    memo1.Lines.Add('Vtimes:' + intToStr(Vtimes));

    for i := 1 to Htimes do // 横向循环
    begin
      pbar.Position:=i;
      for j := 1 to Vtimes do  // 纵向循环
      begin
        // 创建宽度为  srcTileWidth 的 BGRABitmap,复制BGRABmpPart后用于变形
        BGRAStretch := TBGRABitmap.Create(srcTileWidth, srcTileWidth);
        // 创建宽度为  srcTileWidth 的 BGRABitmap,临时存储未变形的切片
        BGRABmpPart := TBGRABitmap.Create(srcTileWidth, srcTileWidth);

        // 定义待复制的源瓦片 Rect 参数 ----------------
        Rect.TopLeft:=Point(srcTileWidth * (i -  1), srcTileWidth * (j - 1));

        // 右下角坐标,如果超出变形后原图大小,会出错
        // Rect.BottomRight:=Point(srcTileWidth * i, srcTileWidth * j);
        // srcPicStrechW  rectBotmRighX, rectBotmRighY

        //取得 rect 右下角横坐标位置
        if srcTileWidth * i <= BGRABmpAll.Width then
          rectBotmRighX := srcTileWidth * i
        else
          rectBotmRighX := BGRABmpAll.Width; // srcPicStrechW 变形后原图宽度

        //取得 rect 右下角纵坐标位置
        if srcTileWidth * j <= BGRABmpAll.Height then
          rectBotmRighY := srcTileWidth * j
        else
          rectBotmRighY := BGRABmpAll.Height; // srcPicStrechW 变形后原图宽度

        Rect.BottomRight:=Point(rectBotmRighX, rectBotmRighY);
        // 定义待复制的源瓦片 Rect 参数 ================

        // 切出指定位置、宽度为 srcTileWidth 的图形
        BGRABmpAll.DrawPart(Rect, BGRABmpPart.Canvas, 0, 0, true);
        // 变形为边长256的图形
        BGRAStretch := BGRABmpPart.Resample(256, 256) as TBGRABitmap;

        // BGRAStretch.TextOutAngle(100, 100, -450, 'Hello world',c,);
        BGRAStretch.FontHeight := 50;
        BGRAStretch.FontAntialias := true;
        clrText := ColorToBGRA(ColorToRGB(clYellow));  // 字体颜色
        BGRAStretch.TextOutAngle(0, 0, 0, intToStr(iGradeCur) + ':' + intToStr(i) + ',' + intToStr(j), clrText, taLeftJustify);
        clrText := ColorToBGRA(ColorToRGB(clBtnText)); // 为产生阴影,换一种字体颜色
        BGRAStretch.TextOutAngle(1, 1, 0, intToStr(iGradeCur) + ':' + intToStr(i) + ',' + intToStr(j), clrText, taLeftJustify);

        //BGRAStretch.SetPixel(30,5, cText);

        //destFn := destPath + '\tile' + intToStr(tukuaiCenterX + i - iGradeCur div 2) + '_' + intToStr(tukuaiCenterY - (j - iGradeCur div 2)) + '.jpg';
        //destFn := destPath + '\tile' + intToStr(tukuaiCenterX - Htimes div 2 + i - 1) + '_' + intToStr(tukuaiCenterY + Vtimes div 2 - j) + '.jpg';

        tilCoordX := tukuaiCenterX - Htimes div 2 + i - 1;
        tilCoordY := tukuaiCenterY + (Vtimes + 1) div 2 - j;

        BGRAStretch.FontHeight := 30;
        clrText := ColorToBGRA(ColorToRGB(clYellow));  // 字体颜色
        BGRAStretch.TextOutAngle(10, 160, 0, intToStr(tilCoordX) + ',' + intToStr(tilCoordY), clrText, taLeftJustify);
        clrText := ColorToBGRA(ColorToRGB(clBtnText)); // 为产生阴影,换一种字体颜色
        BGRAStretch.TextOutAngle(11, 161, 0, intToStr(tilCoordX) + ',' + intToStr(tilCoordY), clrText, taLeftJustify);

        if (tilCoordX = tukuaiCenterX) and (tilCoordY = tukuaiCenterY) then // 中心图块
        begin
          BGRAStretch.FontHeight := 60;
          clrText := ColorToBGRA(ColorToRGB(clWhite));  // 字体颜色
          BGRAStretch.TextOutAngle(0, 10, 0, '中心图块', clrText, taLeftJustify);
          clrText := ColorToBGRA(ColorToRGB(clBtnText)); // 为产生阴影,换一种字体颜色
          BGRAStretch.TextOutAngle(1, 11, 0, '中心图块', clrText, taLeftJustify);
        end;

        destFn := destPath + '\tile' + intToStr(tilCoordX) + '_' + intToStr(tilCoordY) + '.jpg';


        memo1.Lines.Add(destFn);
        BGRAStretch.SaveToFile(destFn);

        BGRAStretch.Free;
        BGRABmpPart.Free;

      end;
    end;
    // BGRABmpPart.Free;
    pbar.Position:=0;
  end;

  memo1.Lines.Add('');
  memo1.Lines.Add('切片完成。');
  //bitA.Free;
  //bitTile.Free;
end;

// 获取经纬度
procedure TFormMain.Button6Click(Sender: TObject);
var
  s,jwd_x,jwd_y: string;
  RegexObj: TRegExpr;
  i, xiangsuX, xiangsuY,
  iYutuJibie // 当前级别
  : integer;
  //xiangsuX, xiangsuY: float;
begin
  memo1.Clear;

  if trim(CombxSrcPic.Text) = '' then
  begin
    showmessage('请确定当前级别。');
    exit;
  end;

  iYutuJibie := strToInt(CombxSrcPic.Text);

  if (iYutuJibie > 18) or (iYutuJibie < 1) then
  begin
    showmessage('级别范围1——18.');
    exit;
  end;

  memoMap.Lines.SaveToFile(ExtractFilePath(ParamStr(0)) + 'locamap');

  FormMap.ShowModal;

  // 从剪贴板获得经纬度  格式: jwd:116.716754,40.049897;pmzb:12992991,4845443.71
  EditClipBoard.PasteFromClipboard;
  s := EditClipBoard.Text;
  memo1.Lines.Add('点击的经纬度、平面坐标');
  memo1.Lines.Add(s);
  memo1.Lines.Add('');

  RegexObj := TRegExpr.Create;
  RegexObj.Expression := '\d+\.*\d+';
  RegexObj.ModifierI := true;

  // 取得经度 纬度 平面横坐标 平面纵坐标
  i := 0;
  if RegexObj.Exec(s) then
  repeat
    if i = 0 then
      EditJD.Text := RegexObj.Match[0]   // 经度
    else if i = 1 then
      EditWD.Text := RegexObj.Match[0]   // 纬度
    else if i = 2 then
      EditPmzbX.Text := RegexObj.Match[0]  // 平面坐标 x
    else if i = 3 then
      EditPmzbY.Text := RegexObj.Match[0]; // 平面坐标 y

    i := i + 1;
  until not RegexObj.ExecNext;

  RegexObj.Free;

  // 计算中心像素坐标
  // 像素坐标 = |平面坐标 × 2 iGradeCur - 18| (iGradeCur - 18是2的指数)
  xiangsuX := trunc(abs(strToFloat(EditPmzbX.Text) * power(2, iYutuJibie - 18)));
  xiangsuY := trunc(abs(strToFloat(EditPmzbY.Text) * power(2, iYutuJibie - 18)));
  memo1.Lines.Add('中心像素坐标:' + intToStr(xiangsuX) + '-' + intToStr(xiangsuY));

  // 计算图块坐标
  // 图块坐标 = |像素坐标 ÷ 256|
  tukuaiCenterX := trunc(xiangsuX / 256);
  tukuaiCenterY := trunc(xiangsuY / 256);
  memo1.Lines.Add('中心图块坐标:'  + intToStr(tukuaiCenterX) + '-' + intToStr(tukuaiCenterY));

end;

procedure TFormMain.BtnPreviewClick(Sender: TObject);
begin
  FormPicShow.ShowModal;
end;

procedure TFormMain.FormCreate(Sender: TObject);
begin
  LabelLink.Font.Style:= [fsUnderline];
  LabelLink.Cursor:= crHandPoint;

  EditSrcPic.Text := sysToUtf8(ExtractFilePath(paramStr(0)) + 'map.jpg');

  //loadPic;
end;

procedure TFormMain.FormDestroy(Sender: TObject);
begin
  BGRABmpAll.Free;
end;

procedure TFormMain.LabelLinkClick(Sender: TObject);
begin
  OpenURL('http://api.map.baidu.com/lbsapi/getpoint/index.html');
end;

end.


posted @ 2014-06-17 08:37  cuibq  阅读(831)  评论(0编辑  收藏  举报