JQuery UI Slider - 多个滑块的组合Max,也具有单独的最大值

时间:2012-02-20 11:17:26

标签: javascript jquery jquery-ui uislider

- 已解决 -
我知道它们存在,但我还没有找到一个可以轻松满足一些非常基本需求的滑块,我相信我不是唯一一个对这个问题感到沮丧的人。这就是我到目前为止所拥有的。我正朝着正确的方向前进吗?

我正在尝试做什么:

使用JQuery UI滑块,我需要在页面上有多个滑块。每个滑块的最小值为0,动态最大值和动态默认值。我在输入字段中使用html元素min,max,value设置这些值,稍后通过javascript / jquery检索它们。

除了各自的最大值之外,还需要一个总最大值,即所有滑块的总和不能超过。最后,我还需要从输入框(以及滑块本身)更新滑块的可用性。

我到目前为止:

在研究了许多幻灯片之后,我决定使用JQuery UI滑块,最终发现这篇文章让我开始实现目标:Combined total for multiple jQuery-UI Sliders

我将代码更改为(1)使用隐藏的输入字段来检索总的最大值; (2)用当前值填充输入文本字段而不是span; (3)从max =属性设置单个滑块的初始最大值; (4)将滑块设置为禁用:如果初始最大值为0,则为true;否则为0。 (5)听取文本框的更改,并根据输入的数字移动滑块,如果它不超过最大值。

当前代码(JSFiddle):(请参阅答案中的jsfiddle)

(滑块#1初始最大值为0 - 禁用//其他滑块最大值为500 //总表格最大值为1000)

滑块正在做什么:

  1. 调整得很好,并且正确地保持在总最大限制之下
  2. 正确设置初始个人最大和最大总数
  3. 当max为0
  4. 时设置并保持禁用状态
  5. 更改文本框时正确滑动
  6. 我需要什么帮助/有关修复的想法:

    1. 更改滑块往往会更改单个滑块max =总最大值
    2. 更改文本框会正确滑动当前栏,但不会更新其他滑块以防止它们超过总计最大值
    3. 我目前在文本框上使用.focusout()事件,因为.change()导致它变得疯狂b / c滑块也在更改文本框。
    4. 让我知道你的想法。

      更新

      我很难过没有听到任何回应,但我想我已经破解了。需要限制原始代码的可靠性并且主要是重写。我将在下面提交我的解决方案作为答案。请试试看,让我知道你的想法。这是一个非常好的过程,所以希望有人发现这很有用:)

2 个答案:

答案 0 :(得分:5)

JSFiddle: http://jsfiddle.net/mikecruz/bAntY/

Javascript / Jquery:

var sliders = $("#sliders .slider");

sliders.each(function() {
    var max = document.getElementById("sliderMax").value;
    var value = Number($(this).text(), 10),
        availableTotal = max;

    $(this).empty().slider({
        value: 0,
        min: 0,
        max: $(this).siblings().attr("max"),
        range: "max",
        step: 1,
        animate: 100,
        disabled: (function(curMax) {
            if (curMax < 1) {
                return 1;
            }
            return 0;
        })($(this).siblings().attr("max")),
        stop: function(event, ui) {
            // Update text box to current value and call .change()
            $(this).siblings().attr("value", ui.value);
            $(this).siblings().trigger('change');
        }
    });
});

