#!/usr/bin/perl

# My ultimate goal for this script is to convert a PDF containing a USGS
# topographic map into a number of tiles for use in Google Earth or my own
# not-yet-written mapping application.
#
# Go download your own PDFs from the USGS' website. Go to
# http://www.usgs.gov/pubprod/ , click on "Download digital scans of topo
# maps", then "Map Locator". Or, if that fails, try:
#
# http://usgs-catalog4.srv.mst.edu/store3/digital_download/mapping_ap.jsp
#
# My algorithm so far is:
#
# 1. Rasterize the PDF at 200 dpi.
# 2. Figure out where the corners of the map are. This may be best done
#    manually.
# 3. Deskew the image. In my test, it seemed to be as easy as rotating the
#    image by the correct angle, calculated as the average of the angles
#    between the corners.
# 4. Convert the corner coordinates from NAD-27 to WGS-84.
# 5. Write a kmz file containing the image and the reference.
# 
# Copyright (C) 2010 Ted Logan # http://tedlogan.com/ # ted.logan@gmail.com
# 
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
# 
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
# 
# You should have received a copy of the GNU General Public License along
# with this program; if not, write to the Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.

use strict;

use File::Spec;
use File::Temp qw(tempfile tempdir);
use Geo::Coordinates::DecimalDegrees;
use Image::Magick;
use Math::NumberCruncher;
use Math::Trig;
use Math::Trig ':pi';

sub read_pixel_coords {
	while(1) {
		my $coords = <>;
		if(my @c = $coords =~ /^\s*(\d+)\s*[, ]\s*(\d+)/) {
			return @c;
		}
	}
}

sub read_dms {
	my @c;
	do {
		my $c = <>;
		chomp $c;
		@c = split /\s+/, $c;
	} while(@c < 1);
	while(@c < 3) {
		push @c, 0;
	}
	return dms2decimal(@c);
}

my $filename = shift @ARGV;
unless($filename) {
	print "Enter image filename: ";
	$filename = <>; chomp $filename;
}

my $basename = (File::Spec->splitpath($filename))[2];

# Is this a PDF? Rasterize it, and give the user the filename
my $temp_raster;
if($filename =~ /\.pdf$/i) {
	my $pdf = $filename;

	my $template = $basename;
	$template =~ s/\.pdf$/XXXX/;
	$temp_raster = new File::Temp(
		TEMPLATE => $template,
		SUFFIX => '.png'
	);
	$filename = $temp_raster->filename();

	print "Rasterizing $pdf at 200 dpi\n";

	system("convert -density 200x200 \"$pdf\" -depth 8 \"$filename\"") == 0
		or die "Rasterizing PDF failed\n";

	print "\n";
	print "PDF rasterized as:\n";
	print "\t$filename\n";
	print "\n";

	system("gimp \"$filename\" &");
}

# Try to extract state and quad name from the filename
my ($state, $name) = $basename =~ /^([A-Z][A-Z])_([A-Za-z_-]+)/;
$name =~ s/_/ /g;
$name =~ s/^\s+//;
$name =~ s/\s+$//;

print "Enter state: ";
$state = do {
	if($state) {
		print "[$state] ";
	}
	my $input = <>; chomp $input;
	if($input) {
		$input;
	} else {
		$state;
	}
};

print "Enter quad name: ";
$name = do {
	if($name) {
		print "[$name] ";
	}
	my $input = <>; chomp $input;
	if($input) {
		$input;
	} else {
		$name;
	}
};

# Determine the image size so we know where the center is
my ($width, $height) = do {
	my $img = new Image::Magick;
	if($img->Read($filename)) {
		die "Unable to read image $filename\n";
	}
	$img->Get('width', 'height');
};

print "Enter pixel coordinates of top-left corner: ";
my @top_left = read_pixel_coords();

print "Enter pixel coordinates of top-right corner: ";
my @top_right = read_pixel_coords();

print "Enter pixel coordinates of bottom-left corner: ";
my @bottom_left = read_pixel_coords();

print "Enter pixel coordinates of bottom-right corner: ";
my @bottom_right = read_pixel_coords();

# Input coordinates in NAD-27
print "Enter latitude of the northern edge of the map: ";
my $north = read_dms();

print "Enter latitude of the southern edge of the map: ";
my $south = read_dms();

print "Enter longitude of the eastern edge of the map: ";
my $east = read_dms();

print "Enter longitude of the western edge of the map: ";
my $west = read_dms();

#
# Calculate the angle to rotate the map by
#
my $rotate = - do {
	sub angle {
		return atan2($_[1]->[1] - $_[0]->[1], $_[1]->[0] - $_[0]->[0]);
	}

	my @a;

	# top edge
	push @a, angle(\@top_left, \@top_right);
	# bottom edge
	push @a, angle(\@bottom_left, \@bottom_right);
	# left edge
	push @a, angle(\@top_left, \@bottom_left) - pip2; # PI/2
	# right edge
	push @a, angle(\@top_right, \@bottom_right) - pip2; # PI/2

	Math::NumberCruncher::Mean(\@a);
};

