/*
geoContrast v1.0
https://github.com/lucasfogliarini/geoContrast
*/
$(function(){
window.geoContrast = {
options_default: {
options_gmaps: {//api: http://goo.gl/x3h6o3
//types: ['geocode','(regions)','(cities)', 'establishment']
},
format: 'short',//formatted
gmaps_through_pin: true,
assign_through_tab: true,
assigned:{
input_title: '',
pin_title: 'Click to open on Google Maps.',
pin_img: ''
},
unassigned:{
input_title: '',
pin_title: 'Location not assigned.',
pin_img: ''
}
}
}
$.fn.geoContrast = function(options){
var $inputs = this;
for (var i = 0; i < $inputs.length; i++){
var current = $inputs[i];
var $current = $(current);
//props
current.index = i;
current.options = $.extend(true,'¬¬',window.geoContrast.options_default,options);
current.autocomplete_gmaps = new google.maps.places.Autocomplete(current, current.options.options_gmaps);
current.geocoder = new google.maps.Geocoder();
current.autocomplete_gmaps.geocontrast = current;//accessibility
current.$pin = $('');
current.$bounds = $('');
current.$lat = $('');
current.$lng = $('');
current.$formatted = $('');
//endprops
//methods
current.toggle = function(state){
var input_title, pin, pin_title;
if (state){
pin_img = this.options.assigned.pin_img;
pin_title = this.options.assigned.pin_title;
input_title = this.options.assigned.input_title;
if(input_title){
this.title = input_title;
}
}
else{
this.place_info = undefined;
this.sync_coords();
this.sync_bounds();
this.$formatted.val("");
pin_img = this.options.unassigned.pin_img;
pin_title = this.options.unassigned.pin_title;
input_title = this.options.unassigned.input_title;
}
this.$pin.css('background','no-repeat url('+pin_img+')');
this.$pin.attr('title',pin_title);
this.title = input_title;
}
current.assigned = function(){
return this.place_info !== undefined && this.place_info.formatted_address !== undefined;
}
current.bounds_assigned = function(){
return this.assigned() && this.place_info.geometry.bounds !== undefined;
}
current.sync_coords = function(){
try{
var lat;
var lng;
if(this.assigned()) {
lat = this.place_info.geometry.location.lat();
lng = this.place_info.geometry.location.lng();
}
this.$lat.val(lat);
this.$lng.val(lng);
} catch(ex){
throw ex;
}
}
current.sync_bounds = function(){
try{
if (this.$bounds !== undefined){
var bounds;
if (this.bounds_assigned()){
bounds = this.place_info.geometry.bounds.toUrlValue();
}else if(this.assigned()){
bounds = "0,0,0,0";
}
this.$bounds.val(bounds);
}
} catch(ex){
throw ex;
}
}
current.sync = function(){
var assigned = this.assigned();
if(assigned){
this.place_info.formatted_address = geoContrast.format_address(this.place_info.address_components);
switch(this.options.format){
case 'short':
this.value = this.place_info.name;
this.$formatted[0].value = this.place_info.formatted_address;
break;
case 'formatted':
this.value = this.place_info.formatted_address;
break;
}
this.sync_coords();
this.sync_bounds();
}
this.toggle(assigned);
}
current.find_address = function(address, call){
if(address !== ""){
var geocontrast = this;
this.geocoder.geocode({"address": address }, function(results) {
if (results.length > 0) {
geocontrast.place_info = results[0];
var addresses = geocontrast.place_info.address_components.filter(function(each){
return each.types[0] != "street_number";
});
geocontrast.place_info.name = addresses[0].long_name;
if (call !== undefined) {
call.call(geocontrast);
};
}
});
}
}
current.append_bounds = function(bounds_name){
bounds_name = bounds_name ? bounds_name : "bounds";
this.$bounds.prop('name',bounds_name);
$(this).before(this.$bounds);
}
current.append_coords = function(lat_name,lng_name){
lat_name = lat_name ? lat_name : "latitude";
lng_name = lng_name ? lng_name : "longitude";
lat_name = geoContrast.remake_name(this,function(args){
args.attribute = lat_name;
});
lng_name = geoContrast.remake_name(this,function(args){
args.attribute = lng_name;
});
this.$lat.prop('name',lat_name);
this.$lng.prop('name',lng_name);
$(this).before(this.$lng);
$(this).before(this.$lat);
}
current.first_hint = function(){
var eq = this.index;
return $('.pac-container:eq('+eq+') > :first').text();
}
//endmethods
//init
$current = $(current);
$current.addClass('geocontrast');
$current.css('padding-right','20px');
$current.after(current.$pin);
current.$pin.css({
position:'absolute',
left: current.offsetLeft + current.offsetWidth - 18,
top: current.offsetTop + ((current.offsetHeight - 16) / 2),
height: '16px',
width: '16px',
cursor: 'pointer'
});
if (current.options.format == "short") {
$current.before(current.$formatted);
var name = geoContrast.remake_name(current,function(args){
args.attribute += "_formatted";
});
current.$formatted.attr("name", name);
};
google.maps.event.addListener(current.autocomplete_gmaps, 'place_changed', function(){
this.geocontrast.place_info = this.getPlace();
this.geocontrast.place_info.geometry.bounds = this.geocontrast.place_info.geometry.viewport;//:/
this.geocontrast.sync();
});
$(document).on('input','.geocontrast',function(){
if(this.assigned()){
this.toggle(false);
}
});
$(document).on('keydown','.geocontrast',function(e){
switch(e.keyCode){
case 9:
if (this.options.assign_through_tab){
this.find_address(this.first_hint(), function(){
this.sync();
});
}
break;
case 13:
e.preventDefault();
this.find_address(this.first_hint(), function(){
this.sync();
});
break;
}
});
$(document).on('click','.pin_geocontrast',function(){
var input = $(this).prev()[0];
if (input.assigned() && input.options.gmaps_through_pin){
window.open('https://www.google.com.br/maps/place/'+input.place_info.formatted_address);
}
});
current.toggle(false);
current.find_address($current.data('find-address'),function(){
this.sync();
});
//endinit
}
return $inputs;
}
window.geoContrast.remake_name = function(input, delegate){
var args = {
attribute: input.name
}, attr_bracket
var first_bracket = input.name.indexOf('[');
if (first_bracket > 0){
attr_bracket = input.name.slice(first_bracket);
args.attribute = input.name.slice(first_bracket+1,-1);
}
delegate(args);
return first_bracket > 0 ? input.name.replace(attr_bracket,"["+args.attribute+"]") : args.attribute;
}
window.geoContrast.format_address = function(address_components){
var route, street_number, neighborhood, locality, area_lvl_1, country;
for (var i = 0; i < address_components.length; i++) {
route = address_components[i].types[0] == "route" ? address_components[i] : route;
street_number = address_components[i].types[0] == "street_number" ? address_components[i] : street_number;
neighborhood = address_components[i].types[0] == "neighborhood" ? address_components[i] : neighborhood;
locality = address_components[i].types[0] == "locality" ? address_components[i] : locality;
area_lvl_1 = address_components[i].types[0] == "administrative_area_level_1" ? address_components[i] : area_lvl_1;
country = address_components[i].types[0] == "country" ? address_components[i] : country;
};
var route_street_number, neighborhood_locality, area_lvl_1_country, format;
route_street_number = route ? route.long_name : "";
route_street_number += street_number ? ", " + street_number.long_name : "";
neighborhood_locality = neighborhood ? neighborhood.long_name + ', ' : "";
neighborhood_locality += locality ? locality.long_name : "";
area_lvl_1_country = neighborhood_locality ? area_lvl_1.short_name + ", " : area_lvl_1 ? area_lvl_1.long_name + ", " : "";
area_lvl_1_country += country ? country.long_name : "";
format = route_street_number ? route_street_number + ' - ' : '';
format += neighborhood_locality ? neighborhood_locality + ' - ' : '';
format += area_lvl_1_country ? area_lvl_1_country : '';
return format;
}
});