}
const Selector = {
- DATA_TOGGLE_CARROT : '[data-toggle^="button"]',
- DATA_TOGGLE : '[data-toggle="buttons"]',
- INPUT : 'input:not([type="hidden"])',
- ACTIVE : '.active',
- BUTTON : '.btn'
+ DATA_TOGGLE_CARROT : '[data-toggle^="button"]',
+ DATA_TOGGLES : '[data-toggle="buttons"]',
+ DATA_TOGGLE : '[data-toggle="button"]',
+ DATA_TOGGLES_BUTTONS : '[data-toggle="buttons"] .btn',
+ INPUT : 'input:not([type="hidden"])',
+ ACTIVE : '.active',
+ BUTTON : '.btn'
}
const Event = {
CLICK_DATA_API : `click${EVENT_KEY}${DATA_API_KEY}`,
FOCUS_BLUR_DATA_API : `focus${EVENT_KEY}${DATA_API_KEY} ` +
- `blur${EVENT_KEY}${DATA_API_KEY}`
+ `blur${EVENT_KEY}${DATA_API_KEY}`,
+ LOAD_DATA_API : `load${EVENT_KEY}${DATA_API_KEY}`
}
/**
let triggerChangeEvent = true
let addAriaPressed = true
const rootElement = $(this._element).closest(
- Selector.DATA_TOGGLE
+ Selector.DATA_TOGGLES
)[0]
if (rootElement) {
$(button).toggleClass(ClassName.FOCUS, /^focus(in)?$/.test(event.type))
})
+$(window).on(Event.LOAD_DATA_API, () => {
+ // ensure correct active class is set to match the controls' actual values/states
+
+ // find all checkboxes/readio buttons inside data-toggle groups
+ let buttons = [].slice.call(document.querySelectorAll(Selector.DATA_TOGGLES_BUTTONS))
+ for (let i = 0, len = buttons.length; i < len; i++) {
+ const button = buttons[i]
+ const input = button.querySelector(Selector.INPUT)
+ if (input.checked || input.hasAttribute('checked')) {
+ button.classList.add(ClassName.ACTIVE)
+ } else {
+ button.classList.remove(ClassName.ACTIVE)
+ }
+ }
+
+ // find all button toggles
+ buttons = [].slice.call(document.querySelectorAll(Selector.DATA_TOGGLE))
+ for (let i = 0, len = buttons.length; i < len; i++) {
+ const button = buttons[i]
+ if (button.getAttribute('aria-pressed') === 'true') {
+ button.classList.add(ClassName.ACTIVE)
+ } else {
+ button.classList.remove(ClassName.ACTIVE)
+ }
+ }
+})
+
/**
* ------------------------------------------------------------------------
* jQuery
assert.strictEqual($btn.attr('aria-pressed'), 'true', 'btn aria-pressed state is true')
})
+ QUnit.test('should assign active class on page load to buttons with aria-pressed="true"', function (assert) {
+ assert.expect(1)
+ var done = assert.async()
+ var $btn = $('<button class="btn" data-toggle="button" aria-pressed="true">mdo</button>')
+ $btn.appendTo('#qunit-fixture')
+ $(window).trigger($.Event('load'))
+ setTimeout(function () {
+ assert.ok($btn.hasClass('active'), 'button with aria-pressed="true" has been given class active')
+ done()
+ }, 5)
+ })
+
+ QUnit.test('should assign active class on page load to button checkbox with checked attribute', function (assert) {
+ assert.expect(1)
+ var done = assert.async()
+ var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
+ '<label class="btn btn-primary">' +
+ '<input type="checkbox" id="radio" checked> Checkbox' +
+ '</label>' +
+ '</div>'
+ var $group = $(groupHTML).appendTo('#qunit-fixture')
+ var $btn = $group.children().eq(0)
+
+ $(window).trigger($.Event('load'))
+ setTimeout(function () {
+ assert.ok($btn.hasClass('active'), 'checked checkbox button has been given class active')
+ done()
+ }, 5)
+ })
+
+ QUnit.test('should remove active class on page load from buttons without aria-pressed="true"', function (assert) {
+ assert.expect(1)
+ var done = assert.async()
+ var $btn = $('<button class="btn active" data-toggle="button" aria-pressed="false">mdo</button>')
+ $btn.appendTo('#qunit-fixture')
+ $(window).trigger($.Event('load'))
+ setTimeout(function () {
+ assert.ok(!$btn.hasClass('active'), 'button without aria-pressed="true" has had active class removed')
+ done()
+ }, 5)
+ })
+
+ QUnit.test('should remove active class on page load from button checkbox without checked attribute', function (assert) {
+ assert.expect(1)
+ var done = assert.async()
+ var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
+ '<label class="btn btn-primary active">' +
+ '<input type="checkbox" id="radio"> Checkbox' +
+ '</label>' +
+ '</div>'
+ var $group = $(groupHTML).appendTo('#qunit-fixture')
+ var $btn = $group.children().eq(0)
+
+ $(window).trigger($.Event('load'))
+ setTimeout(function () {
+ assert.ok(!$btn.hasClass('active'), 'unchecked checkbox button has had active class removed')
+ done()
+ }, 5)
+ })
+
QUnit.test('should trigger input change event when toggled button has input field', function (assert) {
assert.expect(1)
var done = assert.async()
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
'<label class="btn btn-primary">' +
- '<input type="radio" id="radio" autocomplete="off">Radio' +
+ '<input type="radio" id="radio">Radio' +
'</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
QUnit.test('should not add aria-pressed on labels for radio/checkbox inputs in a data-toggle="buttons" group', function (assert) {
assert.expect(2)
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
- '<label class="btn btn-primary"><input type="checkbox" autocomplete="off"> Checkbox</label>' +
- '<label class="btn btn-primary"><input type="radio" name="options" autocomplete="off"> Radio</label>' +
+ '<label class="btn btn-primary"><input type="checkbox"> Checkbox</label>' +
+ '<label class="btn btn-primary"><input type="radio" name="options"> Radio</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
assert.expect(4)
var groupHTML = '<div class="btn-group disabled" data-toggle="buttons" aria-disabled="true" disabled>' +
'<label class="btn btn-danger disabled">' +
- '<input type="checkbox" aria-disabled="true" autocomplete="off" disabled>' +
+ '<input type="checkbox" aria-disabled="true" disabled>' +
'</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
assert.expect(4)
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
'<label class="btn btn-danger">' +
- '<input type="checkbox" autocomplete="off" disabled>' +
+ '<input type="checkbox" disabled>' +
'</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
assert.expect(4)
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
'<label class="btn">' +
- '<input type="checkbox" autocomplete="off">' +
+ '<input type="checkbox">' +
'</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
assert.expect(4)
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
'<div class="btn">' +
- '<input type="checkbox" autocomplete="off" aria-label="Check">' +
+ '<input type="checkbox" aria-label="Check">' +
'</div>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
assert.expect(4)
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
'<label class="btn">' +
- '<input type="checkbox" autocomplete="off">' +
+ '<input type="checkbox">' +
'</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
assert.expect(2)
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
'<label class="btn">' +
- '<input type="hidden" autocomplete="off">' +
+ '<input type="hidden">' +
'</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
assert.expect(2)
var groupHTML = '<div class="btn-group" data-toggle="buttons">' +
'<label class="btn">' +
- '<input type="text" autocomplete="off">' +
+ '<input type="text">' +
'</label>' +
'</div>'
var $group = $(groupHTML).appendTo('#qunit-fixture')
<div class="container">
<h1>Button <small>Bootstrap Visual Test</small></h1>
- <button type="button" class="btn btn-primary" data-toggle="button" aria-pressed="false" autocomplete="off">
+ <button type="button" class="btn btn-primary" data-toggle="button" aria-pressed="false">
Single toggle
</button>
<div class="btn-group" data-toggle="buttons">
<label class="btn btn-primary active">
- <input type="checkbox" checked autocomplete="off"> Checkbox 1 (pre-checked)
+ <input type="checkbox" checked> Checkbox 1 (pre-checked)
</label>
<label class="btn btn-primary">
- <input type="checkbox" autocomplete="off"> Checkbox 2
+ <input type="checkbox"> Checkbox 2
</label>
<label class="btn btn-primary">
- <input type="checkbox" autocomplete="off"> Checkbox 3
+ <input type="checkbox"> Checkbox 3
</label>
</div>
<div class="btn-group" data-toggle="buttons">
<label class="btn btn-primary active">
- <input type="radio" name="options" id="option1" autocomplete="off" checked> Radio 1 (preselected)
+ <input type="radio" name="options" id="option1" checked> Radio 1 (preselected)
</label>
<label class="btn btn-primary">
- <input type="radio" name="options" id="option2" autocomplete="off"> Radio 2
+ <input type="radio" name="options" id="option2"> Radio 2
</label>
<label class="btn btn-primary">
- <input type="radio" name="options" id="option3" autocomplete="off"> Radio 3
+ <input type="radio" name="options" id="option3"> Radio 3
</label>
</div>
</div>
Add `data-toggle="button"` to toggle a button's `active` state. If you're pre-toggling a button, you must manually add the `.active` class **and** `aria-pressed="true"` to the `<button>`.
{% capture example %}
-<button type="button" class="btn btn-primary" data-toggle="button" aria-pressed="false" autocomplete="off">
+<button type="button" class="btn btn-primary" data-toggle="button" aria-pressed="false">
Single toggle
</button>
{% endcapture %}
{% capture example %}
<div class="btn-group-toggle" data-toggle="buttons">
<label class="btn btn-secondary active">
- <input type="checkbox" checked autocomplete="off"> Checked
+ <input type="checkbox" checked> Checked
</label>
</div>
{% endcapture %}
{% capture example %}
<div class="btn-group btn-group-toggle" data-toggle="buttons">
<label class="btn btn-secondary active">
- <input type="radio" name="options" id="option1" autocomplete="off" checked> Active
+ <input type="radio" name="options" id="option1" checked> Active
</label>
<label class="btn btn-secondary">
- <input type="radio" name="options" id="option2" autocomplete="off"> Radio
+ <input type="radio" name="options" id="option2"> Radio
</label>
<label class="btn btn-secondary">
- <input type="radio" name="options" id="option3" autocomplete="off"> Radio
+ <input type="radio" name="options" id="option3"> Radio
</label>
</div>
{% endcapture %}