$(".value").change(function() {
    var thisAmount = Number($(this).prop("value"));
    var totalMax = Number(document.getElementById("sliderMax").value);
    var indMin = Number($(this).attr("min"));
    var indMax = Number($(this).attr("max"));
    var total = 0;

    //Get the values of all other text boxes
    $('.value').not(this).each(function() {
        total += Number($(this).prop("value"));
    });

    //Find the remaining from our total sliders max
    var remaining = totalMax - total;

    if(remaining < 0) {
      remaining = 0;   
    }
    //if we are under our minimums, go for it! Otherwise, reduce the number.
    if (thisAmount >= indMin && thisAmount < indMax && thisAmount < totalMax && thisAmount < remaining) {
        $(this).siblings(".slider").slider("option", "value", thisAmount);
        //total += thisAmount;
    }
    else {
        //var setMax = ((indMax + totalMax) - Math.abs(indMax - totalMax)) / 2;
        var setMax = Math.min(indMax, totalMax, remaining);
        $(this).siblings(".slider").slider("option", "value", setMax);
        $(this).prop("value", setMax);
        //total += (thisAmount - setMax);
    }

    //above was getting buggy, so lets just reset total and get it again
    total = 0;
    //Get the values of all text boxes
    $('.value').each(function() {
        total += Number($(this).prop("value"));
    });

    //Find our new remaining number after updating total for this value
    remaining = totalMax - total;
    if(remaining < 0) {
      remaining = 0;  
    }
    //Set each slider to the current point and update their max values.
    $('.value').each(function() {
    var sliderVal = Number($(this).prop("value"));
    var sliderMin = Number($(this).attr("min"));
    var sliderMax = Number($(this).attr("max"));
    var setNewMax = (((sliderMax + totalMax) - Math.abs(sliderMax - totalMax)) / 2);
    var newMax = sliderVal + remaining;

    $(this).prop("max", newMax);        
    if(newMax < setNewMax) {
        $(this).siblings('.slider').slider("option", "max", newMax);
        $(this).siblings('span').text('/' + (newMax));
    }
    else {
        $(this).siblings('.slider').slider("option", "max", setNewMax);
        $(this).siblings('span').text('/' + (setNewMax));
    }
    $(this).siblings(".slider").slider("option", "value", sliderVal);
    });

    $('#sliderTotal').attr("value", total);
});

<强> HTML
每个滑块需要一个li 您可以使用CSS来另外设置不同的部分。例如。给slideBG一个背景图片 您可能希望隐藏总计框,当然,您可以编写最大值的脚本。

<form>
<ul id="sliders">
    <li>
        <span class="slideBG"><div class="slider"></div>
        <input type="text" class="value" name="slider1" value="0" min="0" max="0"/>
        <span>0</span></span>
    </li>
    <li>
        <span class="slideBG"><div class="slider"></div>
        <input type="text" class="value" name="slider2" value="0" min="0" max="500" />
        <span>0</span></span>
    </li>
    <li>
        <span class="slideBG"><div class="slider"></div>
        <input type="text" class="value" name="slider3" value="0" min="0" max="500" />
        <span>0</span></span>
    </li>
    <li>
        <span class="slideBG"><div class="slider"></div>
        <input type="text" class="value" name="slider4" value="0" min="0" max="500"/>
           <span>0</span></span>
    </li>
    <li>
        <span class="slideBG"><div class="slider"></div>
        <input type="text" class="value" name="slider5" value="0" min="0" max="500"/>
           <span>0</span></span>
    </li>
    <li>
        <span class="slideBG"><div class="slider"></div>
        <input type="text" class="value" name="slider6" value="0" min="0" max="500"/>
           <span>0</span></span>
    </li>
</ul>
    <br><label for="sliderMax">Total Max:</label>
    <input type="text" id="sliderMax" name="sliderMax" value="1000"/>
    <br><br><label for="sliderTotal">Slider Totals:</label>
    <input type="text" id="sliderTotal" name="sliderTotal" value="0" />
</form>

修改

如果禁用滑块,您可能希望将其添加到以防止人们使用文本框:

$(".slider").each(function() {
    var disabled = Number($(this).slider("option", "disabled"));
    if(disabled == 1) {
        $(this).siblings('.value').attr('disabled', 'disabled');    
    }
});

答案 1 :(得分:2)

您好我刚刚尝试过类似的东西,但是以不同的方式实现了它。我有多个滑块,最多一个。所有滑块的值总和不能超过总数。防爆。 3个滑块,总计100%。滑块可以是10%,50%,40%。

这并不能满足您的所有需求,但对于寻找类似内容的其他用户来说可能会有所帮助。

var sliders = $('.slider');
sliders.each(function(i) {
  var slider = $(sliders[i]);
  var input = $('#' + slider.attr('datanamespace'));
  slider.slider({
    value: slider.attr('datatotal'),
    min: 0,
    max: max,
    step: 1,
    slide: function( event, ui ) {
      if (ui.value < slider.attr('datavalue')) {
        return false;
      }
      var inputs = $('.slideInput');
      var total = ui.value;

      for (var i = 0; i < inputs.length; i++) {
        var inp = $(inputs[i]);
        if (inp.attr('id') != input.attr('id')) {
          total = total + parseInt(inp.val());
        }
      }

      if (total > max) {
        return false;
      }
      input.val(ui.value);
    }
  });
  input.val(slider.slider("value"));
});