├── netbox_themes ├── api │ ├── __init__.py │ └── serializers.py ├── migrations │ ├── __init__.py │ ├── 0002_alter_theme_base_theme.py │ └── 0001_initial_netbox_themes.py ├── exceptions.py ├── navigation.py ├── tables.py ├── templates │ ├── netbox_themes │ │ └── theme.html │ └── base │ │ └── base.html ├── views.py ├── __init__.py ├── urls.py ├── templatetags │ └── netbox_themes_template_filters.py ├── models.py └── forms.py ├── media ├── doom.png ├── excel.png ├── greggs.png ├── pokemon.png ├── sunrise.png ├── christmas.png ├── startrek.png └── create-theme.png ├── MANIFEST.in ├── .gitignore ├── samples ├── sunrise.css ├── christmas.css ├── greggs.css ├── doom.css └── excel.css ├── setup.py ├── pyproject.toml ├── LICENSE └── README.md /netbox_themes/api/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /netbox_themes/migrations/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /media/doom.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/doom.png -------------------------------------------------------------------------------- /media/excel.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/excel.png -------------------------------------------------------------------------------- /media/greggs.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/greggs.png -------------------------------------------------------------------------------- /media/pokemon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/pokemon.png -------------------------------------------------------------------------------- /media/sunrise.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/sunrise.png -------------------------------------------------------------------------------- /media/christmas.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/christmas.png -------------------------------------------------------------------------------- /media/startrek.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/startrek.png -------------------------------------------------------------------------------- /media/create-theme.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/rcrawf/netbox-plugin-themes/HEAD/media/create-theme.png -------------------------------------------------------------------------------- /MANIFEST.in: -------------------------------------------------------------------------------- 1 | recursive-include netbox_themes/templates * 2 | recursive-include netbox_themes/templatetags * 3 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.egginfo 2 | netbox_themes.egg-info 3 | __pycache__ 4 | netbox_plugin_themes.egg-info/ 5 | dist 6 | venv 7 | -------------------------------------------------------------------------------- /netbox_themes/exceptions.py: -------------------------------------------------------------------------------- 1 | class ThemeDeleteError(Exception): 2 | pass 3 | 4 | class ThemeEditError(Exception): 5 | pass 6 | -------------------------------------------------------------------------------- /netbox_themes/navigation.py: -------------------------------------------------------------------------------- 1 | from netbox.plugins import PluginMenuItem 2 | 3 | menu_items = ( 4 | PluginMenuItem( 5 | link='plugins:netbox_themes:theme_list', 6 | link_text='Themes Manager' 7 | ), 8 | ) 9 | -------------------------------------------------------------------------------- /samples/sunrise.css: -------------------------------------------------------------------------------- 1 | .navbar { 2 | background: linear-gradient(#15e7eb, #fcba03) !important; 3 | } 4 | 5 | .netbox-edition::before { 6 | content: "🌞"; 7 | } 8 | 9 | .netbox-edition::after { 10 | content: "🌞"; 11 | } 12 | 13 | .mdi-magnify::before { 14 | content: "🌞"; 15 | } 16 | -------------------------------------------------------------------------------- /setup.py: -------------------------------------------------------------------------------- 1 | from setuptools import find_packages, setup 2 | 3 | setup( 4 | name='netbox-plugin-themes', 5 | version='0.5.1', 6 | description='Custom CSS themes for NetBox', 7 | install_requires=[], 8 | packages=find_packages(), 9 | include_package_data=True, 10 | zip_safe=False, 11 | ) 12 | -------------------------------------------------------------------------------- /netbox_themes/tables.py: -------------------------------------------------------------------------------- 1 | import django_tables2 as tables 2 | 3 | from netbox.tables import NetBoxTable 4 | from .models import Theme 5 | 6 | class ThemeTable(NetBoxTable): 7 | 8 | class Meta(NetBoxTable.Meta): 9 | model = Theme 10 | fields = ('id', 'name', 'base_theme', 'active') 11 | default_columns = ('id', 'name', 'base_theme', 'active') 12 | -------------------------------------------------------------------------------- /netbox_themes/templates/netbox_themes/theme.html: -------------------------------------------------------------------------------- 1 | {% extends 'generic/object.html' %} 2 | {% load netbox_themes_template_filters %} 3 | 4 | {% block content %} 5 |
6 |
7 |
8 |
CSS Data
9 |
10 |
{{ object.css_data|base64decode|linebreaks }}
11 |
12 |
13 |
14 |
15 | {% endblock content %} 16 | -------------------------------------------------------------------------------- /netbox_themes/migrations/0002_alter_theme_base_theme.py: -------------------------------------------------------------------------------- 1 | # Generated by Django 5.1.5 on 2025-01-19 04:17 2 | 3 | from django.db import migrations, models 4 | 5 | 6 | class Migration(migrations.Migration): 7 | 8 | dependencies = [ 9 | ('netbox_themes', '0001_initial_netbox_themes'), 10 | ] 11 | 12 | operations = [ 13 | migrations.AlterField( 14 | model_name='theme', 15 | name='base_theme', 16 | field=models.CharField(default='toggle', max_length=10), 17 | ), 18 | ] 19 | -------------------------------------------------------------------------------- /netbox_themes/views.py: -------------------------------------------------------------------------------- 1 | from netbox.views import generic 2 | from . import forms, models, tables 3 | 4 | class ThemeView(generic.ObjectView): 5 | queryset = models.Theme.objects.all() 6 | 7 | class ThemeListView(generic.ObjectListView): 8 | queryset = models.Theme.objects.all() 9 | table = tables.ThemeTable 10 | 11 | class ThemeEditView(generic.ObjectEditView): 12 | queryset = models.Theme.objects.all() 13 | form = forms.ThemeForm 14 | 15 | class ThemeDeleteView(generic.ObjectDeleteView): 16 | queryset = models.Theme.objects.all() 17 | -------------------------------------------------------------------------------- /netbox_themes/api/serializers.py: -------------------------------------------------------------------------------- 1 | from rest_framework import serializers 2 | 3 | from netbox.api.serializers import NetBoxModelSerializer 4 | from ..models import Theme 5 | 6 | class ThemeSerializer(NetBoxModelSerializer): 7 | 8 | class ThemeSerializer(NetBoxModelSerializer): 9 | url = serializers.HyperlinkedIdentityField( 10 | view_name='plugins-api:netbox_themes-api:theme-detail' 11 | ) 12 | 13 | class Meta: 14 | model = Theme 15 | fields = ( 16 | 'id', 'name', 'active', 'base_theme', 'css_data', 17 | ) 18 | -------------------------------------------------------------------------------- /netbox_themes/__init__.py: -------------------------------------------------------------------------------- 1 | from netbox.plugins import PluginConfig 2 | from pathlib import Path 3 | from django.conf import settings 4 | 5 | 6 | class NetBoxThemes(PluginConfig): 7 | name = 'netbox_themes' 8 | verbose_name = 'NetBox Themes' 9 | description = 'Manage CSS themes for NetBox' 10 | version = '0.5.1' 11 | base_url = 'themes' 12 | 13 | def ready(self): 14 | plugin_templates_dir = Path(__file__).resolve().parent / 'templates/' 15 | if str(plugin_templates_dir) not in settings.TEMPLATES[0]['DIRS']: 16 | settings.TEMPLATES[0]['DIRS'].insert(0, str(plugin_templates_dir)) 17 | super().ready() 18 | 19 | config = NetBoxThemes 20 | -------------------------------------------------------------------------------- /pyproject.toml: -------------------------------------------------------------------------------- 1 | [build-system] 2 | requires = ["setuptools>=61.0"] 3 | build-backend = "setuptools.build_meta" 4 | 5 | [project] 6 | name = "netbox-plugin-themes" 7 | version = "0.5.1" 8 | authors = [ 9 | { name="Netbox Themes Plugin", email="null@example.com" }, 10 | ] 11 | description = "A Netbox plugin to customise the look of Netbox using CSS themes" 12 | readme = "README.md" 13 | requires-python = ">=3.9" 14 | classifiers = [ 15 | "Programming Language :: Python :: 3", 16 | "License :: OSI Approved :: MIT License", 17 | "Operating System :: OS Independent", 18 | ] 19 | 20 | [project.urls] 21 | Homepage = "https://github.com/rcrawf/netbox-plugin-themes" 22 | Issues = "https://github.com/rcrawf/netbox-plugin-themes/issues" 23 | -------------------------------------------------------------------------------- /netbox_themes/urls.py: -------------------------------------------------------------------------------- 1 | from django.urls import path 2 | from . import models, views 3 | 4 | from netbox.views.generic import ObjectChangeLogView 5 | 6 | app_name = "theme" 7 | 8 | urlpatterns = ( 9 | path('theme/', views.ThemeListView.as_view(), name='theme_list'), 10 | path('theme/add/', views.ThemeEditView.as_view(), name='theme_add'), 11 | path('theme//', views.ThemeView.as_view(), name='theme'), 12 | path('theme//edit/', views.ThemeEditView.as_view(), name='theme_edit'), 13 | path('theme//delete/', views.ThemeDeleteView.as_view(), name='theme_delete'), 14 | 15 | path('theme//changelog/', ObjectChangeLogView.as_view(), name='theme_changelog', kwargs={ 16 | 'model': models.Theme 17 | }), 18 | ) 19 | -------------------------------------------------------------------------------- /samples/christmas.css: -------------------------------------------------------------------------------- 1 | body { 2 | font-family: mono !important; 3 | } 4 | 5 | .netbox-edition::before { 6 | content: "🎄"; 7 | } 8 | 9 | .netbox-edition::after { 10 | content: "🎄"; 11 | } 12 | 13 | body[data-bs-theme="dark"] .navbar-vertical.navbar-expand-lg .text-secondary{ 14 | color: green !important; 15 | } 16 | 17 | body[data-bs-theme="dark"] .navbar-vertical.navbar-expand-lg .text-secondary::before{ 18 | content: "🎄"; 19 | } 20 | 21 | body[data-bs-theme="dark"] .navbar-vertical.navbar-expand-lg .nav-item.dropdown.active::after { 22 | border-color: red !important; 23 | } 24 | 25 | .mdi::before { 26 | content: "🎁" !important; 27 | } 28 | 29 | 30 | .mdi-magnify::before { 31 | content: "🎅" !important; 32 | } 33 | 34 | .mdi-lightbulb-on::before { 35 | content: "⛄" !important; 36 | } 37 | 38 | .mdi-plus-thick::before { 39 | content: "🍷" !important; 40 | } 41 | 42 | .mdi-download::before { 43 | content: "🎄" !important; 44 | } 45 | -------------------------------------------------------------------------------- /netbox_themes/templatetags/netbox_themes_template_filters.py: -------------------------------------------------------------------------------- 1 | from base64 import b64decode 2 | 3 | from django import template 4 | from django.utils.safestring import mark_safe 5 | 6 | from ..models import Theme 7 | 8 | register = template.Library() 9 | 10 | @register.filter(name="base64decode") 11 | def base64decode(value): 12 | """ Decode base64 content """ 13 | return mark_safe(b64decode(value).decode()) 14 | 15 | @register.filter(name="return_custom_css") 16 | def return_custom_css(value): 17 | """ Return a custom theme CSS. """ 18 | theme = Theme.objects.get(active=True) 19 | return mark_safe(b64decode(theme.css_data).decode()) 20 | 21 | @register.filter(name="return_base_theme") 22 | def return_base_theme(value): 23 | """ Return base theme name. """ 24 | theme = Theme.objects.get(active=True) 25 | return theme.base_theme 26 | 27 | @register.filter(name="theme_name") 28 | def theme_name(value): 29 | """ Theme name """ 30 | theme = Theme.objects.get(active=True) 31 | return theme.name 32 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Permission is hereby granted, free of charge, to any person obtaining a copy 2 | of this software and associated documentation files (the "Software"), to deal 3 | in the Software without restriction, including without limitation the rights 4 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 5 | copies of the Software, and to permit persons to whom the Software is 6 | furnished to do so, subject to the following conditions: 7 | 8 | The above copyright notice and this permission notice shall be included in all 9 | copies or substantial portions of the Software. 10 | 11 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 12 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 13 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 14 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 15 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 16 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 17 | SOFTWARE. 18 | -------------------------------------------------------------------------------- /netbox_themes/templates/base/base.html: -------------------------------------------------------------------------------- 1 | {% extends "base/base.html" %} 2 | {% load static %} 3 | {% load netbox_themes_template_filters %} 4 | 5 | {% block head %} 6 | {{ block.super }} 7 | 10 | {% endblock %} 11 | {% block javascript %} 12 | {{ block.super }} 13 | 30 | {% endblock %} 31 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # NetBox Themes Plugin 2 | 3 | Plugin to inject custom CSS themes into NetBox. 4 | 5 | Tested on Netbox >= 4.1.0. 6 | 7 | > [!WARNING] 8 | > This is experimental so use at your own risk 🌞 9 | 10 | # Installation 11 | 12 | Install the plugin: 13 | 14 | ``` 15 | pip install netbox-plugin-themes 16 | ``` 17 | 18 | Add to Netbox configuration: 19 | 20 | ``` 21 | # configuration.py 22 | PLUGINS = [ 23 | 'netbox_themes', 24 | ] 25 | ``` 26 | 27 | Run the migrations and generate static content: 28 | 29 | ``` 30 | manage.py migrate 31 | manage.py collectstatic 32 | ``` 33 | 34 | Then restart the Netbox service. 35 | 36 | # Create a theme! 37 | 38 | This is a basic example to change to a monospace font: 39 | 40 | ![Create a theme](media/create-theme.png?raw=true "Create a theme") 41 | 42 | Or take a look in the `samples` directory: 43 | 44 | ![Doom](media/doom.png?raw=true "Doom theme") 45 | 46 | ![Star Trek](media/startrek.png?raw=true "Star Trek theme") 47 | 48 | ![Pokemon](media/pokemon.png?raw=true "Pokemon theme") 49 | 50 | ![Greggs](media/greggs.png?raw=true "Greggs theme") 51 | 52 | ![Excel](media/excel.png?raw=true "Excel theme") 53 | 54 | ## Build instructions 55 | 56 | ``` 57 | python3 -m pip install --upgrade build 58 | python3 -m build 59 | ``` 60 | -------------------------------------------------------------------------------- /netbox_themes/models.py: -------------------------------------------------------------------------------- 1 | from django.contrib.postgres.fields import ArrayField 2 | from django.db import models 3 | from netbox.models import NetBoxModel 4 | from django.urls import reverse 5 | 6 | from .exceptions import ThemeDeleteError 7 | 8 | BASE_THEME_CHOICES = [ 9 | ('dark', 'Dark'), 10 | ('light', 'Light'), 11 | ('toggle', 'Inherit toggle'), 12 | ] 13 | 14 | class Theme(NetBoxModel): 15 | name = models.CharField( 16 | max_length=100 17 | ) 18 | active = models.BooleanField( 19 | default=False 20 | ) 21 | css_data = models.TextField( 22 | default='' 23 | ) 24 | base_theme = models.CharField( 25 | max_length=10, 26 | choices=BASE_THEME_CHOICES, 27 | default='toggle', 28 | ) 29 | 30 | class Meta: 31 | ordering = ('name',) 32 | 33 | def __str__(self): 34 | return self.name 35 | 36 | def get_absolute_url(self): 37 | return reverse('plugins:netbox_themes:theme', args=[self.id]) 38 | 39 | def delete(self, *args, **kwargs): 40 | if self.active == True: 41 | raise ThemeDeleteError("Cannot delete active theme, deactivate it before deleting") 42 | if self.name == "default": 43 | raise ThemeDeleteError("Cannot delete default theme") 44 | super().delete(*args, **kwargs) 45 | -------------------------------------------------------------------------------- /netbox_themes/migrations/0001_initial_netbox_themes.py: -------------------------------------------------------------------------------- 1 | # Generated by Django 5.0.9 on 2024-10-12 15:42 2 | 3 | import taggit.managers 4 | import utilities.json 5 | from django.db import migrations, models 6 | 7 | from netbox_themes.models import Theme 8 | 9 | def insert_initial_data(apps, schema_editor): 10 | """ Create default theme. """ 11 | default_theme = Theme(name="default", active=True, css_data='e30K') 12 | default_theme.save() 13 | 14 | class Migration(migrations.Migration): 15 | 16 | initial = True 17 | 18 | dependencies = [ 19 | ('extras', '0121_customfield_related_object_filter'), 20 | ] 21 | 22 | operations = [ 23 | migrations.CreateModel( 24 | name='Theme', 25 | fields=[ 26 | ('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False)), 27 | ('created', models.DateTimeField(auto_now_add=True, null=True)), 28 | ('last_updated', models.DateTimeField(auto_now=True, null=True)), 29 | ('custom_field_data', models.JSONField(blank=True, default=dict, encoder=utilities.json.CustomFieldJSONEncoder)), 30 | ('name', models.CharField(max_length=100)), 31 | ('css_data', models.TextField(default='')), 32 | ('base_theme', models.CharField(default='dark', max_length=10)), 33 | ('active', models.BooleanField(default=False)), 34 | ('tags', taggit.managers.TaggableManager(through='extras.TaggedItem', to='extras.Tag')), 35 | ], 36 | options={ 37 | 'ordering': ('name',), 38 | }, 39 | ), 40 | migrations.RunPython( 41 | code=insert_initial_data, 42 | ), 43 | ] 44 | 45 | -------------------------------------------------------------------------------- /netbox_themes/forms.py: -------------------------------------------------------------------------------- 1 | import base64 2 | import re 3 | 4 | from django import forms 5 | 6 | from netbox.forms import NetBoxModelForm 7 | from .exceptions import ThemeEditError 8 | from .models import Theme 9 | 10 | class ThemeForm(NetBoxModelForm): 11 | 12 | # Form view customizations 13 | css_data = forms.CharField( 14 | label="CSS data", 15 | widget=forms.Textarea( 16 | attrs={ 17 | 'style': 'font-family: monospace;', 18 | } 19 | ) 20 | ) 21 | 22 | 23 | class Meta: 24 | model = Theme 25 | fields = ('name', 'active', 'base_theme', 'css_data') 26 | 27 | def __init__(self, *args, **kwargs): 28 | super().__init__(*args, **kwargs) 29 | 30 | # Check if the theme being edited is the "default" theme 31 | if self.instance.name == 'default': 32 | self.fields['name'].widget.attrs['readonly'] = True 33 | self.fields['css_data'].widget.attrs['readonly'] = True 34 | 35 | # Decode the base64 css_data 36 | if self.instance and self.instance.pk: 37 | try: 38 | # Decode the Base64 encoded css_data and set the initial value 39 | decoded_css_data = base64.b64decode(self.instance.css_data.encode()).decode() 40 | self.initial['css_data'] = decoded_css_data # Set the decoded value as initial 41 | except Exception: 42 | pass # Handle decoding errors if needed 43 | 44 | def clean_base_theme(self): 45 | """Don't allow editing of the default base theme.""" 46 | if self.instance and self.instance.name == "default": 47 | return "toggle" 48 | else: 49 | return self.cleaned_data['base_theme'] 50 | 51 | def save(self, commit=True): 52 | # If this theme is being activated, deactivate the others 53 | if self.cleaned_data.get('active'): 54 | # Deactivate any currently active theme 55 | Theme.objects.filter(active=True).update(active=False) 56 | 57 | # If the theme is being deactivated, activate the default 58 | instance = super().save(commit=False) 59 | # only if the theme isn't a new resource 60 | if instance.pk: 61 | if self.cleaned_data.get('active') == False: 62 | # and we're switching from True -> False (not editing an inactive theme) 63 | if self.instance._prechange_snapshot.get("active") == True: 64 | Theme.objects.filter(name="default").update(active=True) 65 | 66 | # B64 encode the css_data 67 | # instance = super().save(commit=False) 68 | if self.cleaned_data.get('css_data'): 69 | instance.css_data = base64.b64encode(self.cleaned_data['css_data'].encode()).decode() 70 | 71 | # Prevent no themes being active 72 | if self.cleaned_data.get("name") == "default": 73 | pre = Theme.objects.get(pk=self.instance.pk) 74 | if pre.active == True and self.cleaned_data.get("active") == False: 75 | raise ThemeEditError("Cannot deactive default theme") 76 | 77 | # Save the current theme (it will be the only active one if checked) 78 | return super().save(commit) 79 | 80 | def clean_css_data(self): 81 | """ Crude check to stop scripts being uploaded. """ 82 | data = self.cleaned_data['css_data'] 83 | forbidden_patterns = [ 84 | r"<[^>]+>", 85 | ] 86 | for pattern in forbidden_patterns: 87 | if re.search(pattern, data, re.IGNORECASE): 88 | raise forms.ValidationError("Invalid CSS") 89 | return data 90 | -------------------------------------------------------------------------------- /samples/greggs.css: -------------------------------------------------------------------------------- 1 | body { 2 | [data-bs-theme="dark"] {} 3 | } 4 | 5 | body[data-bs-theme="dark"] .navbar-brand, 6 | body[data-bs-theme="dark"] .sticky-top 7 | { 8 | background: #00558F; 9 | background-color: #00558F; 10 | color: white; 11 | } 12 | 13 | body[data-bs-theme="dark"] .navbar-nav, 14 | body[data-bs-theme="dark"] .navbar-vertical, 15 | body[data-bs-theme="dark"] .navbar 16 | { 17 | background: #00558F; 18 | background-color: #00558F; 19 | color: white; 20 | } 21 | 22 | body[data-bs-theme="dark"] .color-mode-toggle, 23 | body[data-bs-theme="light"] .color-mode-toggle 24 | { 25 | display: none; 26 | } 27 | 28 | body[data-bs-theme="dark"] .container-fluid d-flex justify-content-between align-items-center { 29 | background-color: #081B2A; 30 | } 31 | 32 | body[data-bs-theme=dark] .navbar-vertical.navbar-expand-lg .text-secondary { 33 | color: #FDB71A!important; 34 | } 35 | 36 | body[data-bs-theme="dark"] .navbar-vertical.navbar-expand-lg { 37 | background-color: #00558F; 38 | background-image: none; 39 | } 40 | 41 | body[data-bs-theme="dark"] .navbar-vertical.navbar-expand-lg .nav-item.dropdown.active::after { 42 | border-color: #FDB71A!important; 43 | } 44 | body[data-bs-theme="light"] .navbar-brand-image, 45 | body[data-bs-theme="dark"] .navbar-brand-image { 46 | display: none; 47 | } 48 | 49 | 50 | 51 | body[data-bs-theme="dark"] .navbar-brand { 52 | display: inline-block; 53 | background-image: url(""); 54 | background-size: contain; 55 | background-repeat: no-repeat; 56 | width: 250px; /* Set the width and height of the new image */ 57 | height: 50px; 58 | } 59 | 60 | .navbar-brand { 61 | margin-left: auto; 62 | margin-right: auto; 63 | margin-top: 5px; 64 | } 65 | 66 | body[data-bs-theme="light"] .netbox-edition , 67 | body[data-bs-theme="dark"] .netbox-edition { 68 | visibility: hidden; 69 | height: 50px; 70 | } 71 | -------------------------------------------------------------------------------- /samples/doom.css: -------------------------------------------------------------------------------- 1 | body { 2 | [data-bs-theme="dark"] {} 3 | } 4 | 5 | body[data-bs-theme="dark"] .color-mode-toggle, 6 | body[data-bs-theme="light"] .color-mode-toggle 7 | { 8 | display: none; 9 | } 10 | 11 | .navbar { 12 | background: linear-gradient(black, #480300ff) !important; 13 | } 14 | 15 | .text-uppercase { 16 | font-size: 18px !important; 17 | } 18 | 19 | .d-inline-flex { 20 | font-size: 18px !important; 21 | } 22 | 23 | body { 24 | font-family: 'amazdoom', sans-serif; 25 | color: red; 26 | font-size: 16px; 27 | text-shadow: black 1px 0 10px; 28 | } 29 | 30 | .navbar-nav { 31 | font-size: 18px; 32 | } 33 | 34 | body[data-bs-theme=dark] .navbar-vertical.navbar-expand-lg .text-secondary { 35 | color: red!important; 36 | } 37 | 38 | body[data-bs-theme="dark"] .navbar-vertical.navbar-expand-lg .nav-item.dropdown.active::after { 39 | border-color: red!important; 40 | } 41 | 42 | .navbar-vertical.navbar-expand-lg .navbar-brand { 43 | height: 75px; 44 | margin-left: auto; 45 | margin-right:auto; 46 | margin-top: 10px; 47 | } 48 | 49 | 50 | body[data-bs-theme="light"] .netbox-edition , 51 | body[data-bs-theme="dark"] .netbox-edition { 52 | height: 50px; 53 | margin-top: 50px !important; 54 | font-size: 20px !important; 55 | } 56 | 57 | body[data-bs-theme="light"] .navbar-brand-image, 58 | body[data-bs-theme="dark"] .navbar-brand-image { 59 | display: none; 60 | } 61 | 62 | body[data-bs-theme="dark"] .navbar-brand { 63 | display: inline-block; 64 | background-image: url(""); 65 | background-size: contain; 66 | background-repeat: no-repeat; 67 | width: 250px; /* Set the width and height of the new image */ 68 | height: 50px; 69 | } 70 | 71 | /* https://www.dafont.com/amazdoom.font */ 72 | /* http://creativecommons.org/licenses/by-nc/3.0/ */ 73 | 74 | @font-face { 75 | font-family: 'amazdoom'; 76 | src: url(data:font/ttf;base64,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CWFmaWkxMDA0OAlhZmlpMTAwNDkIQ3lySGFyZEcIZ2x5cGgxNDYIY3lyaGFyZGcJYWZpaTEwMDUzCWFmaWkxMDEwMQphZmlpMTAwNTYJCWFmaWkxMDEwNAhnbHlwaDE3NwlhZmlpMTAwNjIJYWZpaTEwMTEwCWFmaWkxMDE0NQlhZmlpMTAxOTMJYWZpaTYxMzUyCWFmaWkxMDAyMwlhZmlpMTAwNzEJYWZpaTEwMDk3CWFmaWkxMDA2NQlhZmlpMTAwNjYJYWZpaTEwMDY3CWFmaWkxMDA2OQlhZmlpMTAwNzAJYWZpaTEwMDcyCWFmaWkxMDA3MwlhZmlpMTAwNzQJYWZpaTEwMDc1CWFmaWkxMDA3NwlhZmlpMTAwNzgJYWZpaTEwMDc5CWFmaWkxMDA4MAlhZmlpMTAwODEJYWZpaTEwMDgyCWFmaWkxMDA4MwlhZmlpMTAwODQJYWZpaTEwMDg1CWFmaWkxMDA4NglhZmlpMTAwODcJYWZpaTEwMDg4CWFmaWkxMDA4OQlhZmlpMTAwOTAJYWZpaTEwMDkxCWFmaWkxMDA5MglhZmlpMTAwOTMJYWZpaTEwMDk0CWFmaWkxMDA5NQlhZmlpMTAwOTYJRGFiYmxlIERlAAAA) format('truetype'); 77 | font-weight: normal; 78 | font-style: normal; 79 | } 80 | 81 | .mdi::before { 82 | content: url(""); 83 | } 84 | -------------------------------------------------------------------------------- /samples/excel.css: -------------------------------------------------------------------------------- 1 | .btn { 2 | background-color: #207247; 3 | color: white; 4 | border-radius: 0px; 5 | } 6 | 7 | .navbar-nav .nav-item { 8 | border-width: 1px !important; 9 | border-color: #d7d7d7 !important; 10 | border-style: solid !important; 11 | background-color: white !important; 12 | border-bottom: 0px !important; 13 | } 14 | 15 | .container-fluid { 16 | background-color: white; 17 | } 18 | 19 | td { 20 | border-width: 1px !important; 21 | border-color: #d7d7d7 !important; 22 | border-style: solid !important; 23 | background-color: white !important; 24 | } 25 | 26 | .w-1 { 27 | background-color: #f6f8fb !important; 28 | } 29 | 30 | 31 | .fw-bold, 32 | .py-1 { 33 | margin-left: 15px; 34 | } 35 | 36 | .navbar-nav .nav-item::before { 37 | content: ""; 38 | position: absolute; 39 | top: 0; 40 | left: 0; 41 | width: 25px; 42 | height: 100%; 43 | background-color: #e5e5e5; 44 | } 45 | 46 | .mdi-domain::before { 47 | font-family: monospace; 48 | content: "1" !important; 49 | left: 5px; 50 | position: absolute; 51 | font-size: 12px; 52 | } 53 | 54 | .mdi-door-sliding::before { 55 | font-family: monospace; 56 | content: "2" !important; 57 | left: 9px; 58 | position: absolute; 59 | font-size: 12px; 60 | } 61 | 62 | .mdi-server::before { 63 | font-family: monospace; 64 | content: "3" !important; 65 | left: 9px; 66 | position: absolute; 67 | font-size: 12px; 68 | } 69 | 70 | .mdi-connection::before { 71 | font-family: monospace; 72 | content: "4" !important; 73 | left: 9px; 74 | position: absolute; 75 | font-size: 12px; 76 | } 77 | 78 | .mdi-wifi::before { 79 | font-family: monospace; 80 | content: "5" !important; 81 | left: 9px; 82 | position: absolute; 83 | font-size: 12px; 84 | } 85 | 86 | .mdi-counter::before { 87 | font-family: monospace; 88 | content: "6" !important; 89 | left: 9px; 90 | position: absolute; 91 | font-size: 12px; 92 | } 93 | 94 | .mdi-graph-outline::before { 95 | font-family: monospace; 96 | content: "7" !important; 97 | left: 9px; 98 | position: absolute; 99 | font-size: 12px; 100 | } 101 | 102 | .mdi-monitor::before { 103 | font-family: monospace; 104 | content: "8" !important; 105 | left: 9px; 106 | position: absolute; 107 | font-size: 12px; 108 | } 109 | 110 | .mdi-transit-connection-variant::before { 111 | font-family: monospace; 112 | content: "9" !important; 113 | left: 9px; 114 | position: absolute; 115 | font-size: 12px; 116 | } 117 | 118 | .mdi-flash::before { 119 | font-family: monospace; 120 | content: "10" !important; 121 | left: 5px; 122 | position: absolute; 123 | font-size: 12px; 124 | } 125 | 126 | .mdi-file-document-multiple-outline::before { 127 | font-family: monospace; 128 | content: "11" !important; 129 | left: 5px; 130 | position: absolute; 131 | font-size: 12px; 132 | } 133 | 134 | .mdi-toolbox-outline::before { 135 | font-family: monospace; 136 | content: "12" !important; 137 | left: 5px; 138 | position: absolute; 139 | font-size: 12px; 140 | } 141 | 142 | .mdi-cogs::before { 143 | font-family: monospace; 144 | content: "13" !important; 145 | left: 5px; 146 | position: absolute; 147 | font-size: 12px; 148 | } 149 | 150 | .mdi-puzzle::before { 151 | font-family: monospace; 152 | content: "14" !important; 153 | left: 5px; 154 | position: absolute; 155 | font-size: 12px; 156 | } 157 | 158 | .mdi-account-multiple::before { 159 | font-family: monospace; 160 | content: "15" !important; 161 | left: 5px; 162 | position: absolute; 163 | font-size: 12px; 164 | } 165 | 166 | .mdi-door-sliding::before { 167 | font-family: monospace; 168 | content: "2" !important; 169 | left: 5px; 170 | position: absolute; 171 | } 172 | 173 | .mdi-server::before { 174 | font-family: monospace; 175 | content: "3" !important; 176 | left: 5px; 177 | position: absolute; 178 | } 179 | 180 | .mdi-connection::before { 181 | font-family: monospace; 182 | content: "4" !important; 183 | left: 5px; 184 | position: absolute; 185 | } 186 | 187 | .mdi-wifi::before { 188 | font-family: monospace; 189 | content: "5" !important; 190 | left: 5px; 191 | position: absolute; 192 | } 193 | 194 | .mdi-counter::before { 195 | font-family: monospace; 196 | content: "6" !important; 197 | left: 5px; 198 | position: absolute; 199 | } 200 | 201 | .mdi-graph-outline::before { 202 | font-family: monospace; 203 | content: "7" !important; 204 | left: 5px; 205 | position: absolute; 206 | } 207 | 208 | .mdi-monitor::before { 209 | font-family: monospace; 210 | content: "8" !important; 211 | left: 5px; 212 | position: absolute; 213 | } 214 | 215 | .mdi-transit-connection-variant::before { 216 | font-family: monospace; 217 | content: "9" !important; 218 | left: 5px; 219 | position: absolute; 220 | } 221 | 222 | .mdi-flash::before { 223 | font-family: monospace; 224 | content: "10" !important; 225 | left: 5px; 226 | position: absolute; 227 | } 228 | 229 | .mdi-file-document-multiple-outline::before { 230 | font-family: monospace; 231 | content: "11" !important; 232 | left: 5px; 233 | position: absolute; 234 | } 235 | 236 | .mdi-toolbox-outline::before { 237 | font-family: monospace; 238 | content: "12" !important; 239 | left: 5px; 240 | position: absolute; 241 | } 242 | 243 | .mdi-cogs::before { 244 | font-family: monospace; 245 | content: "13" !important; 246 | left: 5px; 247 | position: absolute; 248 | } 249 | 250 | .mdi-puzzle::before { 251 | font-family: monospace; 252 | content: "14" !important; 253 | left: 5px; 254 | position: absolute; 255 | } 256 | 257 | .mdi-account-multiple::before { 258 | font-family: monospace; 259 | content: "15" !important; 260 | left: 5px; 261 | position: absolute; 262 | } 263 | 264 | 265 | body[data-bs-theme="light"] .navbar-brand-image { 266 | display: none; 267 | } 268 | 269 | body[data-bs-theme="light"] .netbox-edition { 270 | visibility: hidden; 271 | } 272 | 273 | body[data-bs-theme="light"] .navbar-brand { 274 | display: inline-block; 275 | background-image: url(""); 276 | background-size: contain; 277 | background-repeat: no-repeat; 278 | width: 350px; /* Set the width and height of the new image */ 279 | height: 75px; 280 | margin-left: 75px; 281 | } 282 | --------------------------------------------------------------------------------