// NetBox-specific Styles and Overrides. @use 'sass:map'; @use 'sass:math'; @import './sidenav'; @import './overrides'; @import './utilities'; @import './variables'; @each $color, $value in $theme-colors { // Override CSS values on each theme color. // Use Bootstrap's method of coloring alert links to appropriately color close buttons within // another colored element. // See: https://github.com/twbs/bootstrap/blob/2bdbb42dcf6bfb99b5e9e5444d9e64589eb8c08f/scss/_alert.scss#L50-L52 // See: https://github.com/twbs/bootstrap/blob/2bdbb42dcf6bfb99b5e9e5444d9e64589eb8c08f/scss/_close.scss#L12 $shifted-bg: shift-color($value, $alert-bg-scale); $shifted-color: shift-color($value, $alert-color-scale); @if (contrast-ratio($shifted-bg, $shifted-color) < $min-contrast-ratio) { $shifted-color: mix($value, color-contrast($shifted-bg), abs($alert-color-scale)); } $btn-close-bg: url("data:image/svg+xml,"); .bg-#{$color} button.btn-close { background: transparent escape-svg($btn-close-bg) center / $btn-close-width auto no-repeat; } .btn.btn-ghost-#{$color} { color: $value; &:hover { background-color: rgba($value, 0.12); } } // Use Bootstrap's method of coloring the .alert-link class automatically. // See: https://github.com/twbs/bootstrap/blob/2bdbb42dcf6bfb99b5e9e5444d9e64589eb8c08f/scss/_alert.scss#L50-L52 .alert.alert-#{$color}, .table-#{$color} { // Exclude buttons. a:not(.btn) { font-weight: $font-weight-bold; color: $shifted-color; } // Apply a border to buttons contained within colored elements, if they're not already a // bordered button class. .btn:not([class*='btn-outline']) { border-color: $gray-700; } } // Toasts required a slightly different approach because the background color isn't "shifted", // it's the direct theme color. .toast.bg-#{$color} { $shifted-color: shift-color($value, $alert-color-scale); @if (contrast-ratio($value, $shifted-color) < $min-contrast-ratio) { $shifted-color: mix($value, color-contrast($value), abs($alert-color-scale)); } a:not(.btn) { font-weight: $font-weight-bold; color: $shifted-color; } } // Use proper contrasting color foreground color for special components. .badge, .toast, .toast-header, .progress-bar { &.bg-#{$color} { color: color-contrast($value); } } } // Ensure progress bars (utilization graph) in tables aren't too narrow to display the percentage. table td > .progress { min-width: 6rem; } // Override Bootstrap form-control font-size when contained by .small element. .small .form-control { font-size: $font-size-sm; } // Automatically space out adjacent columns, but not within card bodies. :not(.card-body) > .col:not(:last-child):not(:only-child) { margin-bottom: $spacer; } .nav-mobile { display: none; flex-direction: column; align-items: center; justify-content: space-between; width: 100%; @include media-breakpoint-down(lg) { display: flex; } .nav-mobile-top { display: flex; align-items: center; justify-content: space-between; width: 100%; } } .card > .table.table-flush { margin-bottom: 0; overflow: hidden; border-bottom-right-radius: $card-border-radius; border-bottom-left-radius: $card-border-radius; thead th[scope='col'] { padding-top: map.get($spacers, 3); padding-bottom: map.get($spacers, 3); text-transform: uppercase; vertical-align: middle; background-color: $table-flush-header-bg; border-top: 1px solid $card-border-color; border-bottom-color: $card-border-color; } th, td { padding-right: map.get($spacers, 4) !important; padding-left: map.get($spacers, 4) !important; border-right: 0; border-left: 0; } tr[class] { border-color: $card-border-color !important; &:last-of-type { border-bottom-color: transparent !important; border-bottom-right-radius: $card-border-radius; border-bottom-left-radius: $card-border-radius; } } } // Primarily used for the new release notification, but could be used for other alerts as needed. // Wrap any alerts in .header-alert-container to ensure the layout is consistent. .header-alert-container { // Center-align the alert(s). display: flex; align-items: center; justify-content: center; // Apply the same spacing that's applied to the #content div's first child (.px-3). padding: 0 $spacer; // By default, alerts inside .header-alert-container should take up the full width. .alert { width: 100%; // Adjust the max-width for larger screens so there's not a big ugly blue blob taking up the // entire screen. @include media-breakpoint-up(md) { max-width: 75%; } @include media-breakpoint-up(lg) { max-width: 50%; } } } .alert { code { color: $gray-600; } } span.profile-button .dropdown-menu { right: 0; left: auto; display: block !important; margin-top: 0.5rem; box-shadow: $box-shadow; transition: opacity 0.2s ease-in-out; &:not(.show) { pointer-events: none; opacity: 0; } &.show { pointer-events: auto; opacity: 1; } } div#advanced-search-content div.card div.card-body div.col:not(:last-child) { margin-right: 1rem; } table { td { a { text-decoration: none; &:hover { text-decoration: underline; } } .dropdown { // Presence of 'overflow: scroll' on a table causes dropdowns to be improperly hidden when // opened. See: https://github.com/twbs/bootstrap/issues/24251 position: static; } } th { a, a:hover { color: $body-color; text-decoration: none; } } td, th { font-size: $font-size-sm; line-height: $line-height-sm; vertical-align: middle; &.min-width { width: 1%; } .form-check-input { // Ensure checkboxes aren't too small inside object tables. margin-top: 0.125em; font-size: $font-size-base; } .btn-sm { line-height: $line-height-xs; } p { // Remove spacing from paragraph elements within tables. margin-bottom: 0.5em; } p:last-child { margin-bottom: 0; } } th.asc > a::after { content: '\f0140'; font-family: 'Material Design Icons'; } th.desc > a::after { content: '\f0143'; font-family: 'Material Design Icons'; } &.table > :not(caption) > * > * { padding-right: $table-cell-padding-x-sm !important; padding-left: $table-cell-padding-x-sm !important; } &.object-list { th { font-size: $font-size-xs; line-height: $line-height-xs; vertical-align: bottom; } } &.attr-table { th { font-weight: normal; width: 25%; } } } div.title-container { display: flex; // On small screens, `flex-direction: column;` ensures the control buttons don't appear on the // same line as the title, but rather break to the next line. flex-direction: column; flex-wrap: wrap; justify-content: space-between; @include media-breakpoint-up(lg) { // On large screens, `flex-direction: row;` allows the control buttons to appear vertically // aligned with the title. flex-direction: row; } #content-title { display: flex; flex: 1 0; flex-direction: column; padding-bottom: map.get($spacers, 2); } } // Object list control buttons (Add/Clone/Import/Export) .controls { margin-bottom: map.get($spacers, 2); @media print { // Never print controls. Use this over the .noprint utility so plugin authors don't need to // remember to add the class. display: none !important; } // Each group of buttons. .control-group { display: flex; flex-wrap: wrap; // Left-align controls on mobile. justify-content: flex-start; // Right-align controls on larger screens. @include media-breakpoint-up(lg) { justify-content: flex-end; } > * { // Pad each control button. margin: map.get($spacers, 1); &:first-child { // Don't pad the left side of the first control button. margin-left: 0; } &:last-child { // Don't pad the right side of the last control button. margin-right: 0; } } } } .object-subtitle { display: block; font-size: $font-size-sm; color: $text-muted; @include media-breakpoint-up(md) { display: inline-block; } > span { display: block; // Hide the separator on small screens. &.separator { display: none; } &, &.separator { @include media-breakpoint-up(md) { display: inline-block; } } } } // Global Search nav.search { // Don't overtake dropdowns z-index: 999; justify-content: center; background-color: $navbar-light-color; .search-container { display: flex; width: 100%; @include media-breakpoint-down(lg) { display: none; } } // Search Input & Selected Object Value & Object Selector .input-group { // Selected Object .search-obj-selected { border-color: $input-border-color; } // Object Selector Dropdown Button .dropdown-toggle { // Generate the same styles as a regular Bootstrap button. @include button-variant($input-group-addon-bg, $input-border-color); margin-left: 0; font-weight: $input-group-addon-font-weight; line-height: $input-line-height; color: $input-group-addon-color; background-color: $input-group-addon-bg; border: $input-border-width solid $input-border-color; @include border-radius($input-border-radius); border-left: 1px solid var(--nbx-search-filter-border-left-color); &:focus { box-shadow: unset !important; } // Don't show the dropdown icon — the filter icon is basically the same thing. &:after { display: none; } } // Object Selector Dropdown Menu .search-obj-selector { // Limit the height and enable scrolling on mobile devices. max-height: 70vh; overflow-y: auto; .dropdown-item, .dropdown-header { font-size: $font-size-sm; } .dropdown-header { text-transform: uppercase; } } } } // Styles for the quicksearch and its clear button; // Overrides input-group styles and adds transition effects .quicksearch { input[type='search'] { border-radius: $border-radius !important; } button { margin-left: -32px !important; z-index: 100 !important; outline: none !important; border-radius: $border-radius !important; transition: visibility 0s, opacity 0.2s linear; } button :hover { opacity: 50%; transition: visibility 0s, opacity 0.1s linear; } } main.layout { display: flex; flex-wrap: nowrap; height: 100vh; height: -webkit-fill-available; max-height: 100vh; overflow-x: auto; overflow-y: hidden; // Override styles when printing. Fixes issue where only the first page is visible when printing. @media print { position: static !important; display: block !important; height: 100%; overflow-x: visible !important; overflow-y: visible !important; } } main.login-container { display: flex; flex-direction: column; align-items: center; justify-content: center; width: 100%; max-width: 100vw; height: calc(100vh - 4rem); padding-top: 40px; padding-bottom: 40px; + footer.footer button.color-mode-toggle { color: var(--nbx-color-mode-toggle-color); } } .footer { background-color: $tab-content-bg; padding: 0; .nav-link { padding: 0.5rem; } @include media-breakpoint-down(md) { // Pad the bottom of the footer on mobile devices to account for mobile browser controls. margin-bottom: 8rem; } } footer.login-footer { height: 4rem; margin-top: auto; .container-fluid { display: flex; justify-content: flex-end; padding: $container-padding-x $grid-gutter-width; } } h1.accordion-item-title, h2.accordion-item-title, h3.accordion-item-title, h4.accordion-item-title, h5.accordion-item-title, h6.accordion-item-title { padding: 0.25rem 0.5rem; font-size: $font-size-sm; font-weight: $font-weight-bold; color: var(--nbx-sidebar-title-color); text-transform: uppercase; } .form-login { width: 100%; max-width: 330px; padding: 15px; input:focus { z-index: 1; } input[type='text'] { margin-bottom: -1px; border-bottom-right-radius: 0; border-bottom-left-radius: 0; } input[type='password'] { margin-bottom: 10px; border-top-left-radius: 0; border-top-right-radius: 0; } .form-control { position: relative; box-sizing: border-box; height: auto; padding: 10px; font-size: 16px; } } .navbar { border-bottom: 1px solid $border-color; } .navbar-brand { padding-top: 0.75rem; padding-bottom: 0.75rem; font-size: 1rem; } nav.nav.nav-pills { .nav-item.nav-link { padding: 0.25rem 0.5rem; font-size: $font-size-sm; border-radius: $border-radius; &:hover { color: $accordion-button-active-color; background-color: $accordion-button-active-bg; } } } // Ensure the content container is full-height, and that the content block is also full height so // that the footer is always at the bottom. div.content-container { position: relative; display: flex; flex-direction: column; width: calc(100% - #{$sidenav-width-closed}); min-height: 100vh; overflow-x: hidden; overflow-y: auto; // Don't show an outline when the content container is focused. &:focus, &:focus-visible { outline: 0; } div.content { background-color: $tab-content-bg; flex: 1; } @include media-breakpoint-down(lg) { width: 100%; } // Make the content full-width when printing. @media print { width: 100% !important; margin-left: 0 !important; } } // Prevent scrolling of body content when nav menu is open on mobile. .sidebar.collapse.show ~ .content-container > .content { @media (max-width: map.get($grid-breakpoints, 'md')) { position: fixed; top: 0; left: 0; overflow-y: hidden; } } .tooltip { pointer-events: none; } span.color-label { display: block; width: 5rem; height: 1rem; padding: $badge-padding-y $badge-padding-x; border: 1px solid #303030; border-radius: $border-radius; box-shadow: $box-shadow-sm; } .badge a { color: inherit; } .btn { white-space: nowrap; } .card { box-shadow: $box-shadow-sm; .card-header { padding: $card-cap-padding-x; color: var(--nbx-body-color); border-bottom: none; } .card-header + .card-body { padding-top: 0; } .card-body.small .form-control, .card-body.small .form-select { font-size: $input-font-size-sm; } .card-divider { width: 100%; height: 1px; margin: $hr-margin-y 0; border-top: 1px solid $card-border-color; opacity: $hr-opacity; } // Remove shadow when printing. @media print { box-shadow: unset !important; } } .form-floating { position: relative; > .input-group > .form-control, > .input-group > .form-select { height: $form-floating-height; padding: $form-floating-padding-y $form-floating-padding-x; } > .input-group > label { position: absolute; top: 0; left: 0; height: 100%; // allow textareas padding: $form-floating-padding-y $form-floating-padding-x; pointer-events: none; border: $input-border-width solid transparent; // Required for aligning label's text with the input as it affects inner box model transform-origin: 0 0; @include transition($form-floating-transition); } > .input-group > .form-control { &::placeholder { color: transparent; } &:focus, &:not(:placeholder-shown) { padding-top: $form-floating-input-padding-t; padding-bottom: $form-floating-input-padding-b; } // Duplicated because `:-webkit-autofill` invalidates other selectors when grouped &:-webkit-autofill { padding-top: $form-floating-input-padding-t; padding-bottom: $form-floating-input-padding-b; } } > .input-group > .form-select, > .choices > .choices__inner, > .ss-main span.placeholder, // SlimSelect Single > .ss-main div.ss-values // SlimSelect Multiple { padding-top: $form-floating-input-padding-t; padding-bottom: $form-floating-input-padding-b; } > .input-group > .form-control:focus, > .input-group > .form-control:not(:placeholder-shown), > .input-group > .form-select, > .choices, > .ss-main { ~ label { opacity: $form-floating-label-opacity; transform: $form-floating-label-transform; z-index: 4; } } // Duplicated because `:-webkit-autofill` invalidates other selectors when grouped > .input-group > .form-control:-webkit-autofill { ~ label { z-index: 4; opacity: $form-floating-label-opacity; transform: $form-floating-label-transform; } } } .form-object-edit { margin: 0 auto; max-width: 800px; } textarea.form-control[rows='10'] { height: 18rem; } textarea.markdown, textarea.form-control[name='csv'] { font-family: $font-family-monospace; } .card:not(:only-of-type) { margin-bottom: $spacer; } .stat-btn { min-width: $spacer * 3; } nav.breadcrumb-container { width: fit-content; padding: $badge-padding-y $badge-padding-x; font-size: $font-size-sm; ol.breadcrumb { margin-bottom: 0; li.breadcrumb-item > a { text-decoration: none; } li.breadcrumb-item > a:hover { text-decoration: underline; } } } label.required { font-weight: $font-weight-bold; &:after { position: absolute; display: inline-block; margin: 0 0 0 2px; font-family: 'Material Design Icons'; font-size: 8px; font-style: normal; font-weight: $font-weight-medium; text-decoration: none; content: '\f06C4'; } } // Applied to containing element around table bulk-action buttons (bulk-edit, bulk-disconnect // bulk-delete, etc). Each usage of .bulk-buttons needs to have groups of buttons wrapped with // .bulk-button-group so that proper spacing is applied (even if there is only one group). div.bulk-buttons { display: flex; justify-content: space-between; margin: math.div($spacer, 2) 0; // Each group of buttons needs to be contained separately for alignment purposes. This way, you // can put some buttons in a group that aligns left, and other buttons in a group that aligns // right. > div.bulk-button-group { display: flex; flex-wrap: wrap; // For small screens: don't fill the available space (thereby expanding the size of the button). align-items: flex-start; &:first-of-type:not(:last-of-type) { // If there are multiple bulk button groups and this is the first, the first button in the // group should *not* have left spacing applied, so the button group aligns with the rest // of the page elements. > *:first-child { margin-left: 0; } } &:last-of-type:not(:first-of-type) { // If there are multiple bulk button groups and this is the last, the last button in the // group should *not* have right spacing applied, so the button group aligns with the rest // of the page elements. > *:last-child { margin-right: 0; } } // However, the rest of the buttons should have spacing applied in all directions. > * { margin: math.div($spacer, 4); } } } table tbody { @each $color, $value in $theme-colors { tr.#{$color} { background-color: rgba($value, 0.15); border-color: $gray-500; } } } // Style objects with statuses/roles within a table. As of implementation, used for IP addresses // assigned to interfaces. table .table-badge-group { .table-badge { display: block; width: min-content; font-size: $font-size-sm; font-weight: $font-weight-base; &:not(.badge) { // Apply badge horizontal padding so that IP addresses *not* within a badge appear aligned // with IP addresses that *are* within a badge. padding: 0 $badge-padding-x; } &.badge:not(:last-of-type):not(:only-child) { margin-bottom: map.get($spacers, 1); } } } pre.change-data { padding-right: 0; padding-left: 0; > span { display: block; padding-right: $spacer; padding-left: $spacer; &.added { background-color: var(--nbx-change-added); } &.removed { background-color: var(--nbx-change-removed); } } } pre.change-diff { border-color: transparent; &.change-removed { background-color: var(--nbx-change-removed); } &.change-added { background-color: var(--nbx-change-added); } } div.card-overlay { position: absolute; display: flex; align-items: center; justify-content: center; width: 100%; height: 100%; background-color: rgba($white, 0.75); border-radius: $border-radius; > div.spinner-border { width: 6rem; height: 6rem; color: $secondary; } } .table-controls { display: flex; @include media-breakpoint-up(md) { // `!important` needed because of inherited margin-bottom from `.col` margin-top: 0 !important; margin-bottom: 0 !important; } .table-configure { justify-content: flex-start; @include media-breakpoint-up(md) { justify-content: flex-end; } } .form-switch.form-check-inline { flex: 1 0 auto; font-size: $font-size-sm; } } // Tabbed content .nav-tabs { background-color: $body-bg; .nav-link { &:hover { // Don't show a bottom-border on a hovered nav link because it overlaps with the .nav-tab border. border-bottom-color: transparent; } &.active { // Set the background-color of an active tab to the same color as the .tab-content // background-color so it visually indicates which tab is open. background-color: $tab-content-bg; border-bottom-color: $tab-content-bg; // Move the active tab down 1px to overtake the .nav-tabs element's border, but only for that // tab. This is an ugly hack, but it works. transform: translateY(1px); } } } .tab-content { display: flex; flex-direction: column; padding: $spacer; } // Override masonry-layout styles when printing. .masonry { @media print { position: static !important; display: block !important; height: unset !important; } .masonry-item { @media print { position: static !important; top: unset !important; left: unset !important; display: block !important; } } } // Object hierarchy indicators. .record-depth { display: inline; font-size: $font-size-base; user-select: none; opacity: 0.33; // Add spacing to the last or only dot. span:only-of-type, span:last-of-type { margin-right: map.get($spacers, 1); } } // Remove the max-width from image preview popovers as this is controlled on the image element. .popover.image-preview-popover { max-width: unset; } /* Rendered Markdown */ .rendered-markdown table { width: 100%; } .rendered-markdown th { border-bottom: 2px solid #dddddd; padding: 8px; } .rendered-markdown td { border-top: 1px solid #dddddd; padding: 8px; } th[align="left"] { text-align: left; } th[align="center"] { text-align: center; } th[align="right"] { text-align: right; } /* Markdown widget */ .markdown-widget { .nav-link { border-bottom: 0; &.active { background-color: var(--nbx-body-bg); } } .nav-tabs { background-color: var(--nbx-pre-bg); } } // Preformatted text blocks td pre { margin-bottom: 0; } pre.block { padding: $spacer; background-color: var(--nbx-pre-bg); border: 1px solid var(--nbx-pre-border-color); border-radius: $border-radius; } #django-messages { position: fixed; right: $spacer; bottom: 0; margin: $spacer; } // Page-specific styles. html { // Shade the home page content background-color. &[data-netbox-url-name='home'] { .content-container, .search { background-color: $gray-100 !important; } &[data-netbox-color-mode='dark'] { .content-container, .search { background-color: $darkest !important; } } } // Don't show the django-messages toasts on the login screen in favor of the alert component. &[data-netbox-url-name='login'] { #django-messages { display: none; } } // Apply row colours to interface lists &[data-netbox-url-name='device_interfaces'] { tr[data-cable-status=connected] { background-color: rgba(map.get($theme-colors, "green"), 0.15); } tr[data-cable-status=planned] { background-color: rgba(map.get($theme-colors, "blue"), 0.15); } tr[data-cable-status=decommissioning] { background-color: rgba(map.get($theme-colors, "yellow"), 0.15); } tr[data-mark-connected=true] { background-color: rgba(map.get($theme-colors, "success"), 0.15); } tr[data-virtual=true] { background-color: rgba(map.get($theme-colors, "primary"), 0.15); } tr[data-enabled=disabled] { background-color: rgba(map.get($theme-colors, "danger"), 0.15); } // Only show the correct button depending on the cable status tr[data-cable-status=connected] button.mark-installed { display: none; } tr:not([data-cable-status=connected]) button.mark-planned { display: none; } } }