CSS:区域外的位置元素,不更改滚动条

时间:2018-02-19 04:07:27

标签: html css typeahead.js

我有一个模态对话框,当模态内容太大而无法包含时,会出现一个垂直滚动条。

我的问题是:当我想要“弹出”模态的控件(实际上是一个Twitter Typeahead https://twitter.github.io/typeahead.js/)出现时,它会导致模态上的滚动条延伸。

可能最好用例子解释:

所需行为

https://jsfiddle.net/0eara5xc/14/显示我想要的行为(在输入框中输入字母'a'),但是当常规内容太大(我需要)时,模态不会滚动。

不受欢迎的行为

只要我将模态滚动,弹出窗口就会导致滚动条增加。

.modal-body {
    max-height: 40vh;
    overflow: hidden;
    overflow-y: auto;
}

https://jsfiddle.net/0eara5xc/13/(向下滚动并在输入框中输入字母'a') 我实际上希望弹出窗口按照第一个例子流到模态之外。

Twitter Typeahead控件创建了自己的位置:relative和position:absolute elements,所以我认为我不能使用CSS来绝对定位动态创建的弹出元素。

2 个答案:

答案 0 :(得分:1)

更新你的小提琴:

.tt-menu {
  background-color: red;
  top:auto !important;
  bottom:100%;
}

https://jsfiddle.net/zd4xf0oo/

答案 1 :(得分:1)

这里的问题是你想要有两件相互矛盾的事情。在我达成解决方案的想法之前,我将用几个例子来解释。

如果我们采用流入元素(容器和内部元素),我们有两个坐标:

(1)内部元素的高度小于或等于到容器的高度。在这种情况下,所有溢出(autoscrollhidden)都是相同的。

(2)内部元素的高度大于容器的高度,在这种情况下我们:

  • 保持默认行为,我们会有溢出,我们可以查看元素



.el {
  border:5px solid green;
  height:100px;
}
.el > div {
  height:200px;
  width:100%;
  background:red;
}

<div class="el">
   <div>
   
   </div>
 </div>
&#13;
&#13;
&#13;

  • 使用overflow:hidden隐藏部分内部元素,以便无法看到溢出部分。

&#13;
&#13;
.el {
  border:5px solid green;
  height:100px;
  overflow:hidden;
}
.el > div {
  height:200px;
  width:100%;
  background:red;
}
&#13;
<div class="el">
   <div>
   
   </div>
 </div>
&#13;
&#13;
&#13;

  • 使用overflow:scroll隐藏部分内部元素,只能看到滚动上的溢出部分

&#13;
&#13;
.el {
  border:5px solid green;
  height:100px;
  overflow:scroll;
}
.el > div {
  height:200px;
  width:100%;
  background:red;
}
&#13;
<div class="el">
   <div>
   
   </div>
 </div>
&#13;
&#13;
&#13;

从技术上讲,我们不能让元素溢出滚动。

当使用absolute:position时,几乎在所有情况下我们将处于情境(2)中(除非我们为容器定义了一个高的高度),因为绝对定位的元素不在流中并且不是在容器的高度计算中考虑所以从逻辑上讲,内部元素的高度会更大,并且上面的逻辑将适用:

&#13;
&#13;
.el {
  border:5px solid green;
  position:relative;
}
.el > div {
  height:100px;
  background:red;
}
.el > div:last-child {
  position:absolute;
  background:blue;
  right:0;
  left:0;
}
&#13;
<div class="el">
   <div>
     we see the element
   </div>
   <div>
   
   </div>
 </div>

 <div class="el" style="overflow:hidden;margin-top:100px;">
   <div>
      we don't see the element
   </div>
   <div>
   
   </div>
 </div>
 
  <div class="el" style="overflow:scroll;">
   <div>
      we see the element element on scroll
   </div>
   <div>
   
   </div>
 </div>
&#13;
&#13;
&#13;

即使overflow属性设置为绝对定位元素的另一个元素祖先,相同的逻辑也适用。正如您可以阅读here

  

在大多数情况下,可以为边界中的任何框计算溢出   和该框本身的属性,加上每个的溢出   的孩子。

因此,当任何子元素的高度增加或溢出时,使用overflow:scroll将始终使滚动增加并且无法在外部使用任何溢出元素,因为这是默认行为并且您更改它设置overflow:scroll

解决方案

避免将滚动应用于元素的第一个简单解决方案是使元素相对于应用滚动的元素的祖先定位。换句话说:在具有overflow:scroll的元素和绝对位置元素之间,您不应找到任何定位元素。正如您可以阅读此复杂语句here

  

......一个盒子可能会溢出,这意味着它的内容部分或全部   在盒子外面,例如:

     

[...]

     

后代盒子绝对定位,部分位于盒子外面。   这些盒子并不总是被它们的溢出属性所削减   祖先;具体来说,它们不会因任何溢出而被削减   他们自己和他们的包含块之间的祖先

此解决方案当然会破坏您的代码。

另一个想法是制作元素position:fixed。像这样它将相对于浏览器窗口,我们可以在滚动之外看到它但是很难使用CSS调整它的位置。在这种情况下,我们可以考虑一些JS来动态调整元素的位置:

$('.typeahead').on('keypress change',function(e){
  var y = $(this).offset().top + $(this).height();
  var x = $(this).offset().left;
  $('.tt-menu').css('top',y);
  $('.tt-menu').css('left',x);
})

如您所见,我们只是根据输入的位置计算元素的上/左位置。

以下是完整代码:

&#13;
&#13;
$(function () {
$('.typeahead').on('keypress change',function(e){
  var y = $(this).offset().top + $(this).height()+5;
  var x = $(this).offset().left;
  $('.tt-menu').css('top',y);
  $('.tt-menu').css('left',x);
})



  $('#btnPopup').on('click', function(){
  	$('#divPopup').toggle();
  });
  
  var substringMatcher = function(strs) {

  return function findMatches(q, cb) {

    var matches, substringRegex;



    // an array that will be populated with substring matches

    matches = [];



    // regex used to determine if a string contains the substring `q`

    substrRegex = new RegExp(q, 'i');



    // iterate through the pool of strings and for any string that

    // contains the substring `q`, add it to the `matches` array

    $.each(strs, function(i, str) {

      if (substrRegex.test(str)) {

        matches.push(str);

      }

    });



    cb(matches);

  };

};



var states = ['Alabama', 'Alaska', 'Arizona', 'Arkansas', 'California',

  'Colorado', 'Connecticut', 'Delaware', 'Florida', 'Georgia', 'Hawaii',

  'Idaho', 'Illinois', 'Indiana', 'Iowa', 'Kansas', 'Kentucky', 'Louisiana',

  'Maine', 'Maryland', 'Massachusetts', 'Michigan', 'Minnesota',

  'Mississippi', 'Missouri', 'Montana', 'Nebraska', 'Nevada', 'New Hampshire',

  'New Jersey', 'New Mexico', 'New York', 'North Carolina', 'North Dakota',

  'Ohio', 'Oklahoma', 'Oregon', 'Pennsylvania', 'Rhode Island',

  'South Carolina', 'South Dakota', 'Tennessee', 'Texas', 'Utah', 'Vermont',

  'Virginia', 'Washington', 'West Virginia', 'Wisconsin', 'Wyoming'

];



$('.typeahead').typeahead({

  hint: true,

  highlight: true,

  minLength: 1

},

{

  name: 'states',

  source: substringMatcher(states)

});
  
});
&#13;
.modal-body {
  
 max-height: 40vh;
 overflow: hidden;
 overflow-y: auto;
}

.modal-header, .modal-footer {
  background-color: black;
  color: white;
}

#divSpacer {
  height: 40vh;
  background-color: rgb(220, 220, 220);
}

.tt-menu {
  background-color: red;
  position:fixed!important;
}
&#13;
<script type="text/javascript" src="//cdnjs.cloudflare.com/ajax/libs/jquery/3.2.1/jquery.js"></script>
<link rel="stylesheet" type="text/css" href="/css/result-light.css">
<link rel="stylesheet" type="text/css" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/css/bootstrap.min.css">
<script type="text/javascript" src="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/js/bootstrap.min.js"></script>
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/typeahead.js/0.11.1/typeahead.bundle.min.js"></script>
<div>
  <div class="modal-lg">
    <div class="modal-header">
      My modal title
    </div>
    <div class="modal-body">
      <div id="divSpacer">
        Lots of space taken up here...
      </div>
      
       <input class="typeahead" type="text" placeholder="States of USA">
      
    </div>
    <div class="modal-footer">
    </div>
  </div>
</div>
&#13;
&#13;
&#13;