2013-07-19 47 views
0

我有一個使用谷歌地圖API的vf頁面。我有一個簡單的頂點控制器,它具有使用JS遠程處理的遠程操作方法。initialize()方法未定義

如果我從帳戶運行該頁面並將accountId傳遞到standardController,該頁面會拉起並顯示帶有該帳戶標記位置的Google地圖。

如果我直接打開頁面而不傳入帳戶ID,頁面上拉,但不顯示在地圖上,我得到兩個腳本錯誤:

語法錯誤的: VAR LAT =;

ReferenceError:初始化未定義 intitialize();

我無法確定是什麼導致了錯誤。誰能幫忙?

這裏是我的控制器:

public with sharing class AccountMapControllerExtension { 

public Account account {get; set;} 

public AccountMapControllerExtension(ApexPages.StandardController stdController) { 
    if(ApexPages.currentPage().getParameters().get('id') != null) { 
     account = [select Id, Name, Geolocation__Latitude__s, Geolocation__Longitude__s, BillingStreet, BillingCity, BillingState, BillingPostalCode, BillingCountry from Account where id =: ApexPages.currentPage().getParameters().get('id')]; 
    } 
} 

@RemoteAction 
public static List<Account> getNearbyAccounts(Decimal latitude, Decimal longitude) { 
    String q = 'select Id, Name, Geolocation__Latitude__s, Geolocation__Longitude__s, BillingStreet, BillingCity, BillingCountry, BillingState, BillingPostalCode from Account '; 
    q += 'where DISTANCE(Geolocation__c, GEOLOCATION('; 
    q += String.valueOf(latitude) + ', ' + String.valueOf(longitude); 
    q += '), \'km\') < 500'; 

    return Database.query(q);  
} 
} 

這裏是VF頁:

<apex:page standardController="Account" extensions="AccountMapControllerExtension" doctype="html-5.0" > 
<apex:sectionHeader title="Map of Nearby Accounts"/> 
<title>Account Map</title> 

<head> 
    <!-- STYLE SHEETS --> 
    <apex:stylesheet value="{!URLFOR ($Resource.jQueryUI, '/css/ui-lightness/jquery-ui-1.9.0.custom.css')}" /> 