print "\n\n";
printf "Image should be rotated by %.5f degrees\n", rad2deg($rotate);

# Rotate the points around the top-right corner of the image so we know where
# to crop
sub rotate {
	my ($x, $y, $angle) = @_;
	$x -= $width;
	my $x_new = $width + cos($angle)*$x - sin($angle)*$y;
	my $y_new = sin($angle)*$x + cos($angle)*$y;
	return ($x_new, $y_new);
}

@top_left = rotate(@top_left, $rotate);
@top_right = rotate(@top_right, $rotate);
@bottom_left = rotate(@bottom_left, $rotate);
@bottom_right = rotate(@bottom_right, $rotate);

=for later
# Check how close each edge is
my $delta_left = abs($top_left[0] - $bottom_left[0]);
my $delta_right = abs($top_right[0] - $bottom_right[0]);
my $delta_top = abs($top_left[1] - $top_right[1]);
my $delta_bottom = abs($bottom_left[1] - $bottom_right[1]);

printf "Delta left: %d\n", $delta_left;
printf "Delta right: %d\n", $delta_right;
printf "Delta top: %d\n", $delta_top;
printf "Delta bottom: %d\n", $delta_bottom;

# TODO: Come up with some metrics to determine what acceptable deltas are.
# If they're too far off, odds are either the image is skewed (not simply
# rotated) or the coordinates were entered wrong.
=cut

my $left = Math::NumberCruncher::Mean([$top_left[0], $bottom_left[0]]);
my $right = Math::NumberCruncher::Mean([$top_right[0], $bottom_right[0]]);
my $top = Math::NumberCruncher::Mean([$top_left[1], $top_right[1]]);
my $bottom = Math::NumberCruncher::Mean([$bottom_left[1], $bottom_right[1]]);

my $crop = sprintf "%dx%d+%d+%d", $right - $left, $bottom - $top, $left, $top;

print "Crop geometry: $crop\n";

my $kmzdir = tempdir(CLEANUP => 1);

# Rotate and crop the file into the temp directory we've created for the kmz
my $jpeg = $filename;
$jpeg =~ s/.*\///;
$jpeg =~ s/\.[a-z]+$/.jpg/i;

print "Rotating and cropping $filename (to $kmzdir/$jpeg)...\n";
system("convert -depth 8 \"$filename\" -rotate $rotate -crop $crop \"$kmzdir/$jpeg\"") == 0
	or die "Unable to rotate and crop image\n";
print "done.\n";

#
# Convert coordinates to WGS-84
#

# Create a "universal CSV" file for gpsbabel
# Note that this requires gpsbabel >= 1.3.4
my ($infh, $unicsv) = tempfile();
print $infh "name,lon,lat\n";
printf $infh "north,%.6f,%.6f\n", ($east + $west) / 2, $north;
printf $infh "south,%.6f,%.6f\n", ($east + $west) / 2, $south;
printf $infh "east,%.6f,%.6f\n", $east, ($north + $south) / 2;
printf $infh "west,%.6f,%.6f\n", $west, ($north + $south) / 2;
close $infh;

my ($outfh, $csv) = tempfile();

system("gpsbabel -i unicsv,datum='N. America 1927 mean' -f $unicsv -o csv -F $csv") == 0
	or die "Failed to run gpsbabel to convert datums\n";

unlink $unicsv;

my %point;
while(<$outfh>) {
	chomp;
	my ($lat, $lon, $name) = split /,\s*/;
	$point{$name} = [$lon, $lat];
}

close $outfh;
unlink $csv;

# Write the KML file with the coordinates
open KML, ">$kmzdir/doc.kml"
	or die "Can't open doc.kml: $!\n";
print KML <<XML;
<?xml version="1.0" encoding="UTF-8"?>
<kml xmlns="http://earth.google.com/kml/2.2">
<GroundOverlay>
        <name>$state - $name</name>
        <Icon>
                <href>$jpeg</href>
        </Icon>
        <LatLonBox>
                <north>$point{north}->[1]</north>
                <south>$point{south}->[1]</south>
                <east>$point{east}->[0]</east>
                <west>$point{west}->[0]</west>
        </LatLonBox>
</GroundOverlay>
</kml>
XML
close KML;

# Create the kmz
my $kmz = $filename;
$kmz =~ s/\.[a-z]+$/.kmz/i;
print "Generating $kmz ...\n";
system("zip -j $kmz $kmzdir/*") == 0
	or die "Unable to create kmz\n";
print "Done.\n";

