【问题标题】:How to disable enter when autocomplete option not selected: Google Maps SearchBox未选择自动完成选项时如何禁用输入:Google Maps SearchBox
【发布时间】:2017-09-06 22:30:20
【问题描述】:

使用带有自动完成功能的 Google 地图搜索框:

如果用户输入显示结果的名称,并在选择实际结果之前按 Enter 键,“places_matched”事件仍会触发,API 会为他们选择结果。

如何强制用户选择/突出显示自动完成结果,或者至少检测到他们实际上没有选择结果并阻止请求的地方?

在此处查看 jsfiddle:http://jsfiddle.net/4rs1mgno/2/

在文本输入中,输入任何内容,然后按 Enter。我输入“测试”,它会输出“BYU 测试中心”。我不想发生任何事情,因为他们实际上并没有选择一个选项。

这是小提琴中的 JS 代码:

 $(function(){      
 var input = document.getElementById('pac-input');
    var options = {
      keyword: 'establishment'
    }
    var searchBox = new google.maps.places.SearchBox(input, options);

    searchBox.addListener('places_changed',     function(evt) {
      console.log('changed');

      var places = searchBox.getPlaces();

      if (places.length == 0) {
        return;
      }
      places.forEach(function(place) {
        $('#output').append(place['name']);
      });
    });
});

谢谢!

这里有很多关于 SO 的类似问题,但没有一个与此匹配。