</head> 
<style>  

    .urlLinks { 

     color: blue; 
     text-decoration:underline 

    } 


    #map { 
     font-family: Arial; 
     font-size:12px; 
     line-height:normal !important; 
     height:400px;   
     padding: 20px; 
    }  
    .roundCornerCss{ 
     /* outer shadows (note the rgba is red, green, blue, alpha) */ 
     -webkit-box-shadow: 0px 0px 12px rgba(0, 0, 0, 0.4); 
     -moz-box-shadow: 0px 1px 6px rgba(23, 69, 88, .5); 

     /* rounded corners */ 
     -webkit-border-radius: 12px; 
     -moz-border-radius: 7px; 
     border-radius: 7px; 

     /* gradients */ 
     background: -webkit-gradient(linear, left top, left bottom, 
     color-stop(0%, white), color-stop(15%, white), color-stop(100%, #D7E9F5)); 
     background: -moz-linear-gradient(top, white 0%, white 55%, #D5E4F3 130%); 
    } 

    #loadingScreen { 
     background: url({!URLFOR($Resource.Markers, 'Markers/ajax-loader.gif')}) no-repeat 5px 8px; 
     padding-left: 50px; 
    } 

    /* hide the close x on the loading screen */ 
    .loadingScreenWindow .ui-dialog-titlebar-close { 
     display: none; 
    } 

</style> 
<body> 
<apex:includeScript value="{!URLFOR($Resource.jQueryUI, '/js/jquery-1.8.2.min.js')}"/> 
<apex:includeScript value="{!URLFOR($Resource.jQueryUI, '/js/jquery-1.8.2.js')}"/> 
<apex:includeScript value="{!URLFOR($Resource.jQueryUI, '/js/jquery-ui-1.9.0.custom.js')}"/>  
<script type="text/javascript" 
    src="https://maps.google.com/maps/api/js?sensor=false"> 
</script> 
<script type="text/javascript"> 

// Global Variables 

var geocoder; 
var map; 
var infowindow = new google.maps.InfoWindow(); 
var places = []; 
var title_content = new Array();      
var popup_content = new Array();      
var address = new Array(); 
var address_position = 0;      
var timeout = 600; 
var pendingCalls = {}; 

var markersArray = []; 

$j = $.noConflict(); // No Conflict Custom $j 


// Initialize 

$j(document).ready(function(){ 

    regDialogs(); 

}); 

// Load Google Map 

function initialize() { 

     geocoder = new google.maps.Geocoder(); 
     var latlng = new google.maps.LatLng(42.3583, -71.0603); // Boston, MA as Center 
     var myOptions = { 
      zoom: 3, 
      center: latlng, 
      mapTypeId: 'roadmap' 
     } 

     map = new google.maps.Map(document.getElementById("map"), myOptions); 
} 

function regDialogs(){ 

    $j("#pendDiv").dialog({ 
     autoOpen: false, 
     title: 'Pending Call Reports' 
    }); 

    $j("#loadingScreen").dialog({ 
     autoOpen: false, // set this to false so we can manually open it 
     dialogClass: "loadingScreenWindow", 
     closeOnEscape: false, 
     draggable: false, 
     width: 460, 
     minHeight: 50, 
     modal: true, 
     buttons: {}, 
     resizable: false, 
     open: function() { 
      // scrollbar fix for IE 
      $j('body').css('overflow','hidden'); 
     }, 
     close: function() { 
      // reset overflow 
      $j('body').css('overflow','auto'); 
     } 
    }); // end of dialog 


    $j("#selectOption").change(function(){ 

     if($j("select option:selected").text() == 'CUSTOM RANGE'){ 

      var dates = '<br/>Select Date range: <input type="text" id="startDate"/> &nbsp; to &nbsp;<input type="text" id="endDate"/>'; 
      $j('#rangeDiv').append(dates); 
      regDateHandlers(); 

     }else { 

      $j('#rangeDiv').empty(); 

     } 
    });  
} 

// Custom Date popups JQUERY UI 

function regDateHandlers(){ 

    $j(function() { 
     $j("#startDate").datepicker({dateFormat: "yy-mm-dd"}); 
    }); 

    $j(function() { 
     $j("#endDate").datepicker({dateFormat: "yy-mm-dd"}); 
    }); 

} 


// Waiting Dialog on Loading 
function waitingDialog() { 

    $j("#loadingScreen").html('<p>Please Wait ...</p>'); 
    $j("#loadingScreen").dialog('option', 'title', 'Loading'); 
    $j("#loadingScreen").dialog('open'); 

} 
// Close Waiting Dialog 
function closeWaitingDialog() { 
    $j("#loadingScreen").dialog('close'); 

} 

function getCurrentAccountMap() { 
    var billingStreet = '{!account.BillingStreet}'; 
    var billingCity = '{!account.BillingCity}'; 
    var billingState = '{!account.BillingState}'; 
    var billingCountry = '{!account.BillingCountry}'; 
    var billingPostalCode = '{!account.BillingPostalCode}'; 
    address_position = 0; 

    $j("#messages").empty(); 

    clearOverlay(); // Remove existing markers if any 

    var addr = billingStreet + ',' + billingCity + ',' + 
       billingState + ',' + billingCountry + ',' + 
       billingPostalCode; 

    address.push(addr);  
    addMarker(address_position); 
} 

/* 

    Main method that gets the Accounts and Call Records for the Selected Date Range. 

*/ 

function getAccountsNearbyMap(){ 

    waitingDialog(); // Start Loading 
    var lat = {!account.Geolocation__Latitude__s}; 
    var lng = {!account.Geolocation__Longitude__s}; 

    /* Uses JS Remoting to get the Accounts for the Selected Date Range, with Call Reports only*/ 

    AccountMapControllerExtension.getNearbyAccounts(lat,lng,function(result,event){ 

     $j("#messages").empty(); 

     clearOverlay(); // Remove existing markers if any 

     if(event.type == 'exception'){ 
       alert('Error ' + event.message); 
     }else { 

      address = new Array(); 
      address_position = 0; 
      var completeDate = ''; 
      var pendClick = ''; 
      var hasCallRecords = false; 

      $j.each(result,function(rec){ 
       var addr = result[rec].BillingStreet + ',' + result[rec].BillingCity + ',' + 
            result[rec].BillingState + ',' + result[rec].BillingCountry + ',' + 
            result[rec].BillingPostalCode; 
       address.push(addr); 
      }); 

      if(address.length > 0){ 
       addMarker(address_position); // Add the markers based on the Date Range Selected 
      }else{ 
       $j("#messages").append('<h3 style="color:red;">No Accounts with Call Records Found for this Date Range</h3><br/>'); 
       closeWaitingDialog(); 
       return false; 

      } 
     } 

    }); 

    } 

    /*Open the Dialog with the Call Reports that are not completed.*/ 

    function openDialog(clicked){ 

    $j("#pendDiv").empty(); 

    var clickedId = clicked.id; 

    var pendingCS = ''; 

    $j.each(pendingCalls[clickedId],function(c){ 
     pendingCS += '<a href="/'+pendingCalls[clickedId][c].CallId+'" target="_blank" class="urlLinks">' + pendingCalls[clickedId][c].Call + "</a><br/>"; 
    }); 

    $j("#pendDiv").append(pendingCS); 

    $j("#pendDiv").dialog("open"); 

    } 

    /* Add Markers Dynamically */ 

    function addMarker(position){ 
     geocoder.geocode({'address': address[position]}, function(results, status){ 
      if (status == google.maps.GeocoderStatus.OK) { 
       places[position] = results[0].geometry.location;          
       var marker = new google.maps.Marker({ 
        position: places[position], 
        title:title_content[position], 
        icon: getMapIconUrl(position), 
        map: map 
       }); 
       markersArray.push(marker); 
       google.maps.event.addListener(marker, 'click', function() { 
        if (!infowindow) { 
         infowindow = new google.maps.InfoWindow({maxWidth: 200}); 
        } 
        infowindow.setContent(popup_content[position]); 
        infowindow.open(map, marker); 
       }); 


      } 
      else{ 
       if (status == google.maps.GeocoderStatus.OVER_QUERY_LIMIT){ 
        setTimeout(function() { addMarker(position); }, (timeout * 3)); 
       } 
      } 
      address_position++; 
      if (address_position < address.length){ 
       setTimeout(function() { addMarker(address_position); }, (timeout)); 
      } 
      closeWaitingDialog(); 

     }); 
    } 

    function clearOverlay(){ 

     for (var i = 0; i < markersArray.length; i++) { 
      markersArray[i].setMap(null); 
     } 

    } 

    /* 
     Sets the Marker Type/Color based on the Marker if Pending or Complete 
    */ 

    function getMapIconUrl(markerNumber){ 

     var mapIconUrl = "{!URLFOR($Resource.Markers, 'Markers/GreenPin.png')}"; 

     if(String(popup_content[markerNumber]).indexOf('Pending') !== -1) 
      mapIconUrl = "{!URLFOR($Resource.Markers, 'Markers/PinkPin.png')}"; 

     return mapIconUrl; 
    } 
    window.onload = function() { 
     getCurrentAccountMap(); 
    }; 
</script> 
<!-- HTML Content --> 
<div id="inputDiv"> 
<select id="selectOption"> 
    <option value="month">LAST MONTH</option> 
    <option value="quarter">LAST QUARTER</option> 
    <option value="year">LAST YEAR</option> 
    <option value="custom">CUSTOM RANGE</option> 
</select><br/> 
<div id="rangeDiv"></div><br/> 
<input type="button" onclick="getAccountsNearbyMap();" class="btn" value="Submit"/> 
    </div><br/><br/> 

<div id="messages"></div><br/> 

<div id="map" class="roundCornerCss"></div> 

<script> 
     initialize(); 
</script> 

<div id="canvas_div"></div> 
<div id="pendDiv"></div>   
<div id="loadingScreen"></div> 
</body> 
</apex:page> 

回答

0

好吧,這是你的問題。

var lat = {!account.Geolocation__Latitude__s}; 
var lng = {!account.Geolocation__Longitude__s}; 

這不是一個有效的JS語句。在JS中,{}表示期望排列key:value的地圖。在你的情況下,沒有這樣的事情發生。如果您打算將!account.Geolocation__Latitude__s的值指定爲lat,則必須從兩個語句中刪除括號{},然後嘗試。

+0

這是您如何引用頂點的方法和屬性在VF頁。另外,如果我通過accountId,它爲什麼會起作用? – Dman100

+0

不只是任何帳戶。我假設您在通過的賬戶上擁有Geolocation__Latitude__s和Geolocation__Longitude__s的價值?如果您通過一個帳戶這些值爲空,您應該看到相同的錯誤。 – Hraefn

0

這是你的問題,正確的是:它被加載到瀏覽器之前

var lat = {!account.Geolocation__Latitude__s}; 
var lng = {!account.Geolocation__Longitude__s}; 

Visualforce構建HTML文檔。如果您有任何Geolocation__Latitude__sGeolocation__Longitude__s沒有價值,那麼瀏覽器的Javascript引擎會看到以下

var lat = ; 
var lng = ; 

和錯誤的。你已經有一個控制器擴展;創建緯度的新方法和經度值,將返回默認值,像

public Integer getLatitude() { 
    if (account.Id <> null) { 
    if (account.Geolocation__Latitude__s <> null) { 
     return account.Geolocation__Latitude__s; 
    } 
    } 
    // default value 
    return 0.0; 
} 

,並呼籲通過

var lat = {!latitude};