]> git.ipfire.org Git - thirdparty/bootstrap.git/commitdiff
Split Sass, update docs
authorMark Otto <markdotto@gmail.com>
Tue, 23 Sep 2025 17:05:39 +0000 (10:05 -0700)
committerMark Otto <markdotto@gmail.com>
Wed, 24 Sep 2025 06:20:58 +0000 (23:20 -0700)
scss/_variables.scss
scss/forms/_check.scss [new file with mode: 0644]
scss/forms/_form-check.scss [deleted file]
scss/forms/_radio.scss [new file with mode: 0644]
scss/forms/_switch.scss
scss/forms/index.scss
site/data/sidebar.yml
site/src/content/docs/forms/checkbox.mdx
site/src/content/docs/forms/checks-radios.mdx [deleted file]
site/src/content/docs/forms/radio.mdx
site/src/content/docs/forms/switch.mdx

index ea2746449968cca7d765ae7a38f7111be2a7d310..a2a2024159cb3deb112d1e8e48a341fd673af309 100644 (file)
@@ -588,12 +588,6 @@ $mark-bg-dark:                      $yellow-800 !default;
 // Forms
 //
 
-// $form-select-indicator-color-dark:  $body-color-dark !default;
-// $form-select-indicator-dark:        url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'><path fill='none' stroke='#{$form-select-indicator-color-dark}' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/></svg>") !default;
-
-$form-switch-color-dark:            rgba($white, .25) !default;
-$form-switch-bg-image-dark:         url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'><circle r='3' fill='#{$form-switch-color-dark}'/></svg>") !default;
-
 // scss-docs-start form-validation-colors-dark
 $form-valid-color-dark:             $green-300 !default;
 $form-valid-border-color-dark:      $green-300 !default;
