Quickly add Google Driving Directions to your rails application

Posted on August 27, 2007

Do you want driving directions on your website but don’t want to mess around with the Google Maps api?

Well, you can easily implement Google driving directions by calling the url: http://maps.google.com/maps?saddr=start_address&daddr=dest_address&hl=en

So, in Rails you could do something like this:
<%= button_to_function "Get Driving Directions",  "window.open(\"http://maps.google.com/maps?saddr=#{URI.escape(source_address)}&daddr=#{URI.escape(destination_address)}&hl=en\")" %>

Where source_address and destination_address are string variables containing things like:

London -> Manchester
SE5 0LL -> NW6 6RJ
221b Baker St -> 10 Downing Street

This method handles all the geocoding for ya – and is way simpler than messing with the GDirections api object.

How to dynamically serve a KML file with Ruby on Rails

Posted on August 24, 2007

Serving a KML file in Ruby on Rails is way easier than doing it in asp.net.

Here’s how I do it for my little display kml on a google map application.

First of all, you want to trick Google into thinking that the kml file is just a regular file with the extension “kml”. I do this by using a custom route in routes.rb that looks a little something like this:
map.connect ':controller/:action/:ignore_this_bit/:uuid.:format 
#The :ignore_this_bit is optional if you want to get around the google  
#caching issue detailed below
So that means you could serve the file with a url like this: http://www.my_domain.com/my_controller/get_kml/'random_number'/11.kml If you put a random number in there then you’ll get around Googles 4 minute cache. Next you create a controller method that will spit out the KML and render it to the browser using the send_data function like the example detailed below. (I load my kml data using a UUID by the way – but you can do it any way you want):
  def get_kml
    @kml_data = KmlData.find_by_uuid(params[:uuid])
    if @kml_data.kml     
        send_data @kml_data.kml
    else
      render :text => 'Kml is empty'
    end
  end

The @kml_data.kml variable contains the raw xml formatted kml.

The real guts of the function is the send_data function. This is rubys way of sending a file to the browser, which is how google expects to view the file.

And that’s it! Piece of cake!

For Loop in SQL Server 2000

Posted on August 24, 2007

Here’s a just-so-crazy-it-might-work way of doing a for loop in sql server.

This comes in real handy when making fake data for testing purposes.

DECLARE @count INT
DECLARE @command_to_loop NVARCHAR(1000)
-- This example runs from 1 to 10
SET @count = 1
WHILE(@count <= 10)
BEGIN
    -- This is just my crazy example
    -- Replace @command_to_loop with whatever you want to execute loop
    SET @command_to_loop = 'insert into customers (name) values 
                       (''Billy Bob' + cast(@count as varchar(3)) + ''')'
    print @command_to_loop
    EXEC sp_executesql @command_to_loop
    SET @count = @count + 1
END

How to dynamically serve a KML file in Asp.net 2.0

Posted on August 06, 2007

KML is the markup language that Google uses to represent markers in Google Earth – and NOW in Google Maps!

How it works, is that you store the KML in a file on a web server where Google can get at it. Google then goes at caches the file, and generates the output map markers.

But what if you want to display dynamic markers using KML?

Here’s how:

First add an aspx page to your project. This will be the page to serve the KML.

Create an xml document that will represent the kml file using the XmlDocument class. Should look something like this:
XmlDocument xmlDoc = new XmlDocument();

XmlDeclaration xmlDeclaration = xmlDoc.CreateXmlDeclaration("1.0","utf-8",null); 

// Create the root element
XmlElement rootNode  = xmlDoc.CreateElement("kml");
rootNode.SetAttribute("xmlns", @"http://earth.google.com/kml/2.1");
xmlDoc.InsertBefore(xmlDeclaration, xmlDoc.DocumentElement); 
xmlDoc.AppendChild(rootNode);
XmlElement documentNode = xmlDoc.CreateElement("Document");
rootNode.AppendChild(documentNode);

XmlElement iconNode = xmlDoc.CreateElement("Icon");
XmlElement iconHrefNode = xmlDoc.CreateElement("Href");
XmlText iconHrefText = xmlDoc.CreateTextNode("marker_image.gif");
iconHrefNode.AppendChild(iconHrefText);
iconNode.AppendChild(iconHrefNode);

documentNode.AppendChild(iconNode);

// MyMarkerClass is my own object used to hold my internal longlat stuff - replace with your own mechanism here
foreach (MyMarkerClass marker in markers)
{
  // Create a new <Placemark> element and add it to the root node
  XmlElement placeMarkNode = xmlDoc.CreateElement("Placemark");
  documentNode.AppendChild(placeMarkNode);

  // Create the required nodes
  XmlElement nameNode = xmlDoc.CreateElement("name");
  XmlText nameText = xmlDoc.CreateTextNode(marker.Name);
  placeMarkNode.AppendChild(nameNode);
  nameNode.AppendChild(nameText);
  XmlElement descNode = xmlDoc.CreateElement("description");                
  XmlText descriptionText = xmlDoc.CreateTextNode(marker.Description);                
  placeMarkNode.AppendChild(descNode);
  descNode.AppendChild(descriptionText);

  XmlElement pointNode = xmlDoc.CreateElement("Point");
  placeMarkNode.AppendChild(pointNode);

  XmlElement coordsNode = xmlDoc.CreateElement("coordinates");
  XmlText coordsText = xmlDoc.CreateTextNode(string.Format("{0},{1},{2}", marker.Long, marker.Lat, 0));
  pointNode.AppendChild(coordsNode);
  coordsNode.AppendChild(coordsText);                
}
Then, in your Page_Load method, write the KML directly to the output like this:
Response.ContentType = "application/x-msdownload";

XmlTextWriter writer = new XmlTextWriter(Response.OutputStream, null);
xmlDoc.WriteTo(writer);
writer.Flush();

Almost there! Now you’ll need to rewrite the url so that Google can try and find it with an innocuous “myfile.kml” path. We created a url rewriting method in house, but there are tutorials on line on how to do it (try here or here )

Now all you need to do is include the url rewritten kml path in your javascript, and Google will use it beautifully!