【问题讨论】:

    标签: javascript google-maps google-maps-api-3


    【解决方案1】:

    您可以使用Autocomplete ServicePlaces Service 并模仿SearchBox 的行为。使用getPlacePredictions,您可以轻松控制搜索字段和搜索结果项的行为。

    这个例子已经有一些 CSS 使它看起来“像”搜索框。您当然可以适应和增强。代码已被注释,因此它应该说明一切。

    var autocompleteService, placesService, results, map;
    
    function initialize() {
    
      results = document.getElementById('results');
    
      var mapOptions = {
        zoom: 5,
        center: new google.maps.LatLng(50, 50)
      };
    
      map = new google.maps.Map(document.getElementById('map-canvas'), mapOptions);
    
      // Bind listener for address search
      google.maps.event.addDomListener(document.getElementById('address'), 'input', function() {
    
        results.style.display = 'block';
        getPlacePredictions(document.getElementById('address').value);
      });
    
      // Show results when address field is focused (if not empty)
      google.maps.event.addDomListener(document.getElementById('address'), 'focus', function() {
    
        if (document.getElementById('address').value !== '') {
    
          results.style.display = 'block';
          getPlacePredictions(document.getElementById('address').value);
        }
      });
    
      // Hide results when click occurs out of the results and inputs
      google.maps.event.addDomListener(document, 'click', function(e) {
    
        if ((e.target.parentElement.className !== 'pac-container') && (e.target.parentElement.className !== 'pac-item') && (e.target.tagName !== 'INPUT')) {
    
          results.style.display = 'none';
        }
      });
    
      autocompleteService = new google.maps.places.AutocompleteService();
      placesService = new google.maps.places.PlacesService(map);
    }
    
    // Get place predictions
    function getPlacePredictions(search) {
    
      autocompleteService.getPlacePredictions({
        input: search,
        types: ['establishment', 'geocode']
      }, callback);
    }
    
    // Get place details
    function getPlaceDetails(placeId) {
    
      var request = {
        placeId: placeId
      };
    
      placesService.getDetails(request, function(place, status) {
    
        if (status === google.maps.places.PlacesServiceStatus.OK) {
    
          var center = place.geometry.location;
          var marker = new google.maps.Marker({
            position: center,
            map: map
          });
    
          map.setCenter(center);
    
          // Hide autocomplete results
          results.style.display = 'none';
        }
      });
    }
    
    // Place search callback
    function callback(predictions, status) {
    
      // Empty results container
      results.innerHTML = '';
    
      // Place service status error
      if (status != google.maps.places.PlacesServiceStatus.OK) {
        results.innerHTML = '<div class="pac-item pac-item-error">Your search returned no result. Status: ' + status + '</div>';
        return;
      }
    
      // Build output for each prediction
      for (var i = 0, prediction; prediction = predictions[i]; i++) {
    
        // Insert output in results container
        results.innerHTML += '<div class="pac-item" data-placeid="' + prediction.place_id + '" data-name="' + prediction.terms[0].value + '"><span class="pac-icon pac-icon-marker"></span>' + prediction.description + '</div>';
      }
    
      var items = document.getElementsByClassName("pac-item");
    
      // Results items click
      for (var i = 0, item; item = items[i]; i++) {
    
        item.onclick = function() {
    
          getPlaceDetails(this.dataset.placeid);
        };
      }
    }
    
    google.maps.event.addDomListener(window, 'load', initialize);
    body,
    html {
      font-family: Arial, sans-serif;
      padding: 0;
      margin: 0;
      height: 100%;
    }
    
    #map-canvas {
      height: 150px;
      width: 200px;
      margin-bottom: 10px;
    }
    
    .search {
      width: 200px;
      padding: 5px;
      float: left;
    }
    
    label {
      display: block;
      width: 160px;
      font-size: 11px;
      color: #777;
      margin-bottom: 5px;
    }
    
    input {
      border: 1px solid #ccc;
      width: 170px;
      padding: 3px 5px;
      box-sizing: border-box;
      -moz-box-sizing: border-box;
      -webkit-box-sizing: border-box;
      box-shadow: 0 2px 6px rgba(0, 0, 0, .1);
    }
    
    .pac-container {
      background-color: #fff;
      z-index: 1000;
      border-radius: 2px;
      font-size: 11px;
      box-shadow: 0 2px 6px rgba(0, 0, 0, .3);
      -moz-box-sizing: border-box;
      -webkit-box-sizing: border-box;
      box-sizing: border-box;
      overflow: hidden;
      width: 170px;
    }
    
    .pac-icon {
      width: 15px;
      height: 20px;
      margin-right: 7px;
      margin-top: 6px;
      display: inline-block;
      vertical-align: top;
      background-image: url(https://maps.gstatic.com/mapfiles/api-3/images/autocomplete-icons.png);
      background-size: 34px;
    }
    
    .pac-icon-marker {
      background-position: -1px -161px;
    }
    
    .pac-item {
      cursor: pointer;
      padding: 0 4px;
      text-overflow: ellipsis;
      overflow: hidden;
      white-space: nowrap;
      line-height: 30px;
      vertical-align: middle;
      text-align: left;
      border-top: 1px solid #e6e6e6;
      color: #999;
    }
    
    .pac-item:hover {
      background-color: #efefef;
    }
    
    .pac-item-error,
    .pac-item-error:hover {
      color: #aaa;
      padding: 0 5px;
      cursor: default;
      background-color: #fff;
    }
    <div class="search">
      <label for="address">Address:</label>
      <input id="address" placeholder="Enter address" type="text" tabindex="1" />
      <br />
      <div id="results" class="pac-container"></div>
    </div>
    <div id="map-canvas"></div>
    
    <script src="https://maps.googleapis.com/maps/api/js?libraries=places&key=AIzaSyCkUOdZ5y7hMm0yrcCQoCvLwzdM6M8s5qk"></script>

    【讨论】:

      【解决方案2】:

      这里是更新后的fiddle 解决方案。这个想法非常简单,因为每次您在 DOM 树中本地击键时,搜索框最多缓存 5 个结果,直到您找到匹配项。它们缓存在div.pac-container 元素中。

       $(function(){      
         var input = document.getElementById('pac-input');
         var options = {
            keyword: 'establishment'
         }
         var searchBox = new google.maps.places.SearchBox(input, options);
      
         searchBox.addListener('places_changed',  function(evt) {
      
           var places = searchBox.getPlaces();
      
           if ($('.pac-item').length > 0) {
             return;
           }
      
           places.forEach(function(place) {
             $('#output').append(place['name']);
           });
         });
      });
      

      【讨论】:

      • 虽然这可能(今天)有效,但它依赖于一个类(.pac-item 一个),如果 Google 在下一个版本中更改其类名,那么您的代码将停止工作。
      【解决方案3】:

      如果用户按下回车,则不会发生任何事情。
      编辑 3:

      $(function(){      
       var input = document.getElementById('pac-input');
          var options = {
            keyword: 'establishment'
          }
          var searchBox = new google.maps.places.Autocomplete(input, options);
          
          google.maps.event.addListener(searchBox, 'place_changed', function () {
      
            var place = searchBox.getPlace();
            if(place.address_components != null && place.address_components.length >= 1){
              console.log(place.address_components.length);
              $('#output').append(place['name']);
            }
            
      
        });
          
      });
      body {         
          font:14px sans-serif;           
      } 
      input {
          margin: 0.6em 0.6em 0; 
          width:398px;
      }
      #map_canvas {         
          height: 400px;         
          width: 400px;         
          margin: 0.6em;       
      }
      <script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
      <script src="http://maps.google.com/maps/api/js?libraries=places&region=us&language=en&sensor=false"></script>
      <input id="pac-input" type="text" size="50">             
      <div id="output"></div>

      【讨论】:

      • 您的解决方案出错。 Uncaught ReferenceError: e is not defined.
      • 我的错,替换e with evt
      • evt 未定义。请检查this link 没有任何arg 传递给event
      • 你已经试过了,只是需要更多的负担。你可以解决这个问题。
      • @csharpbd searchBox.addListener('places_changed', function(evt) { 你可以看到evt 已定义,:)
      猜你喜欢
      • 1970-01-01
      • 2011-12-26
      • 2017-10-04
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2012-06-02
      • 2012-07-08
      • 1970-01-01
      相关资源
      最近更新 更多