diff --git a/scss/forms/_check.scss b/scss/forms/_check.scss
new file mode 100644 (file)
index 0000000..bc88bea
--- /dev/null
@@ -0,0 +1,102 @@
+@use "../config" as *;
+@use "../colors" as *;
+@use "../variables" as *;
+@use "../functions" as *;
+@use "../vendor/rfs" as *;
+@use "../mixins/border-radius" as *;
+@use "../mixins/box-shadow" as *;
+@use "../mixins/color-mode" as *;
+@use "../mixins/focus-ring" as *;
+@use "../mixins/transition" as *;
+@use "form-variables" as *;
+
+// scss-docs-start check-variables
+$check-border-color: var(--#{$prefix}border-color) !default;
+$check-checked-bg: var(--#{$prefix}primary-base) !default;
+$check-checked-border-color: $check-checked-bg !default;
+$check-indeterminate-bg: var(--#{$prefix}primary-base) !default;
+$check-indeterminate-border-color: $check-indeterminate-bg !default;
+$check-disabled-bg: var(--#{$prefix}secondary-bg) !default;
+$check-disabled-border-color: $check-disabled-bg !default;
+$check-disabled-opacity: .65 !default;
+// scss-docs-end check-variables
+
+@layer forms {
+  b-checkgroup {
+    display: flex;
+    gap: var(--#{$prefix}gap, .5rem);
+    align-items: var(--#{$prefix}align-items, start);
+
+    .description {
+      color: var(--#{$prefix}secondary-text);
+    }
+  }
+
+  .check {
+    // scss-docs-start check-css-variables
+    --#{$prefix}check-bg: transparent;
+    --#{$prefix}check-border-color: #{$check-border-color};
+    --#{$prefix}check-checked-bg: #{$check-checked-bg};
+    --#{$prefix}check-checked-border-color: #{$check-checked-border-color};
+    --#{$prefix}check-indeterminate-bg: #{$check-indeterminate-bg};
+    --#{$prefix}check-indeterminate-border-color: #{$check-indeterminate-border-color};
+    --#{$prefix}check-disabled-bg: #{$check-disabled-bg};
+    --#{$prefix}check-disabled-border-color: #{$check-disabled-border-color};
+    --#{$prefix}check-disabled-opacity: #{$check-disabled-opacity};
+    // scss-docs-end check-css-variables
+
+    display: grid;
+    grid-template-columns: repeat(1, minmax(0, 1fr));
+    margin-block: .125rem;
+
+    :where(svg, input) {
+      flex-shrink: 0;
+      grid-row-start: 1;
+      grid-column-start: 1;
+      width: 1rem;
+      height: 1rem;
+    }
+
+    :where(input) {
+      appearance: none;
+      // later: maybe set a tertiary bg color?
+      background-color: var(--#{$prefix}check-bg);
+      border: 1px solid var(--#{$prefix}check-border-color);
+      // stylelint-disable-next-line property-disallowed-list
+      border-radius: .25em;
+    }
+
+    :where(input:checked, input:indeterminate) {
+      background-color: var(--#{$prefix}check-checked-bg);
+      border-color: var(--#{$prefix}check-checked-border-color);
+    }
+
+    &:has(input:checked) .checked,
+    &:has(input:indeterminate) .indeterminate {
+      display: block;
+      color: var(--#{$prefix}primary-contrast);
+      stroke: currentcolor;
+    }
+
+    &:has(input:disabled) {
+      --#{$prefix}check-bg: var(--#{$prefix}check-disabled-bg);
+
+      ~ label {
+        color: var(--#{$prefix}secondary-text);
+        cursor: default;
+      }
+    }
+
+    &:has(input:disabled:checked) {
+      opacity: var(--#{$prefix}check-disabled-opacity);
+    }
+
+    :where(svg) {
+      pointer-events: none;
+    }
+
+    :where(svg path) {
+      display: none;
+    }
+  }
+}
diff --git a/scss/forms/_form-check.scss b/scss/forms/_form-check.scss
deleted file mode 100644 (file)
index eed5ad3..0000000
+++ /dev/null
@@ -1,482 +0,0 @@
-@use "../config" as *;
-@use "../colors" as *;
-@use "../variables" as *;
-@use "../functions" as *;
-@use "../vendor/rfs" as *;
-@use "../mixins/border-radius" as *;
-@use "../mixins/box-shadow" as *;
-@use "../mixins/color-mode" as *;
-@use "../mixins/focus-ring" as *;
-@use "../mixins/transition" as *;
-@use "form-variables" as *;
-
-// scss-docs-start form-check-variables
-$form-check-input-width:                  1em !default;
-$form-check-min-height:                   $font-size-base * $line-height-base !default;
-$form-check-padding-start:                $form-check-input-width + .5em !default;
-$form-check-margin-bottom:                .125rem !default;
-$form-check-label-color:                  null !default;
-$form-check-label-cursor:                 null !default;
-$form-check-transition:                   null !default;
-
-$form-check-input-active-filter:          brightness(90%) !default;
-
-$form-check-input-bg:                     $input-bg !default;
-$form-check-input-border:                 var(--#{$prefix}border-width) solid var(--#{$prefix}border-color) !default;
-$form-check-input-border-radius:          .25em !default;
-$form-check-radio-border-radius:          50% !default;
-$form-check-input-focus-border:           $input-focus-border-color !default;
-$form-check-input-focus-box-shadow:       $focus-ring-box-shadow !default;
-
-$form-check-input-checked-color:          $component-active-color !default;
-$form-check-input-checked-bg-color:       $component-active-bg !default;
-$form-check-input-checked-border-color:   $form-check-input-checked-bg-color !default;
-$form-check-input-checked-bg-image:       url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'><path fill='none' stroke='#{$form-check-input-checked-color}' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='m6 10 3 3 6-6'/></svg>") !default;
-$form-check-radio-checked-bg-image:       url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'><circle r='2' fill='#{$form-check-input-checked-color}'/></svg>") !default;
-
-$form-check-input-indeterminate-color:          $component-active-color !default;
-$form-check-input-indeterminate-bg-color:       $component-active-bg !default;
-$form-check-input-indeterminate-border-color:   $form-check-input-indeterminate-bg-color !default;
-$form-check-input-indeterminate-bg-image:       url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'><path fill='none' stroke='#{$form-check-input-indeterminate-color}' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='M6 10h8'/></svg>") !default;
-
-$form-check-input-disabled-opacity:        .5 !default;
-$form-check-label-disabled-opacity:        $form-check-input-disabled-opacity !default;
-$form-check-btn-check-disabled-opacity:    $btn-disabled-opacity !default;
-
-$form-check-inline-margin-end:    1rem !default;
-// scss-docs-end form-check-variables
-
-// scss-docs-start form-switch-variables
-$form-switch-color:               rgba($black, .25) !default;
-$form-switch-width:               1.5em !default;
-$form-switch-padding-start:       $form-switch-width + .5em !default;
-$form-switch-bg-image:            url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'><circle r='3' fill='#{$form-switch-color}'/></svg>") !default;
-$form-switch-border-radius:       $form-switch-width !default;
-$form-switch-transition:          background-position .15s ease-in-out !default;
-
-$form-switch-focus-color:         $input-focus-border-color !default;
-$form-switch-focus-bg-image:      url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'><circle r='3' fill='#{$form-switch-focus-color}'/></svg>") !default;
-
-$form-switch-checked-color:       $component-active-color !default;
-$form-switch-checked-bg-image:    url("data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'><circle r='3' fill='#{$form-switch-checked-color}'/></svg>") !default;
-$form-switch-checked-bg-position: right center !default;
-// scss-docs-end form-switch-variables
-
-$check-border-color: var(--#{$prefix}border-color) !default;
-$check-checked-bg: var(--#{$prefix}primary-base) !default;
-$check-checked-border-color: $check-checked-bg !default;
-$check-indeterminate-bg: var(--#{$prefix}primary-base) !default;
-$check-indeterminate-border-color: $check-indeterminate-bg !default;
-$check-disabled-bg: var(--#{$prefix}secondary-bg) !default;
-$check-disabled-border-color: $check-disabled-bg !default;
-$check-disabled-opacity: .65 !default;
-
-@layer forms {
-  b-checkgroup,
-  b-radiogroup {
-    display: flex;
-    gap: var(--#{$prefix}gap, .5rem);
-    align-items: var(--#{$prefix}align-items, start);
-
-    .description {
-      color: var(--#{$prefix}secondary-text);
-    }
-  }
-
-  .check,
-  .radio {
-    --#{$prefix}check-bg: transparent;
-    --#{$prefix}check-border-color: #{$check-border-color};
-    --#{$prefix}check-checked-bg: #{$check-checked-bg};
-    --#{$prefix}check-checked-border-color: #{$check-checked-border-color};
-    --#{$prefix}check-indeterminate-bg: #{$check-indeterminate-bg};
-    --#{$prefix}check-indeterminate-border-color: #{$check-indeterminate-border-color};
-    --#{$prefix}check-disabled-bg: #{$check-disabled-bg};
-    --#{$prefix}check-disabled-border-color: #{$check-disabled-border-color};
-    --#{$prefix}check-disabled-opacity: #{$check-disabled-opacity};
-  }
-
-  .check {
-    display: grid;
-    grid-template-columns: repeat(1, minmax(0, 1fr));
-    margin-block: .125rem;
-
-    :where(svg, input) {
-      flex-shrink: 0;
-      grid-row-start: 1;
-      grid-column-start: 1;
-      width: 1rem;
-      height: 1rem;
-    }
-
-    :where(input) {
-      appearance: none;
-      // later: maybe set a tertiary bg color?
-      background-color: var(--#{$prefix}check-bg);
-      border: 1px solid var(--#{$prefix}check-border-color);
-      // stylelint-disable-next-line property-disallowed-list
-      border-radius: .25em;
-    }
-
-    :where(input:checked, input:indeterminate) {
-      background-color: var(--#{$prefix}check-checked-bg);
-      border-color: var(--#{$prefix}check-checked-border-color);
-    }
-
-    &:has(input:checked) .checked,
-    &:has(input:indeterminate) .indeterminate {
-      display: block;
-      color: var(--#{$prefix}primary-contrast);
-      stroke: currentcolor;
-    }
-
-    &:has(input:disabled) {
-      --#{$prefix}check-bg: var(--#{$prefix}check-disabled-bg);
-
-      ~ label {
-        color: var(--#{$prefix}secondary-text);
-        cursor: default;
-      }
-    }
-
-    &:has(input:disabled:checked) {
-      opacity: var(--#{$prefix}check-disabled-opacity);
-    }
-
-    :where(svg) {
-      pointer-events: none;
-    }
-
-    :where(svg path) {
-      display: none;
-    }
-  }
-
-  .radio {
-    position: relative;
-    flex-shrink: 0;
-    width: 1rem;
-    height: 1rem;
-    margin-block: .125rem;
-    appearance: none;
-    background-color: var(--#{$prefix}check-bg);
-    border: 1px solid var(--#{$prefix}check-border-color);
-    // stylelint-disable-next-line property-disallowed-list
-    border-radius: 50%;
-
-    &:checked {
-      color: var(--#{$prefix}primary-contrast);
-      background-color: var(--#{$prefix}check-checked-bg);
-      border-color: var(--#{$prefix}check-checked-border-color);
-
-      &::before {
-        position: absolute;
-        inset: .25rem;
-        content: "";
-        background-color: currentcolor;
-        // stylelint-disable-next-line property-disallowed-list
-        border-radius: 50%;
-      }
-    }
-
-    &:disabled {
-      --#{$prefix}check-bg: var(--#{$prefix}check-disabled-bg);
-
-      ~ label {
-        color: var(--#{$prefix}secondary-text);
-        cursor: default;
-      }
-    }
-  }
-
-  .switch {
-    --#{$prefix}switch-height: 1.25rem;
-    --#{$prefix}switch-width: calc(var(--#{$prefix}switch-height) * 1.5);
-    --#{$prefix}switch-padding: .0625rem;
-    --#{$prefix}switch-bg: var(--#{$prefix}secondary-bg);
-    --#{$prefix}switch-border-width: var(--#{$prefix}border-width);
-    --#{$prefix}switch-border-color: var(--#{$prefix}border-color);
-    --#{$prefix}switch-indicator-bg: var(--#{$prefix}white);
-    --#{$prefix}switch-checked-bg: var(--#{$prefix}primary-base);
-    --#{$prefix}switch-checked-indicator-bg: var(--#{$prefix}white);
-    --#{$prefix}switch-disabled-bg: var(--#{$prefix}secondary-bg);
-    --#{$prefix}switch-disabled-indicator-bg: var(--#{$prefix}secondary-text);
-
-    position: relative;
-    display: flex;
-    flex-shrink: 0;
-    align-items: stretch;
-    justify-content: flex-start;
-    width: var(--#{$prefix}switch-width);
-    height: var(--#{$prefix}switch-height);
-    padding: var(--#{$prefix}switch-padding);
-    background-color: var(--#{$prefix}switch-bg);
-    border: var(--#{$prefix}switch-border-width) solid var(--#{$prefix}switch-border-color);
-    // stylelint-disable-next-line property-disallowed-list
-    border-radius: 10rem;
-    box-shadow: inset 0 1px 2px rgba($black, .05);
-    // stylelint-disable-next-line property-disallowed-list
-    transition: .15s ease-in-out;
-    transition-property: padding-inline-start, background-color;
-
-    &::before {
-      flex-shrink: 0;
-      width: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2) - var(--#{$prefix}switch-border-width) * 2);
-      height: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2) - var(--#{$prefix}switch-border-width) * 2);
-      // width: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2));
-      // height: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2));
-      content: "";
-      background-color: var(--#{$prefix}switch-indicator-bg);
-      // stylelint-disable-next-line property-disallowed-list
-      border-radius: 50%;
-      box-shadow: 0 1px 2px rgba($black, .1);
-    }
-
-    input {
-      position: absolute;
-      inset: 0;
-      appearance: none;
-      background-color: transparent;
-    }
-
-    &:has(input:checked) {
-      padding-inline-start: calc(var(--#{$prefix}switch-height) / 2 + var(--#{$prefix}switch-padding));
-      background-color: var(--#{$prefix}primary-base);
-    }
-
-    &:has(input:disabled) {
-      --#{$prefix}switch-bg: var(--#{$prefix}switch-disabled-bg);
-      --#{$prefix}switch-indicator-bg: var(--#{$prefix}switch-disabled-indicator-bg);
-
-      &::before { opacity: .4; }
-
-      ~ label {
-        color: var(--#{$prefix}secondary-text);
-        cursor: default;
-      }
-    }
-  }
-  .switch-sm {
-    --#{$prefix}switch-height: 1em;
-  }
-  .switch-lg {
-    --#{$prefix}switch-height: 2em;
-  }
-
-  .select {
-    display: grid;
-    grid-template-columns: repeat(1, minmax(0, 1fr));
-
-    > select,
-    > svg {
-      grid-row-start: 1;
-      grid-column-start: 1;
-    }
-
-    > select {
-      padding: .5rem 3rem .5rem 1em;
-      font-size: 14px;
-      line-height: 20px;
-      appearance: none;
-      background-color: var(--#{$prefix}body-bg);
-      border: 1px solid var(--#{$prefix}border-color);
-      // stylelint-disable-next-line property-disallowed-list
-      border-radius: .5em;
-    }
-
-    > svg {
-      align-self: center;
-      justify-self: end;
-      width: 1rem;
-      height: 1rem;
-      margin-right: 1rem;
-    }
-  }
-
-  .form-check {
-    display: block;
-    min-height: $form-check-min-height;
-    padding-left: $form-check-padding-start;
-    margin-bottom: $form-check-margin-bottom;
-
-    .form-check-input {
-      float: left;
-      margin-left: $form-check-padding-start * -1;
-    }
-  }
-
-  .form-check-reverse {
-    padding-right: $form-check-padding-start;
-    padding-left: 0;
-    text-align: right;
-
-    .form-check-input {
-      float: right;
-      margin-right: $form-check-padding-start * -1;
-      margin-left: 0;
-    }
-  }
-
-  .form-check-input {
-    --#{$prefix}form-check-bg: #{$form-check-input-bg};
-
-    flex-shrink: 0;
-    width: $form-check-input-width;
-    height: $form-check-input-width;
-    margin-top: ($line-height-base - $form-check-input-width) * .5; // line-height minus check height
-    vertical-align: top;
-    appearance: none;
-    background-color: var(--#{$prefix}form-check-bg);
-    background-image: var(--#{$prefix}form-check-bg-image);
-    background-repeat: no-repeat;
-    background-position: center;
-    background-size: contain;
-    border: $form-check-input-border;
-    print-color-adjust: exact; // Keep themed appearance for print
-    @include transition($form-check-transition);
-
-    &[type="checkbox"] {
-      @include border-radius($form-check-input-border-radius);
-    }
-
-    &[type="radio"] {
-      // stylelint-disable-next-line property-disallowed-list
-      border-radius: $form-check-radio-border-radius;
-    }
-
-    &:active {
-      filter: $form-check-input-active-filter;
-    }
-
-    &:focus-visible {
-      border-color: $form-check-input-focus-border;
-      @include focus-ring(true);
-      --#{$prefix}focus-ring-offset: 1px;
-      // box-shadow: $form-check-input-focus-box-shadow;
-    }
-
-    &:checked {
-      background-color: $form-check-input-checked-bg-color;
-      border-color: $form-check-input-checked-border-color;
-
-      &[type="checkbox"] {
-        @if $enable-gradients {
-          --#{$prefix}form-check-bg-image: #{escape-svg($form-check-input-checked-bg-image)}, var(--#{$prefix}gradient);
-        } @else {
-          --#{$prefix}form-check-bg-image: #{escape-svg($form-check-input-checked-bg-image)};
-        }
-      }
-
-      &[type="radio"] {
-        @if $enable-gradients {
-          --#{$prefix}form-check-bg-image: #{escape-svg($form-check-radio-checked-bg-image)}, var(--#{$prefix}gradient);
-        } @else {
-          --#{$prefix}form-check-bg-image: #{escape-svg($form-check-radio-checked-bg-image)};
-        }
-      }
-    }
-
-    &[type="checkbox"]:indeterminate {
-      background-color: $form-check-input-indeterminate-bg-color;
-      border-color: $form-check-input-indeterminate-border-color;
-
-      @if $enable-gradients {
-        --#{$prefix}form-check-bg-image: #{escape-svg($form-check-input-indeterminate-bg-image)}, var(--#{$prefix}gradient);
-      } @else {
-        --#{$prefix}form-check-bg-image: #{escape-svg($form-check-input-indeterminate-bg-image)};
-      }
-    }
-
-    &:disabled {
-      pointer-events: none;
-      filter: none;
-      opacity: $form-check-input-disabled-opacity;
-    }
-
-    // Use disabled attribute in addition of :disabled pseudo-class
-    // See: https://github.com/twbs/bootstrap/issues/28247
-    &[disabled],
-    &:disabled {
-      ~ .form-check-label {
-        cursor: default;
-        opacity: $form-check-label-disabled-opacity;
-      }
-    }
-  }
-
-  .form-check-label {
-    color: $form-check-label-color;
-    cursor: $form-check-label-cursor;
-  }
-
-  //
-  // Switch
-  //
-
-  .form-switch {
-    padding-left: $form-switch-padding-start;
-
-    .form-check-input {
-      --#{$prefix}form-switch-bg: #{escape-svg($form-switch-bg-image)};
-
-      width: $form-switch-width;
-      margin-left: $form-switch-padding-start * -1;
-      background-image: var(--#{$prefix}form-switch-bg);
-      background-position: left center;
-      @include border-radius($form-switch-border-radius, 0);
-      @include transition($form-switch-transition);
-
-      &:focus {
-        --#{$prefix}form-switch-bg: #{escape-svg($form-switch-focus-bg-image)};
-      }
-
-      &:checked {
-        background-position: $form-switch-checked-bg-position;
-
-        @if $enable-gradients {
-          --#{$prefix}form-switch-bg: #{escape-svg($form-switch-checked-bg-image)}, var(--#{$prefix}gradient);
-        } @else {
-          --#{$prefix}form-switch-bg: #{escape-svg($form-switch-checked-bg-image)};
-        }
-      }
-    }
-
-    &.form-check-reverse {
-      padding-right: $form-switch-padding-start;
-      padding-left: 0;
-
-      .form-check-input {
-        margin-right: $form-switch-padding-start * -1;
-        margin-left: 0;
-      }
-    }
-  }
-
-  .form-check-inline {
-    display: inline-block;
-    margin-right: $form-check-inline-margin-end;
-  }
-
-  .btn-check {
-    position: absolute;
-    clip: rect(0, 0, 0, 0);
-    pointer-events: none;
-
-    &[disabled],
-    &:disabled {
-      + .btn {
-        pointer-events: none;
-        filter: none;
-        opacity: $form-check-btn-check-disabled-opacity;
-      }
-    }
-  }
-
-  @if $enable-dark-mode {
-    @include color-mode(dark) {
-      .form-switch .form-check-input:not(:checked):not(:focus) {
-        --#{$prefix}form-switch-bg: #{escape-svg($form-switch-bg-image-dark)};
-      }
-    }
-  }
-}
diff --git a/scss/forms/_radio.scss b/scss/forms/_radio.scss
new file mode 100644 (file)
index 0000000..af63fb1
--- /dev/null
@@ -0,0 +1,79 @@
+@use "../config" as *;
+@use "../colors" as *;
+@use "../variables" as *;
+@use "../functions" as *;
+@use "../vendor/rfs" as *;
+@use "../mixins/border-radius" as *;
+@use "../mixins/box-shadow" as *;
+@use "../mixins/color-mode" as *;
+@use "../mixins/focus-ring" as *;
+@use "../mixins/transition" as *;
+@use "form-variables" as *;
+
+// scss-docs-start radio-variables
+$radio-border-color: var(--#{$prefix}border-color) !default;
+$radio-checked-bg: var(--#{$prefix}primary-base) !default;
+$radio-checked-border-color: $radio-checked-bg !default;
+$radio-disabled-bg: var(--#{$prefix}secondary-bg) !default;
+$radio-disabled-border-color: $radio-disabled-bg !default;
+$radio-disabled-opacity: .65 !default;
+// scss-docs-end radio-variables
+
+@layer forms {
+  b-radiogroup {
+    display: flex;
+    gap: var(--#{$prefix}gap, .5rem);
+    align-items: var(--#{$prefix}align-items, start);
+
+    .description {
+      color: var(--#{$prefix}secondary-text);
+    }
+  }
+
+  .radio {
+    // scss-docs-start radio-css-variables
+    --#{$prefix}radio-bg: transparent;
+    --#{$prefix}radio-border-color: #{$radio-border-color};
+    --#{$prefix}radio-checked-bg: #{$radio-checked-bg};
+    --#{$prefix}radio-checked-border-color: #{$radio-checked-border-color};
+    --#{$prefix}radio-disabled-bg: #{$radio-disabled-bg};
+    --#{$prefix}radio-disabled-border-color: #{$radio-disabled-border-color};
+    --#{$prefix}radio-disabled-opacity: #{$radio-disabled-opacity};
+    // scss-docs-end radio-css-variables
+
+    position: relative;
+    flex-shrink: 0;
+    width: 1rem;
+    height: 1rem;
+    margin-block: .125rem;
+    appearance: none;
+    background-color: var(--#{$prefix}radio-bg);
+    border: 1px solid var(--#{$prefix}radio-border-color);
+    // stylelint-disable-next-line property-disallowed-list
+    border-radius: 50%;
+
+    &:checked {
+      color: var(--#{$prefix}primary-contrast);
+      background-color: var(--#{$prefix}radio-checked-bg);
+      border-color: var(--#{$prefix}radio-checked-border-color);
+
+      &::before {
+        position: absolute;
+        inset: .25rem;
+        content: "";
+        background-color: currentcolor;
+        // stylelint-disable-next-line property-disallowed-list
+        border-radius: 50%;
+      }
+    }
+
+    &:disabled {
+      --#{$prefix}radio-bg: var(--#{$prefix}radio-disabled-bg);
+
+      ~ label {
+        color: var(--#{$prefix}secondary-text);
+        cursor: default;
+      }
+    }
+  }
+}
index 67baf23c25fe46172bc9673212cf1a6db9c61f7f..457d170a4f38d2da37eeaeb95e8e848788b4cd15 100644 (file)
@@ -9,3 +9,81 @@
 @use "../mixins/focus-ring" as *;
 @use "../mixins/transition" as *;
 @use "form-variables" as *;
+
+@layer forms {
+  .switch {
+    // scss-docs-start switch-css-variables
+    --#{$prefix}switch-height: 1.25rem;
+    --#{$prefix}switch-width: calc(var(--#{$prefix}switch-height) * 1.5);
+    --#{$prefix}switch-padding: .0625rem;
+    --#{$prefix}switch-bg: var(--#{$prefix}secondary-bg);
+    --#{$prefix}switch-border-width: var(--#{$prefix}border-width);
+    --#{$prefix}switch-border-color: var(--#{$prefix}border-color);
+    --#{$prefix}switch-indicator-bg: var(--#{$prefix}white);
+    --#{$prefix}switch-checked-bg: var(--#{$prefix}primary-base);
+    --#{$prefix}switch-checked-indicator-bg: var(--#{$prefix}white);
+    --#{$prefix}switch-disabled-bg: var(--#{$prefix}secondary-bg);
+    --#{$prefix}switch-disabled-indicator-bg: var(--#{$prefix}secondary-text);
+    // scss-docs-end switch-css-variables
+
+    position: relative;
+    display: flex;
+    flex-shrink: 0;
+    align-items: stretch;
+    justify-content: flex-start;
+    width: var(--#{$prefix}switch-width);
+    height: var(--#{$prefix}switch-height);
+    padding: var(--#{$prefix}switch-padding);
+    background-color: var(--#{$prefix}switch-bg);
+    border: var(--#{$prefix}switch-border-width) solid var(--#{$prefix}switch-border-color);
+    // stylelint-disable-next-line property-disallowed-list
+    border-radius: 10rem;
+    box-shadow: inset 0 1px 2px rgba($black, .05);
+    // stylelint-disable-next-line property-disallowed-list
+    transition: .15s ease-in-out;
+    transition-property: padding-inline-start, background-color;
+
+    &::before {
+      flex-shrink: 0;
+      width: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2) - var(--#{$prefix}switch-border-width) * 2);
+      height: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2) - var(--#{$prefix}switch-border-width) * 2);
+      // width: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2));
+      // height: calc(var(--#{$prefix}switch-height) - calc(var(--#{$prefix}switch-padding) * 2));
+      content: "";
+      background-color: var(--#{$prefix}switch-indicator-bg);
+      // stylelint-disable-next-line property-disallowed-list
+      border-radius: 50%;
+      box-shadow: 0 1px 2px rgba($black, .1);
+    }
+
+    input {
+      position: absolute;
+      inset: 0;
+      appearance: none;
+      background-color: transparent;
+    }
+
+    &:has(input:checked) {
+      padding-inline-start: calc(var(--#{$prefix}switch-height) / 2 + var(--#{$prefix}switch-padding));
+      background-color: var(--#{$prefix}primary-base);
+    }
+
+    &:has(input:disabled) {
+      --#{$prefix}switch-bg: var(--#{$prefix}switch-disabled-bg);
+      --#{$prefix}switch-indicator-bg: var(--#{$prefix}switch-disabled-indicator-bg);
+
+      &::before { opacity: .4; }
+
+      ~ label {
+        color: var(--#{$prefix}secondary-text);
+        cursor: default;
+      }
+    }
+  }
+  .switch-sm {
+    --#{$prefix}switch-height: 1em;
+  }
+  .switch-lg {
+    --#{$prefix}switch-height: 2em;
+  }
+}
index a08cde8b396f223d51947494d0ae365492de89f7..0798f2003bf301ec8b8abf7719f86017145fe347 100644 (file)
@@ -1,7 +1,9 @@
 @forward "labels";
 @forward "form-text";
 @forward "form-control";
-@forward "form-check";
+@forward "check";
+@forward "radio";
+@forward "switch";
 @forward "form-range";
 @forward "floating-labels";
 @forward "input-group";
index a930467e19ce845d844ba8f821c1b41c8a155669..c6078956dbc08a195998f480d3c4a15ffb37b011 100644 (file)
@@ -60,7 +60,6 @@
   pages:
     - title: Overview
     - title: Form control
-    - title: Checks & radios
     - title: Checkbox
     - title: Radio
     - title: Switch
index cd3aac1de815e6cb9fb8eb9d6c4a1c28fa4f1910..409fed5195e369d8529fb050a1ddd15ae6c0093a 100644 (file)
@@ -103,3 +103,15 @@ Add the `disabled` attribute and the associated `<label>`s are automatically sty
     </div>
     <label for="checkDisabledChecked">Example new checkbox</label>
   </b-checkgroup>`} />
+
+## CSS
+
+### Variables
+
+CSS variables for the checkbox component are built on the Sass variables.
+
+<ScssDocs name="check-css-variables" file="scss/forms/_check.scss" />
+
+### Sass variables
+
+<ScssDocs name="check-variables" file="scss/forms/_check.scss" />
diff --git a/site/src/content/docs/forms/checks-radios.mdx b/site/src/content/docs/forms/checks-radios.mdx
deleted file mode 100644 (file)
index 52462ae..0000000
+++ /dev/null
@@ -1,311 +0,0 @@
----
-title: Checks and radios
-description: Create consistent cross-browser and cross-device checkboxes and radios with our completely rewritten checks component.
-aliases: "/docs/[[config:docs_version]]/forms/checks/"
-toc: true
----
-
-## Approach
-
-Browser default checkboxes and radios are replaced with the help of `.form-check`, a series of classes for both input types that improves the layout and behavior of their HTML elements, that provide greater customization and cross browser consistency. Checkboxes are for selecting one or several options in a list, while radios are for selecting one option from many.
-
-Structurally, our `<input>`s and `<label>`s are sibling elements as opposed to an `<input>` within a `<label>`. This is slightly more verbose as you must specify `id` and `for` attributes to relate the `<input>` and `<label>`. We use the sibling selector (`~`) for all our `<input>` states, like `:checked` or `:disabled`. When combined with the `.form-check-label` class, we can easily style the text for each item based on the `<input>`’s state.
-
-Our checks use custom Bootstrap icons to indicate checked or indeterminate states.
-
-## Checks
-
-<Example code={`<div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="checkDefault">
-    <label class="form-check-label" for="checkDefault">
-      Default checkbox
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="checkChecked" checked>
-    <label class="form-check-label" for="checkChecked">
-      Checked checkbox
-    </label>
-  </div>`} />
-
-### Indeterminate
-
-Checkboxes can utilize the `:indeterminate` pseudo class when manually set via JavaScript (there is no available HTML attribute for specifying it).
-
-<Example addStackblitzJs class="bd-example-indeterminate" code={`<div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="checkIndeterminate">
-    <label class="form-check-label" for="checkIndeterminate">
-      Indeterminate checkbox
-    </label>
-  </div>`} />
-
-### Disabled
-
-Add the `disabled` attribute and the associated `<label>`s are automatically styled to match with a lighter color to help indicate the input’s state.
-
-<Example addStackblitzJs class="bd-example-indeterminate" code={`<div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="checkIndeterminateDisabled" disabled>
-    <label class="form-check-label" for="checkIndeterminateDisabled">
-      Disabled indeterminate checkbox
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="checkDisabled" disabled>
-    <label class="form-check-label" for="checkDisabled">
-      Disabled checkbox
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="checkCheckedDisabled" checked disabled>
-    <label class="form-check-label" for="checkCheckedDisabled">
-      Disabled checked checkbox
-    </label>
-  </div>`} />
-
-## Radios
-
-<Example code={`<div class="form-check">
-    <input class="form-check-input" type="radio" name="radioDefault" id="radioDefault1">
-    <label class="form-check-label" for="radioDefault1">
-      Default radio
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="radio" name="radioDefault" id="radioDefault2" checked>
-    <label class="form-check-label" for="radioDefault2">
-      Default checked radio
-    </label>
-  </div>`} />
-
-### Disabled
-
-Add the `disabled` attribute and the associated `<label>`s are automatically styled to match with a lighter color to help indicate the input’s state.
-
-<Example code={`<div class="form-check">
-    <input class="form-check-input" type="radio" name="radioDisabled" id="radioDisabled" disabled>
-    <label class="form-check-label" for="radioDisabled">
-      Disabled radio
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="radio" name="radioDisabled" id="radioCheckedDisabled" checked disabled>
-    <label class="form-check-label" for="radioCheckedDisabled">
-      Disabled checked radio
-    </label>
-  </div>`} />
-
-## Switches
-
-A switch has the markup of a custom checkbox but uses the `.form-switch` class to render a toggle switch. Consider using `role="switch"` to more accurately convey the nature of the control to assistive technologies that support this role. In older assistive technologies, it will simply be announced as a regular checkbox as a fallback. Switches also support the `disabled` attribute.
-
-<Example code={`<div class="form-check form-switch">
-    <input class="form-check-input" type="checkbox" role="switch" id="switchCheckDefault">
-    <label class="form-check-label" for="switchCheckDefault">Default switch checkbox input</label>
-  </div>
-  <div class="form-check form-switch">
-    <input class="form-check-input" type="checkbox" role="switch" id="switchCheckChecked" checked>
-    <label class="form-check-label" for="switchCheckChecked">Checked switch checkbox input</label>
-  </div>
-  <div class="form-check form-switch">
-    <input class="form-check-input" type="checkbox" role="switch" id="switchCheckDisabled" disabled>
-    <label class="form-check-label" for="switchCheckDisabled">Disabled switch checkbox input</label>
-  </div>
-  <div class="form-check form-switch">
-    <input class="form-check-input" type="checkbox" role="switch" id="switchCheckCheckedDisabled" checked disabled>
-    <label class="form-check-label" for="switchCheckCheckedDisabled">Disabled checked switch checkbox input</label>
-  </div>`} />
-
-### Native switches
-
-Progressively enhance your switches for mobile Safari (iOS 17.4+) by adding a `switch` attribute to your input to enable haptic feedback when toggling switches, just like native iOS switches. There are no style changes attached to using this attribute in Bootstrap as all our switches use custom styles.
-
-<Example code={`<div class="form-check form-switch">
-    <input class="form-check-input" type="checkbox" value="" id="checkNativeSwitch" switch>
-    <label class="form-check-label" for="checkNativeSwitch">
-      Native switch haptics
-    </label>
-  </div>`} />
-
-Be sure to read more about [the switch attribute on the WebKit blog](https://webkit.org/blog/15054/an-html-switch-control/). Safari 17.4+ on macOS and iOS both have native-style switches in HTML while other browsers simply fall back to the standard checkbox appearance. Applying the attribute to a non-Bootstrap checkbox in more recent versions of Safari will render a native switch.
-
-## Default (stacked)
-
-By default, any number of checkboxes and radios that are immediate sibling will be vertically stacked and appropriately spaced with `.form-check`.
-
-<Example code={`<div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="defaultCheck1">
-    <label class="form-check-label" for="defaultCheck1">
-      Default checkbox
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="checkbox" value="" id="defaultCheck2" disabled>
-    <label class="form-check-label" for="defaultCheck2">
-      Disabled checkbox
-    </label>
-  </div>`} />
-
-<Example code={`<div class="form-check">
-    <input class="form-check-input" type="radio" name="exampleRadios" id="exampleRadios1" value="option1" checked>
-    <label class="form-check-label" for="exampleRadios1">
-      Default radio
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="radio" name="exampleRadios" id="exampleRadios2" value="option2">
-    <label class="form-check-label" for="exampleRadios2">
-      Second default radio
-    </label>
-  </div>
-  <div class="form-check">
-    <input class="form-check-input" type="radio" name="exampleRadios" id="exampleRadios3" value="option3" disabled>
-    <label class="form-check-label" for="exampleRadios3">
-      Disabled radio
-    </label>
-  </div>`} />
-
-## Inline
-
-Group checkboxes or radios on the same horizontal row by adding `.form-check-inline` to any `.form-check`.
-
-<Example code={`<div class="form-check form-check-inline">
-    <input class="form-check-input" type="checkbox" id="inlineCheckbox1" value="option1">
-    <label class="form-check-label" for="inlineCheckbox1">1</label>
-  </div>
-  <div class="form-check form-check-inline">
-    <input class="form-check-input" type="checkbox" id="inlineCheckbox2" value="option2">
-    <label class="form-check-label" for="inlineCheckbox2">2</label>
-  </div>
-  <div class="form-check form-check-inline">
-    <input class="form-check-input" type="checkbox" id="inlineCheckbox3" value="option3" disabled>
-    <label class="form-check-label" for="inlineCheckbox3">3 (disabled)</label>
-  </div>`} />
-
-<Example code={`<div class="form-check form-check-inline">
-    <input class="form-check-input" type="radio" name="inlineRadioOptions" id="inlineRadio1" value="option1">
-    <label class="form-check-label" for="inlineRadio1">1</label>
-  </div>
-  <div class="form-check form-check-inline">
-    <input class="form-check-input" type="radio" name="inlineRadioOptions" id="inlineRadio2" value="option2">
-    <label class="form-check-label" for="inlineRadio2">2</label>
-  </div>
-  <div class="form-check form-check-inline">
-    <input class="form-check-input" type="radio" name="inlineRadioOptions" id="inlineRadio3" value="option3" disabled>
-    <label class="form-check-label" for="inlineRadio3">3 (disabled)</label>
-  </div>`} />
-
-## Reverse
-
-Put your checkboxes, radios, and switches on the opposite side with the `.form-check-reverse` modifier class.
-
-<Example code={`<div class="form-check form-check-reverse">
-    <input class="form-check-input" type="checkbox" value="" id="reverseCheck1">
-    <label class="form-check-label" for="reverseCheck1">
-      Reverse checkbox
-    </label>
-  </div>
-  <div class="form-check form-check-reverse">
-    <input class="form-check-input" type="checkbox" value="" id="reverseCheck2" disabled>
-    <label class="form-check-label" for="reverseCheck2">
-      Disabled reverse checkbox
-    </label>
-  </div>
-
-  <div class="form-check form-switch form-check-reverse">
-    <input class="form-check-input" type="checkbox" id="switchCheckReverse">
-    <label class="form-check-label" for="switchCheckReverse">Reverse switch checkbox input</label>
-  </div>`} />
-
-## Without labels
-
-Omit the wrapping `.form-check` for checkboxes and radios that have no label text. Remember to still provide some form of accessible name for assistive technologies (for instance, using `aria-label`). See the [forms overview accessibility]([[docsref:/forms/overview/#accessibility]]) section for details.
-
-<Example code={`<div>
-    <input class="form-check-input" type="checkbox" id="checkboxNoLabel" value="" aria-label="...">
-  </div>
-
-  <div>
-    <input class="form-check-input" type="radio" name="radioNoLabel" id="radioNoLabel1" value="" aria-label="...">
-  </div>`} />
-
-## Toggle buttons
-
-Create button-like checkboxes and radio buttons by using `.btn` styles rather than `.form-check-label` on the `<label>` elements. These toggle buttons can further be grouped in a [button group]([[docsref:/components/button-group]]) if needed.
-
-### Checkbox toggle buttons
-
-<Example code={`<input type="checkbox" class="btn-check" id="btn-check" autocomplete="off">
-<label class="btn btn-primary" for="btn-check">Single toggle</label>
-
-<input type="checkbox" class="btn-check" id="btn-check-2" checked autocomplete="off">
-<label class="btn btn-primary" for="btn-check-2">Checked</label>
-
-<input type="checkbox" class="btn-check" id="btn-check-3" autocomplete="off" disabled>
-<label class="btn btn-primary" for="btn-check-3">Disabled</label>`} />
-
-<Example code={`<input type="checkbox" class="btn-check" id="btn-check-4" autocomplete="off">
-<label class="btn" for="btn-check-4">Single toggle</label>
-
-<input type="checkbox" class="btn-check" id="btn-check-5" checked autocomplete="off">
-<label class="btn" for="btn-check-5">Checked</label>
-
-<input type="checkbox" class="btn-check" id="btn-check-6" autocomplete="off" disabled>
-<label class="btn" for="btn-check-6">Disabled</label>`} />
-
-<Callout>
-Visually, these checkbox toggle buttons are identical to the [button plugin toggle buttons]([[docsref:/components/buttons#button-plugin]]). However, they are conveyed differently by assistive technologies: the checkbox toggles will be announced by screen readers as “checked“/“not checked“ (since, despite their appearance, they are fundamentally still checkboxes), whereas the button plugin toggle buttons will be announced as “button“/“button pressed“. The choice between these two approaches will depend on the type of toggle you are creating, and whether or not the toggle will make sense to users when announced as a checkbox or as an actual button.
-</Callout>
-
-### Radio toggle buttons
-
-<Example code={`<input type="radio" class="btn-check" name="options" id="option1" autocomplete="off" checked>
-<label class="btn btn-secondary" for="option1">Checked</label>
-
-<input type="radio" class="btn-check" name="options" id="option2" autocomplete="off">
-<label class="btn btn-secondary" for="option2">Radio</label>
-
-<input type="radio" class="btn-check" name="options" id="option3" autocomplete="off" disabled>
-<label class="btn btn-secondary" for="option3">Disabled</label>
-
-<input type="radio" class="btn-check" name="options" id="option4" autocomplete="off">
-<label class="btn btn-secondary" for="option4">Radio</label>`} />
-
-<Example code={`<input type="radio" class="btn-check" name="options-base" id="option5" autocomplete="off" checked>
-<label class="btn" for="option5">Checked</label>
-
-<input type="radio" class="btn-check" name="options-base" id="option6" autocomplete="off">
-<label class="btn" for="option6">Radio</label>
-
-<input type="radio" class="btn-check" name="options-base" id="option7" autocomplete="off" disabled>
-<label class="btn" for="option7">Disabled</label>
-
-<input type="radio" class="btn-check" name="options-base" id="option8" autocomplete="off">
-<label class="btn" for="option8">Radio</label>`} />
-
-### Outlined styles
-
-Different variants of `.btn`, such as the various outlined styles, are supported.
-
-<Example code={`<input type="checkbox" class="btn-check" id="btn-check-outlined" autocomplete="off">
-<label class="btn btn-outline-primary" for="btn-check-outlined">Single toggle</label><br>
-
-<input type="checkbox" class="btn-check" id="btn-check-2-outlined" checked autocomplete="off">
-<label class="btn btn-outline-secondary" for="btn-check-2-outlined">Checked</label><br>
-
-<input type="radio" class="btn-check" name="options-outlined" id="success-outlined" autocomplete="off" checked>
-<label class="btn btn-outline-success" for="success-outlined">Checked success radio</label>
-
-<input type="radio" class="btn-check" name="options-outlined" id="danger-outlined" autocomplete="off">
-<label class="btn btn-outline-danger" for="danger-outlined">Danger radio</label>`} />
-
-## CSS
-
-### Sass variables
-
-Variables for checks:
-
-<ScssDocs name="form-check-variables" file="scss/forms/_form-check.scss" />
-
-Variables for switches:
-
-<ScssDocs name="form-switch-variables" file="scss/forms/_form-check.scss" />
index bce0f3ddd5d3daf26890352478ad66f47a228a5a..7cec34b80aa55098b30e7d0e05292fc57042f647 100644 (file)
@@ -42,3 +42,15 @@ Add the `disabled` attribute and the associated `<label>`s are automatically sty
     <label for="radioDisabled">Example new radio</label>
   </b-radiogroup>
   `} />
+
+## CSS
+
+### Variables
+
+CSS variables for the radio component are built on the Sass variables.
+
+<ScssDocs name="radio-css-variables" file="scss/forms/_radio.scss" />
+
+### Sass variables
+
+<ScssDocs name="radio-variables" file="scss/forms/_radio.scss" />
index 1b74cdb05cda65028b0be1760599831a2fd10704..c98f424d8cdaf5a088360d1c9e9146b62492a4c8 100644 (file)
@@ -60,3 +60,9 @@ Add a size modifier class to make your switch appear smaller or larger.
   <div class="switch switch-lg">
     <input type="checkbox" value="" id="switchSizeLg" switch>
   </div>`} />
+
+## CSS
+
+### Variables
+
+<ScssDocs name="switch-css-variables" file="scss/forms/_switch.scss" />