├── .gitignore ├── README.md ├── back ├── __init__.py ├── admin.py ├── apps.py ├── forms.py ├── models.py ├── templates │ └── back │ │ ├── index.html │ │ └── success.html ├── tests.py ├── urls.py └── views.py ├── interview_task ├── __init__.py ├── asgi.py ├── settings.py ├── urls.py └── wsgi.py ├── manage.py ├── requirements.txt └── static ├── css ├── vendor │ ├── app.min.css │ ├── fonts │ │ ├── bootstrap-icons.woff │ │ └── bootstrap-icons.woff2 │ ├── images │ │ ├── cover-2.jpg │ │ ├── cover-3.jpg │ │ ├── cover-4.jpg │ │ ├── cover-5.jpg │ │ ├── cover.jpg │ │ ├── logo.png │ │ └── pattern.png │ └── vendor.min.css └── zoom.css ├── favicon.ico ├── image ├── delete.png └── pc.png ├── js ├── vendor │ ├── app.min.js │ └── vendor.min.js └── zoom.js └── webfonts ├── fa-brands-400.ttf ├── fa-brands-400.woff2 ├── fa-regular-400.ttf ├── fa-regular-400.woff2 ├── fa-solid-900.ttf ├── fa-solid-900.woff2 ├── fa-v4compatibility.ttf └── fa-v4compatibility.woff2 /.gitignore: -------------------------------------------------------------------------------- 1 | venv 2 | *.pyc 3 | db.sqlite3 4 | __pycache__/ 5 | .env -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | Please follow the following steps to run the project: 2 | 3 | >python -m venv venv 4 | > 5 | >venv\scripts\activate 6 | > 7 | >pip install -r requirments.txt 8 | > 9 | >python manage.py migrate 10 | > 11 | >python mange.py runserver 0.0.0.0:8000 12 | -------------------------------------------------------------------------------- /back/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/back/__init__.py -------------------------------------------------------------------------------- /back/admin.py: -------------------------------------------------------------------------------- 1 | from django.contrib import admin 2 | from .models import MyUser, Task 3 | 4 | 5 | # Register your models here. 6 | admin.site.register(MyUser) 7 | admin.site.register(Task) 8 | 9 | -------------------------------------------------------------------------------- /back/apps.py: -------------------------------------------------------------------------------- 1 | from django.apps import AppConfig 2 | 3 | 4 | class BackConfig(AppConfig): 5 | default_auto_field = 'django.db.models.BigAutoField' 6 | name = 'back' 7 | -------------------------------------------------------------------------------- /back/forms.py: -------------------------------------------------------------------------------- 1 | from django.forms import ModelForm 2 | from .models import MyUser 3 | from django.utils.translation import gettext_lazy as _ 4 | from django import forms 5 | 6 | class MyUserForm(ModelForm): 7 | class Meta: 8 | model = MyUser 9 | fields = ['username', 'password'] 10 | labels = { 11 | 'username': _('Username'), 12 | 'password': _('Password') 13 | } 14 | widgets = { 15 | 'username': forms.TextInput(attrs={'class':'form-control'}), 16 | 'password': forms.PasswordInput(attrs={'class':'form-control'}) 17 | } -------------------------------------------------------------------------------- /back/models.py: -------------------------------------------------------------------------------- 1 | from pickle import FALSE 2 | from django.db import models 3 | from django.contrib.auth.models import User 4 | 5 | 6 | class MyUser(User): 7 | def __str__(self) -> str: 8 | return self.username 9 | 10 | class Task(models.Model): 11 | userId = models.IntegerField() 12 | id = models.IntegerField(primary_key=True) 13 | title = models.CharField(max_length=255) 14 | body = models.CharField(max_length=255) -------------------------------------------------------------------------------- /back/templates/back/index.html: -------------------------------------------------------------------------------- 1 | {% load static %} 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 |
20 |
21 | 29 |
30 |
31 | 32 | 33 | 34 | 35 | 36 | 37 | -------------------------------------------------------------------------------- /back/templates/back/success.html: -------------------------------------------------------------------------------- 1 | {% load static %} 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 |
20 |
21 | 32 |
33 |
34 | 35 | 36 | 37 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /back/tests.py: -------------------------------------------------------------------------------- 1 | from django.test import TestCase 2 | 3 | # Create your tests here. 4 | -------------------------------------------------------------------------------- /back/urls.py: -------------------------------------------------------------------------------- 1 | from django.urls import path 2 | from . import views 3 | 4 | app_name = 'back' 5 | 6 | urlpatterns = [ 7 | path('', views.index, name='index'), 8 | path('success', views.post_data, name='success') 9 | ] -------------------------------------------------------------------------------- /back/views.py: -------------------------------------------------------------------------------- 1 | from django.shortcuts import render 2 | from .forms import MyUserForm 3 | import requests 4 | from back.models import Task 5 | 6 | 7 | # Create your views here. 8 | def index(request): 9 | context = { 10 | } 11 | return render(request, 'back/index.html', context) 12 | 13 | def post_data(request): 14 | if request.method == 'POST': 15 | api_url = 'http://jsonplaceholder.typicode.com/posts' 16 | try: 17 | response = requests.post(api_url) 18 | print(response.json()) 19 | idval = '' 20 | if 'id' in response.json(): 21 | idval = response.json()['id'] 22 | title = '' 23 | if 'title' in response.json(): 24 | title = response.json()['title'] 25 | bd = '' 26 | if 'body' in response.json(): 27 | bd = response.json()['body'] 28 | userId = 0 29 | if 'userId' in response.json(): 30 | userId = response.json()['userId'] 31 | new_db = Task(id=idval, title=title, body=bd, userId=userId) 32 | except Exception as e: 33 | print("error") 34 | user_form = MyUserForm() 35 | context = { 36 | 'success': '1', 37 | 'id': idval, 38 | 'title': title, 39 | 'body': bd, 40 | 'userId': userId 41 | } 42 | return render(request, 'back/success.html', context) 43 | else: 44 | user_form = MyUserForm() 45 | context = { 46 | 'success': '0' 47 | } 48 | return render(request, 'back/index.html', context) -------------------------------------------------------------------------------- /interview_task/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/interview_task/__init__.py -------------------------------------------------------------------------------- /interview_task/asgi.py: -------------------------------------------------------------------------------- 1 | """ 2 | ASGI config for interview_task project. 3 | 4 | It exposes the ASGI callable as a module-level variable named ``application``. 5 | 6 | For more information on this file, see 7 | https://docs.djangoproject.com/en/4.0/howto/deployment/asgi/ 8 | """ 9 | 10 | import os 11 | 12 | from django.core.asgi import get_asgi_application 13 | 14 | os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'interview_task.settings') 15 | 16 | application = get_asgi_application() 17 | -------------------------------------------------------------------------------- /interview_task/settings.py: -------------------------------------------------------------------------------- 1 | """ 2 | Django settings for interview_task project. 3 | 4 | Generated by 'django-admin startproject' using Django 4.0.6. 5 | 6 | For more information on this file, see 7 | https://docs.djangoproject.com/en/4.0/topics/settings/ 8 | 9 | For the full list of settings and their values, see 10 | https://docs.djangoproject.com/en/4.0/ref/settings/ 11 | """ 12 | 13 | from pathlib import Path 14 | import os 15 | from dotenv import load_dotenv 16 | 17 | 18 | load_dotenv() 19 | 20 | # Build paths inside the project like this: BASE_DIR / 'subdir'. 21 | BASE_DIR = Path(__file__).resolve().parent.parent 22 | 23 | 24 | # Quick-start development settings - unsuitable for production 25 | # See https://docs.djangoproject.com/en/4.0/howto/deployment/checklist/ 26 | 27 | # SECURITY WARNING: keep the secret key used in production secret! 28 | SECRET_KEY = '9cg2x%f4dnbh^uu56k0ig038ir)m6k2e4a&(6$g)%g1#c2w#77'#os.environ['SECRET_KEY'] 29 | 30 | # SECURITY WARNING: don't run with debug turned on in production! 31 | DEBUG = True 32 | 33 | ALLOWED_HOSTS = [] 34 | 35 | 36 | # Application definition 37 | 38 | INSTALLED_APPS = [ 39 | 'django.contrib.admin', 40 | 'django.contrib.auth', 41 | 'django.contrib.contenttypes', 42 | 'django.contrib.sessions', 43 | 'django.contrib.messages', 44 | 'django.contrib.staticfiles', 45 | 'storages', 46 | 'rest_framework', 47 | 'back', 48 | ] 49 | 50 | MIDDLEWARE = [ 51 | 'django.middleware.security.SecurityMiddleware', 52 | 'django.contrib.sessions.middleware.SessionMiddleware', 53 | 'django.middleware.common.CommonMiddleware', 54 | 'django.middleware.csrf.CsrfViewMiddleware', 55 | 'django.contrib.auth.middleware.AuthenticationMiddleware', 56 | 'django.contrib.messages.middleware.MessageMiddleware', 57 | 'django.middleware.clickjacking.XFrameOptionsMiddleware', 58 | ] 59 | 60 | ROOT_URLCONF = 'interview_task.urls' 61 | 62 | TEMPLATES = [ 63 | { 64 | 'BACKEND': 'django.template.backends.django.DjangoTemplates', 65 | 'DIRS': [], 66 | 'APP_DIRS': True, 67 | 'OPTIONS': { 68 | 'context_processors': [ 69 | 'django.template.context_processors.debug', 70 | 'django.template.context_processors.request', 71 | 'django.contrib.auth.context_processors.auth', 72 | 'django.contrib.messages.context_processors.messages', 73 | ], 74 | }, 75 | }, 76 | ] 77 | 78 | WSGI_APPLICATION = 'interview_task.wsgi.application' 79 | 80 | 81 | # Database 82 | # https://docs.djangoproject.com/en/4.0/ref/settings/#databases 83 | 84 | DATABASES = { 85 | 'default': { 86 | 'ENGINE': 'django.db.backends.sqlite3', 87 | 'NAME': BASE_DIR / 'db.sqlite3', 88 | } 89 | } 90 | 91 | 92 | # Password validation 93 | # https://docs.djangoproject.com/en/4.0/ref/settings/#auth-password-validators 94 | 95 | AUTH_PASSWORD_VALIDATORS = [ 96 | { 97 | 'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator', 98 | }, 99 | { 100 | 'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator', 101 | }, 102 | { 103 | 'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator', 104 | }, 105 | { 106 | 'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator', 107 | }, 108 | ] 109 | 110 | 111 | # Internationalization 112 | # https://docs.djangoproject.com/en/4.0/topics/i18n/ 113 | 114 | LANGUAGE_CODE = 'en-us' 115 | 116 | TIME_ZONE = 'UTC' 117 | 118 | USE_I18N = True 119 | 120 | USE_TZ = True 121 | 122 | 123 | # Static files (CSS, JavaScript, Images) 124 | # https://docs.djangoproject.com/en/4.0/howto/static-files/ 125 | STATIC_URL = '/static/' 126 | STATIC_ROOT = os.path.join(BASE_DIR, 'staticfiles') 127 | STATICFILES_DIRS = ( 128 | os.path.join(BASE_DIR, 'static'), 129 | ) 130 | 131 | 132 | # Default primary key field type 133 | # https://docs.djangoproject.com/en/4.0/ref/settings/#default-auto-field 134 | 135 | DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField' 136 | 137 | # Configure Django App for Heroku. 138 | import django_on_heroku 139 | django_on_heroku.settings(locals()) 140 | 141 | 142 | AWS_ACCESS_KEY_ID = ''#os.environ['AWS_ACCESS_KEY_ID'] 143 | AWS_SECRET_ACCESS_KEY = ''#os.environ['AWS_SECRET_ACCESS_KEY'] 144 | AWS_STORAGE_BUCKET_NAME = ''#os.environ['AWS_STORAGE_BUCKET_NAME'] 145 | AWS_S3_REGION_NAME = ''#os.environ['AWS_S3_REGION_NAME'] 146 | AWS_S3_SIGNATURE_VERSION = '4.0.6'#os.environ['AWS_S3_SIGNATURE_VERSION'] 147 | 148 | AWS_S3_FILE_OVERWRITE=False 149 | AWS_DEFAULT_ACL=None 150 | DEFAULT_FILE_STORAGE = 'django.core.files.storage.FileSystemStorage' 151 | AWS_QUERYSTRING_AUTH = False 152 | 153 | REST_FRAMEWORK = { 154 | 'DEFAULT_AUTHENTICATION_CLASSES': [ 155 | 'rest_framework.authentication.TokenAuthentication', 156 | ] 157 | } 158 | 159 | SECURE_CROSS_ORIGIN_OPENER_POLICY = None 160 | 161 | -------------------------------------------------------------------------------- /interview_task/urls.py: -------------------------------------------------------------------------------- 1 | """interview_task URL Configuration 2 | 3 | The `urlpatterns` list routes URLs to views. For more information please see: 4 | https://docs.djangoproject.com/en/4.0/topics/http/urls/ 5 | Examples: 6 | Function views 7 | 1. Add an import: from my_app import views 8 | 2. Add a URL to urlpatterns: path('', views.home, name='home') 9 | Class-based views 10 | 1. Add an import: from other_app.views import Home 11 | 2. Add a URL to urlpatterns: path('', Home.as_view(), name='home') 12 | Including another URLconf 13 | 1. Import the include() function: from django.urls import include, path 14 | 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) 15 | """ 16 | from django.contrib import admin 17 | from django.urls import path, include 18 | from django.conf import settings 19 | from django.conf.urls.static import static 20 | urlpatterns = [ 21 | path('admin/', admin.site.urls), 22 | path('', include('back.urls')) 23 | ]+ static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT) 24 | -------------------------------------------------------------------------------- /interview_task/wsgi.py: -------------------------------------------------------------------------------- 1 | """ 2 | WSGI config for interview_task project. 3 | 4 | It exposes the WSGI callable as a module-level variable named ``application``. 5 | 6 | For more information on this file, see 7 | https://docs.djangoproject.com/en/4.0/howto/deployment/wsgi/ 8 | """ 9 | 10 | import os 11 | 12 | from django.core.wsgi import get_wsgi_application 13 | 14 | os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'interview_task.settings') 15 | 16 | application = get_wsgi_application() 17 | -------------------------------------------------------------------------------- /manage.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | """Django's command-line utility for administrative tasks.""" 3 | import os 4 | import sys 5 | 6 | 7 | def main(): 8 | """Run administrative tasks.""" 9 | os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'interview_task.settings') 10 | try: 11 | from django.core.management import execute_from_command_line 12 | except ImportError as exc: 13 | raise ImportError( 14 | "Couldn't import Django. Are you sure it's installed and " 15 | "available on your PYTHONPATH environment variable? Did you " 16 | "forget to activate a virtual environment?" 17 | ) from exc 18 | execute_from_command_line(sys.argv) 19 | 20 | 21 | if __name__ == '__main__': 22 | main() 23 | -------------------------------------------------------------------------------- /requirements.txt: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/requirements.txt -------------------------------------------------------------------------------- /static/css/vendor/fonts/bootstrap-icons.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/fonts/bootstrap-icons.woff -------------------------------------------------------------------------------- /static/css/vendor/fonts/bootstrap-icons.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/fonts/bootstrap-icons.woff2 -------------------------------------------------------------------------------- /static/css/vendor/images/cover-2.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/images/cover-2.jpg -------------------------------------------------------------------------------- /static/css/vendor/images/cover-3.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/images/cover-3.jpg -------------------------------------------------------------------------------- /static/css/vendor/images/cover-4.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/images/cover-4.jpg -------------------------------------------------------------------------------- /static/css/vendor/images/cover-5.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/images/cover-5.jpg -------------------------------------------------------------------------------- /static/css/vendor/images/cover.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/images/cover.jpg -------------------------------------------------------------------------------- /static/css/vendor/images/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/images/logo.png -------------------------------------------------------------------------------- /static/css/vendor/images/pattern.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/css/vendor/images/pattern.png -------------------------------------------------------------------------------- /static/css/zoom.css: -------------------------------------------------------------------------------- 1 | img[data-action="zoom"] { 2 | cursor: pointer; 3 | cursor: -webkit-zoom-in; 4 | cursor: -moz-zoom-in; 5 | 6 | } 7 | .zoom-img, 8 | .zoom-img-wrap { 9 | position: relative; 10 | z-index: 666; 11 | -webkit-transition: all 300ms; 12 | -o-transition: all 300ms; 13 | transition: all 300ms; 14 | } 15 | img.zoom-img { 16 | cursor: pointer; 17 | cursor: -webkit-zoom-out; 18 | cursor: -moz-zoom-out; 19 | 20 | } 21 | .zoom-overlay { 22 | z-index: 420; 23 | background: #fff; 24 | position: fixed; 25 | top: 0; 26 | left: 0; 27 | right: 0; 28 | bottom: 0; 29 | pointer-events: none; 30 | filter: "alpha(opacity=0)"; 31 | opacity: 0; 32 | -webkit-transition: opacity 300ms; 33 | -o-transition: opacity 300ms; 34 | transition: opacity 300ms; 35 | } 36 | .zoom-overlay-open .zoom-overlay { 37 | filter: "alpha(opacity=100)"; 38 | opacity: 1; 39 | } 40 | .zoom-overlay-open, 41 | .zoom-overlay-transitioning { 42 | cursor: default; 43 | } 44 | .delete_button { 45 | cursor: pointer; 46 | } 47 | .gallery { 48 | position: relative; 49 | padding: 10px; 50 | } 51 | 52 | .img-thumbnail { 53 | opacity: 1; 54 | display: block; 55 | width: 100%; 56 | height: auto; 57 | transition: .5s ease; 58 | backface-visibility: hidden; 59 | } 60 | 61 | .middle { 62 | transition: .5s ease; 63 | opacity: 0; 64 | position: absolute; 65 | top: 10px; 66 | right: 0px; 67 | transform: translate(-50%, -50%); 68 | -ms-transform: translate(-50%, -50%); 69 | text-align: center; 70 | } 71 | .gallery:hover .img-thumbnail { 72 | opacity : 0.5; 73 | background-color: rgba(0, 0, 0, 0.7) 74 | } 75 | .gallery:hover .zoom-img { 76 | opacity : 1; 77 | background-color: rgba(0, 0, 0, 0.7) 78 | } 79 | 80 | .gallery:hover .middle { 81 | opacity: 1; 82 | } 83 | 84 | 85 | 86 | 87 | -------------------------------------------------------------------------------- /static/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/favicon.ico -------------------------------------------------------------------------------- /static/image/delete.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/image/delete.png -------------------------------------------------------------------------------- /static/image/pc.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/image/pc.png -------------------------------------------------------------------------------- /static/js/vendor/app.min.js: -------------------------------------------------------------------------------- 1 | /* 2 | Template Name: HUD - Responsive Bootstrap 5 Admin Template 3 | Version: 1.9.0 4 | Author: Sean Ngu 5 | Website: http://www.seantheme.com/hud/ 6 | ---------------------------- 7 | APPS CONTENT TABLE 8 | ---------------------------- 9 | 10 | 11 | 01. Global Variable 12 | 02. Handle Scrollbar 13 | 03. Handle Sidebar Menu 14 | 04. Handle Sidebar Scroll Memory 15 | 05. Handle Card Action 16 | 06. Handle Tooltip & Popover Activation 17 | 07. Handle Scroll to Top Button 18 | 08. Handle hexToRgba 19 | 09. Handle Scroll To 20 | 10. Handle Toggle Class 21 | 11. Handle Theme Panel 22 | 12. Application Controller 23 | 13. Initialise 24 | 25 | 26 | Application Controller 27 | */ 28 | 29 | 30 | 31 | /* 01. Global Variable 32 | ------------------------------------------------ */ 33 | var app = { 34 | id: '#app', 35 | isMobile: ((/Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent)) || window.innerWidth < 992), 36 | bootstrap: { 37 | tooltip: { 38 | attr: 'data-bs-toggle="tooltip"' 39 | }, 40 | popover: { 41 | attr: 'data-bs-toggle="popover"' 42 | }, 43 | modal: { 44 | attr: 'data-bs-toggle="modal"', 45 | dismissAttr: 'data-bs-dismiss="modal"', 46 | event: { 47 | hidden: 'hidden.bs.modal' 48 | } 49 | }, 50 | nav: { 51 | class: 'nav', 52 | tabs: { 53 | class: 'nav-tabs', 54 | activeClass: 'active', 55 | itemClass: 'nav-item', 56 | itemLinkClass: 'nav-link' 57 | } 58 | } 59 | }, 60 | header: { 61 | id: '#header', 62 | class: 'app-header', 63 | hasScrollClass: 'has-scroll' 64 | }, 65 | sidebar: { 66 | id: '#sidebar', 67 | class: 'app-sidebar', 68 | scrollBar: { 69 | localStorage: 'appSidebarScrollPosition', 70 | dom: '' 71 | }, 72 | menu: { 73 | class: 'menu', 74 | initAttr: 'data-init', 75 | animationTime: 0, 76 | itemClass: 'menu-item', 77 | itemLinkClass: 'menu-link', 78 | hasSubClass: 'has-sub', 79 | activeClass: 'active', 80 | expandingClass: 'expanding', 81 | expandClass: 'expand', 82 | submenu: { 83 | class: 'menu-submenu', 84 | } 85 | }, 86 | mobile: { 87 | toggleAttr: 'data-toggle="app-sidebar-mobile"', 88 | dismissAttr: 'data-dismiss="app-sidebar-mobile"', 89 | toggledClass: 'app-sidebar-mobile-toggled', 90 | closedClass: 'app-sidebar-mobile-closed', 91 | backdrop: { 92 | class: 'app-sidebar-mobile-backdrop' 93 | } 94 | }, 95 | minify: { 96 | toggleAttr: 'data-toggle="app-sidebar-minify"', 97 | toggledClass: 'app-sidebar-minified', 98 | cookieName: 'app-sidebar-minified' 99 | }, 100 | floatSubmenu: { 101 | id: '#app-sidebar-float-submenu', 102 | dom: '', 103 | timeout: '', 104 | class: 'app-sidebar-float-submenu', 105 | container: { 106 | class: 'app-sidebar-float-submenu-container' 107 | }, 108 | arrow: { 109 | id: '#app-sidebar-float-submenu-arrow', 110 | class: 'app-sidebar-float-submenu-arrow' 111 | }, 112 | line: { 113 | id: '#app-sidebar-float-submenu-line', 114 | class: 'app-sidebar-float-submenu-line' 115 | }, 116 | overflow: { 117 | class: 'overflow-scroll mh-100vh' 118 | } 119 | }, 120 | search: { 121 | class: 'menu-search', 122 | toggleAttr: 'data-sidebar-search="true"', 123 | hideClass: 'd-none', 124 | foundClass: 'has-text' 125 | }, 126 | transparent: { 127 | class: 'app-sidebar-transparent' 128 | } 129 | }, 130 | scrollBar: { 131 | attr: 'data-scrollbar="true"', 132 | skipMobileAttr: 'data-skip-mobile', 133 | heightAttr: 'data-height', 134 | wheelPropagationAttr: 'data-wheel-propagation' 135 | }, 136 | content: { 137 | id: '#content', 138 | class: 'app-content', 139 | fullHeight: { 140 | class: 'app-content-full-height' 141 | }, 142 | fullWidth: { 143 | class: 'app-content-full-width' 144 | } 145 | }, 146 | layout: { 147 | sidebarLight: { 148 | class: 'app-with-light-sidebar' 149 | }, 150 | sidebarEnd: { 151 | class: 'app-with-end-sidebar' 152 | }, 153 | sidebarWide: { 154 | class: 'app-with-wide-sidebar' 155 | }, 156 | sidebarMinified: { 157 | class: 'app-sidebar-minified' 158 | }, 159 | sidebarTwo: { 160 | class: 'app-with-two-sidebar' 161 | }, 162 | withoutHeader: { 163 | class: 'app-without-header' 164 | }, 165 | withoutSidebar: { 166 | class: 'app-without-sidebar' 167 | }, 168 | topMenu: { 169 | class: 'app-with-top-menu' 170 | }, 171 | boxedLayout: { 172 | class: 'boxed-layout' 173 | } 174 | }, 175 | scrollToTopBtn: { 176 | showClass: 'show', 177 | heightShow: 200, 178 | toggleAttr: 'data-toggle="scroll-to-top"', 179 | scrollSpeed: 500 180 | }, 181 | scrollTo: { 182 | attr: 'data-toggle="scroll-to"', 183 | target: 'data-target', 184 | linkTarget: 'href' 185 | }, 186 | themePanel: { 187 | class: 'app-theme-panel', 188 | toggleAttr: 'data-toggle="theme-panel-expand"', 189 | cookieName: 'app-theme-panel-expand', 190 | activeClass: 'active', 191 | themeListCLass: 'app-theme-list', 192 | themeListItemCLass: 'app-theme-list-item', 193 | themeCoverClass: 'app-theme-cover', 194 | themeCoverItemClass: 'app-theme-cover-item', 195 | theme: { 196 | toggleAttr: 'data-toggle="theme-selector"', 197 | classAttr: 'data-theme-class', 198 | cookieName: 'app-theme', 199 | activeClass: 'active' 200 | }, 201 | themeCover: { 202 | toggleAttr: 'data-toggle="theme-cover-selector"', 203 | classAttr: 'data-theme-cover-class', 204 | cookieName: 'app-theme-cover', 205 | activeClass: 'active' 206 | } 207 | }, 208 | dismissClass: { 209 | toggleAttr: 'data-dismiss-class', 210 | targetAttr: 'data-dismiss-target' 211 | }, 212 | toggleClass: { 213 | toggleAttr: 'data-toggle-class', 214 | targetAttr: 'data-toggle-target' 215 | }, 216 | font: { 217 | family: (getComputedStyle(document.body).getPropertyValue('--bs-body-font-family')).trim(), 218 | size: (getComputedStyle(document.body).getPropertyValue('--bs-body-font-size')).trim(), 219 | weight: (getComputedStyle(document.body).getPropertyValue('--bs-body-font-weight')).trim() 220 | }, 221 | color: { 222 | theme: (getComputedStyle(document.body).getPropertyValue('--bs-theme')).trim(), 223 | blue: (getComputedStyle(document.body).getPropertyValue('--bs-blue')).trim(), 224 | green: (getComputedStyle(document.body).getPropertyValue('--bs-green')).trim(), 225 | orange: (getComputedStyle(document.body).getPropertyValue('--bs-orange')).trim(), 226 | red: (getComputedStyle(document.body).getPropertyValue('--bs-red')).trim(), 227 | cyan: (getComputedStyle(document.body).getPropertyValue('--bs-cyan')).trim(), 228 | purple: (getComputedStyle(document.body).getPropertyValue('--bs-purple')).trim(), 229 | yellow: (getComputedStyle(document.body).getPropertyValue('--bs-yellow')).trim(), 230 | indigo: (getComputedStyle(document.body).getPropertyValue('--bs-indigo')).trim(), 231 | pink: (getComputedStyle(document.body).getPropertyValue('--bs-pink')).trim(), 232 | black: (getComputedStyle(document.body).getPropertyValue('--bs-black')).trim(), 233 | white: (getComputedStyle(document.body).getPropertyValue('--bs-white')).trim(), 234 | gray: (getComputedStyle(document.body).getPropertyValue('--bs-gray')).trim(), 235 | dark: (getComputedStyle(document.body).getPropertyValue('--bs-dark')).trim(), 236 | gray100: (getComputedStyle(document.body).getPropertyValue('--bs-gray-100')).trim(), 237 | gray200: (getComputedStyle(document.body).getPropertyValue('--bs-gray-200')).trim(), 238 | gray300: (getComputedStyle(document.body).getPropertyValue('--bs-gray-300')).trim(), 239 | gray400: (getComputedStyle(document.body).getPropertyValue('--bs-gray-400')).trim(), 240 | gray500: (getComputedStyle(document.body).getPropertyValue('--bs-gray-500')).trim(), 241 | gray600: (getComputedStyle(document.body).getPropertyValue('--bs-gray-600')).trim(), 242 | gray700: (getComputedStyle(document.body).getPropertyValue('--bs-gray-700')).trim(), 243 | gray800: (getComputedStyle(document.body).getPropertyValue('--bs-gray-800')).trim(), 244 | gray900: (getComputedStyle(document.body).getPropertyValue('--bs-gray-900')).trim(), 245 | 246 | themeRgb: (getComputedStyle(document.body).getPropertyValue('--bs-theme-rgb')).trim(), 247 | blueRgb: (getComputedStyle(document.body).getPropertyValue('--bs-blue-rgb')).trim(), 248 | greenRgb: (getComputedStyle(document.body).getPropertyValue('--bs-green-rgb')).trim(), 249 | orangeRgb: (getComputedStyle(document.body).getPropertyValue('--bs-orange-rgb')).trim(), 250 | redRgb: (getComputedStyle(document.body).getPropertyValue('--bs-red-rgb')).trim(), 251 | cyanRgb: (getComputedStyle(document.body).getPropertyValue('--bs-cyan-rgb')).trim(), 252 | purpleRgb: (getComputedStyle(document.body).getPropertyValue('--bs-purple-rgb')).trim(), 253 | yellowRgb: (getComputedStyle(document.body).getPropertyValue('--bs-yellow-rgb')).trim(), 254 | indigoRgb: (getComputedStyle(document.body).getPropertyValue('--bs-indigo-rgb')).trim(), 255 | pinkRgb: (getComputedStyle(document.body).getPropertyValue('--bs-pink-rgb')).trim(), 256 | blackRgb: (getComputedStyle(document.body).getPropertyValue('--bs-black-rgb')).trim(), 257 | whiteRgb: (getComputedStyle(document.body).getPropertyValue('--bs-white-rgb')).trim(), 258 | grayRgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-rgb')).trim(), 259 | darkRgb: (getComputedStyle(document.body).getPropertyValue('--bs-dark-rgb')).trim(), 260 | gray100Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-100-rgb')).trim(), 261 | gray200Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-200-rgb')).trim(), 262 | gray300Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-300-rgb')).trim(), 263 | gray400Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-400-rgb')).trim(), 264 | gray500Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-500-rgb')).trim(), 265 | gray600Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-600-rgb')).trim(), 266 | gray700Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-700-rgb')).trim(), 267 | gray800Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-800-rgb')).trim(), 268 | gray900Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-900-rgb')).trim() 269 | }, 270 | card: { 271 | expand: { 272 | status: false, 273 | toggleAttr: 'data-toggle="card-expand"', 274 | toggleTitle: 'Expand / Compress', 275 | class: 'card-expand' 276 | } 277 | }, 278 | init: { 279 | attr: 'data-init', 280 | class: 'app-init' 281 | } 282 | }; 283 | 284 | 285 | 286 | /* 02. Handle Scrollbar 287 | ------------------------------------------------ */ 288 | var handleScrollbar = function() { 289 | "use strict"; 290 | var elms = document.querySelectorAll('['+ app.scrollBar.attr +']'); 291 | 292 | for (var i = 0; i < elms.length; i++) { 293 | generateScrollbar(elms[i]) 294 | } 295 | }; 296 | var generateScrollbar = function(elm) { 297 | "use strict"; 298 | 299 | if (elm.scrollbarInit || (app.isMobile && elm.getAttribute(app.scrollBar.skipMobileAttr))) { 300 | return; 301 | } 302 | var dataHeight = (!elm.getAttribute(app.scrollBar.heightAttr)) ? elm.offsetHeight : elm.getAttribute(app.scrollBar.heightAttr); 303 | 304 | elm.style.height = dataHeight; 305 | elm.scrollbarInit = true; 306 | 307 | if(app.isMobile) { 308 | elm.style.overflowX = 'scroll'; 309 | } else { 310 | var dataWheelPropagation = (elm.getAttribute(app.scrollBar.wheelPropagationAttr)) ? elm.getAttribute(app.scrollBar.wheelPropagationAttr) : false; 311 | 312 | if (elm.closest('.'+ app.sidebar.class) && elm.closest('.'+ app.sidebar.class).length !== 0) { 313 | app.sidebar.scrollBar.dom = new PerfectScrollbar(elm, { 314 | wheelPropagation: dataWheelPropagation 315 | }); 316 | } else { 317 | new PerfectScrollbar(elm, { 318 | wheelPropagation: dataWheelPropagation 319 | }); 320 | } 321 | } 322 | elm.setAttribute(app.init.attr, true); 323 | elm.classList.remove('invisible'); 324 | }; 325 | 326 | 327 | 328 | /* 03. Handle Sidebar Menu 329 | ------------------------------------------------ */ 330 | var handleSidebarMenuToggle = function(menus) { 331 | menus.map(function(menu) { 332 | menu.onclick = function(e) { 333 | e.preventDefault(); 334 | var target = this.nextElementSibling; 335 | 336 | menus.map(function(m) { 337 | var otherTarget = m.nextElementSibling; 338 | if (otherTarget !== target) { 339 | otherTarget.style.display = 'none'; 340 | otherTarget.closest('.'+ app.sidebar.menu.itemClass).classList.remove(app.sidebar.menu.expandClass); 341 | } 342 | }); 343 | 344 | var targetItemElm = target.closest('.'+ app.sidebar.menu.itemClass); 345 | 346 | if (targetItemElm.classList.contains(app.sidebar.menu.expandClass) || (targetItemElm.classList.contains(app.sidebar.menu.activeClass) && !target.style.display)) { 347 | targetItemElm.classList.remove(app.sidebar.menu.expandClass); 348 | target.style.display = 'none'; 349 | } else { 350 | targetItemElm.classList.add(app.sidebar.menu.expandClass); 351 | target.style.display = 'block'; 352 | } 353 | } 354 | }); 355 | }; 356 | var handleSidebarMenu = function() { 357 | "use strict"; 358 | 359 | var menuBaseSelector = '.'+ app.sidebar.class +' .'+ app.sidebar.menu.class +' > .'+ app.sidebar.menu.itemClass +'.'+ app.sidebar.menu.hasSubClass; 360 | var submenuBaseSelector = ' > .'+ app.sidebar.menu.submenu.class +' > .'+ app.sidebar.menu.itemClass + '.' + app.sidebar.menu.hasSubClass; 361 | 362 | // menu 363 | var menuLinkSelector = menuBaseSelector + ' > .'+ app.sidebar.menu.itemLinkClass; 364 | var menus = [].slice.call(document.querySelectorAll(menuLinkSelector)); 365 | handleSidebarMenuToggle(menus); 366 | 367 | // submenu lvl 1 368 | var submenuLvl1Selector = menuBaseSelector + submenuBaseSelector; 369 | var submenusLvl1 = [].slice.call(document.querySelectorAll(submenuLvl1Selector + ' > .' + app.sidebar.menu.itemLinkClass)); 370 | handleSidebarMenuToggle(submenusLvl1); 371 | 372 | // submenu lvl 2 373 | var submenuLvl2Selector = menuBaseSelector + submenuBaseSelector + submenuBaseSelector; 374 | var submenusLvl2 = [].slice.call(document.querySelectorAll(submenuLvl2Selector + ' > .' + app.sidebar.menu.itemLinkClass)); 375 | handleSidebarMenuToggle(submenusLvl2); 376 | }; 377 | 378 | 379 | 380 | /* 04. Handle Sidebar Scroll Memory 381 | ------------------------------------------------ */ 382 | var handleSidebarScrollMemory = function() { 383 | if (!app.isMobile) { 384 | try { 385 | if (typeof(Storage) !== 'undefined' && typeof(localStorage) !== 'undefined') { 386 | var elm = document.querySelector('.'+ app.sidebar.class +' ['+ app.scrollBar.attr +']'); 387 | 388 | if (elm) { 389 | elm.onscroll = function() { 390 | localStorage.setItem(app.sidebar.scrollBar.localStorage, this.scrollTop); 391 | } 392 | var defaultScroll = localStorage.getItem(app.sidebar.scrollBar.localStorage); 393 | if (defaultScroll) { 394 | document.querySelector('.'+ app.sidebar.class +' ['+ app.scrollBar.attr +']').scrollTop = defaultScroll; 395 | } 396 | } 397 | } 398 | } catch (error) { 399 | console.log(error); 400 | } 401 | } 402 | }; 403 | 404 | 405 | 406 | /* 05. Handle Card Action 407 | ------------------------------------------------ */ 408 | var handleCardAction = function() { 409 | "use strict"; 410 | 411 | if (app.card.expand.status) { 412 | return false; 413 | } 414 | app.card.expand.status = true; 415 | 416 | // expand 417 | var expandTogglerList = [].slice.call(document.querySelectorAll('['+ app.card.expand.toggleAttr +']')); 418 | var expandTogglerTooltipList = expandTogglerList.map(function (expandTogglerEl) { 419 | expandTogglerEl.onclick = function(e) { 420 | e.preventDefault(); 421 | 422 | var target = this.closest('.card'); 423 | var targetClass = app.card.expand.class; 424 | var targetTop = 40; 425 | 426 | if (document.body.classList.contains(targetClass) && target.classList.contains(targetClass)) { 427 | target.removeAttribute('style'); 428 | target.classList.remove(targetClass); 429 | document.body.classList.remove(targetClass); 430 | } else { 431 | document.body.classList.add(targetClass); 432 | target.classList.add(targetClass); 433 | } 434 | 435 | window.dispatchEvent(new Event('resize')); 436 | }; 437 | 438 | return new bootstrap.Tooltip(expandTogglerEl, { 439 | title: app.card.expand.toggleTitle, 440 | placement: 'bottom', 441 | trigger: 'hover', 442 | container: 'body' 443 | }); 444 | }); 445 | }; 446 | 447 | 448 | 449 | /* 06. Handle Tooltip & Popover Activation 450 | ------------------------------------------------ */ 451 | var handelTooltipPopoverActivation = function() { 452 | "use strict"; 453 | 454 | var tooltipTriggerList = [].slice.call(document.querySelectorAll('['+ app.bootstrap.tooltip.attr +']')) 455 | var tooltipList = tooltipTriggerList.map(function (tooltipTriggerEl) { 456 | return new bootstrap.Tooltip(tooltipTriggerEl); 457 | }); 458 | 459 | var popoverTriggerList = [].slice.call(document.querySelectorAll('['+ app.bootstrap.popover.attr +']')) 460 | var popoverList = popoverTriggerList.map(function (popoverTriggerEl) { 461 | return new bootstrap.Popover(popoverTriggerEl); 462 | }); 463 | }; 464 | 465 | 466 | 467 | /* 07. Handle Scroll to Top Button 468 | ------------------------------------------------ */ 469 | var handleScrollToTopButton = function() { 470 | "use strict"; 471 | 472 | var elmTriggerList = [].slice.call(document.querySelectorAll('['+ app.scrollToTopBtn.toggleAttr +']')); 473 | 474 | document.onscroll = function() { 475 | var doc = document.documentElement; 476 | var totalScroll = (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0); 477 | var elmList = elmTriggerList.map(function(elm) { 478 | if (totalScroll >= app.scrollToTopBtn.heightShow) { 479 | if (!elm.classList.contains(app.scrollToTopBtn.showClass)) { 480 | elm.classList.add(app.scrollToTopBtn.showClass); 481 | } 482 | } else { 483 | elm.classList.remove(app.scrollToTopBtn.showClass); 484 | } 485 | }); 486 | 487 | var elm = document.querySelectorAll(app.id)[0]; 488 | 489 | if (totalScroll > 0) { 490 | elm.classList.add(app.header.hasScrollClass); 491 | } else { 492 | elm.classList.remove(app.header.hasScrollClass); 493 | } 494 | } 495 | 496 | var elmList = elmTriggerList.map(function(elm) { 497 | elm.onclick = function(e) { 498 | e.preventDefault(); 499 | 500 | window.scrollTo({top: 0, behavior: 'smooth'}); 501 | } 502 | }); 503 | }; 504 | 505 | 506 | 507 | /* 08. Handle hexToRgba 508 | ------------------------------------------------ */ 509 | var hexToRgba = function(hex, transparent = 1) { 510 | var c; 511 | if(/^#([A-Fa-f0-9]{3}){1,2}$/.test(hex)){ 512 | c= hex.substring(1).split(''); 513 | if(c.length== 3){ 514 | c= [c[0], c[0], c[1], c[1], c[2], c[2]]; 515 | } 516 | c= '0x'+c.join(''); 517 | return 'rgba('+[(c>>16)&255, (c>>8)&255, c&255].join(',')+','+ transparent +')'; 518 | } 519 | throw new Error('Bad Hex'); 520 | }; 521 | 522 | 523 | 524 | /* 09. Handle Scroll To 525 | ------------------------------------------------ */ 526 | var handleScrollTo = function() { 527 | var elmTriggerList = [].slice.call(document.querySelectorAll('['+ app.scrollTo.attr +']')); 528 | var elmList = elmTriggerList.map(function(elm) { 529 | elm.onclick = function(e) { 530 | e.preventDefault(); 531 | 532 | var targetAttr = (elm.getAttribute(app.scrollTo.target)) ? this.getAttribute(app.scrollTo.target) : this.getAttribute(app.scrollTo.linkTarget); 533 | var targetElm = document.querySelectorAll(targetAttr)[0]; 534 | var targetHeader = document.querySelectorAll(app.header.id)[0]; 535 | var targetHeight = targetHeader.offsetHeight; 536 | if (targetElm) { 537 | var targetTop = targetElm.offsetTop - targetHeight - 24; 538 | window.scrollTo({top: targetTop, behavior: 'smooth'}); 539 | } 540 | } 541 | }); 542 | }; 543 | 544 | 545 | 546 | /* 10. Handle Toggle Class 547 | ------------------------------------------------ */ 548 | var handleToggleClass = function() { 549 | var elmList = [].slice.call(document.querySelectorAll('['+ app.toggleClass.toggleAttr +']')); 550 | 551 | elmList.map(function(elm) { 552 | elm.onclick = function(e) { 553 | e.preventDefault(); 554 | 555 | var targetToggleClass = this.getAttribute(app.toggleClass.toggleAttr); 556 | var targetDismissClass = this.getAttribute(app.dismissClass.toggleAttr); 557 | var targetToggleElm = document.querySelector(this.getAttribute(app.toggleClass.targetAttr)); 558 | 559 | if (!targetDismissClass) { 560 | if (targetToggleElm.classList.contains(targetToggleClass)) { 561 | targetToggleElm.classList.remove(targetToggleClass); 562 | } else { 563 | targetToggleElm.classList.add(targetToggleClass); 564 | } 565 | } else { 566 | if (!targetToggleElm.classList.contains(targetToggleClass) && !targetToggleElm.classList.contains(targetDismissClass)) { 567 | if (targetToggleElm.classList.contains(targetToggleClass)) { 568 | targetToggleElm.classList.remove(targetToggleClass); 569 | } else { 570 | targetToggleElm.classList.add(targetToggleClass); 571 | } 572 | } else { 573 | if (targetToggleElm.classList.contains(targetToggleClass)) { 574 | targetToggleElm.classList.remove(targetToggleClass); 575 | } else { 576 | targetToggleElm.classList.add(targetToggleClass); 577 | } 578 | if (targetToggleElm.classList.contains(targetDismissClass)) { 579 | targetToggleElm.classList.remove(targetDismissClass); 580 | } else { 581 | targetToggleElm.classList.add(targetDismissClass); 582 | } 583 | } 584 | } 585 | } 586 | }); 587 | } 588 | 589 | 590 | 591 | /* 11. Handle Theme Panel 592 | ------------------------------------------------ */ 593 | var handleThemePanel = function() { 594 | "use strict"; 595 | 596 | // 12.1 Theme Panel - Toggle / Dismiss 597 | var elmList = [].slice.call(document.querySelectorAll('['+ app.themePanel.toggleAttr +']')); 598 | 599 | elmList.map(function(elm) { 600 | elm.onclick = function(e) { 601 | e.preventDefault(); 602 | 603 | var targetContainer = document.querySelector('.'+ app.themePanel.class); 604 | var targetExpand = false; 605 | 606 | if (targetContainer.classList.contains(app.themePanel.activeClass)) { 607 | targetContainer.classList.remove(app.themePanel.activeClass); 608 | } else { 609 | targetContainer.classList.add(app.themePanel.activeClass); 610 | targetExpand = true; 611 | } 612 | if (Cookies) { 613 | Cookies.set(app.themePanel.cookieName, targetExpand); 614 | } 615 | } 616 | }); 617 | 618 | // 12.2 Theme Panel - Page Load Cookies 619 | if (Cookies) { 620 | var themePanelExpand = Cookies.get(app.themePanel.cookieName); 621 | 622 | if (themePanelExpand == 'true' || typeof themePanelExpand == 'undefined') { 623 | var elm = document.querySelector('['+ app.themePanel.toggleAttr +']'); 624 | if (elm) { 625 | elm.click(); 626 | } 627 | } 628 | } 629 | 630 | 631 | // 12.3 Theme Panel - Theme Selector 632 | var elmList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.theme.toggleAttr +']')); 633 | elmList.map(function(elm) { 634 | elm.onclick = function() { 635 | for (var x = 0; x < document.body.classList.length; x++) { 636 | var targetClass = document.body.classList[x]; 637 | if (targetClass.search('theme-') > -1) { 638 | document.body.classList.remove(targetClass); 639 | } 640 | } 641 | 642 | var targetTheme = this.getAttribute(app.themePanel.theme.classAttr); 643 | var targetThemeList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.theme.toggleAttr +']')); 644 | 645 | if (targetTheme) { 646 | document.body.classList.add(targetTheme); 647 | } 648 | targetThemeList.map(function(targetElm) { 649 | if (targetElm.getAttribute(app.themePanel.theme.classAttr) != targetTheme) { 650 | targetElm.closest('.'+ app.themePanel.themeListItemCLass).classList.remove(app.themePanel.theme.activeClass); 651 | } 652 | }); 653 | 654 | this.closest('.'+ app.themePanel.themeListItemCLass).classList.add(app.themePanel.theme.activeClass); 655 | 656 | if (Cookies) { 657 | Cookies.set(app.themePanel.theme.cookieName, targetTheme); 658 | app.color.theme = (getComputedStyle(document.body).getPropertyValue('--bs-theme')).trim(); 659 | app.color.themeRgb = (getComputedStyle(document.body).getPropertyValue('--bs-theme-rgb')).trim(); 660 | 661 | document.dispatchEvent(new Event('theme-reload')); 662 | } 663 | } 664 | }); 665 | 666 | if (Cookies) { 667 | if (Cookies.get(app.themePanel.theme.cookieName)) { 668 | var targetElm = document.querySelector('.'+ app.themePanel.class +' ['+ app.themePanel.theme.toggleAttr +']' + '['+ app.themePanel.theme.classAttr +'="'+ Cookies.get(app.themePanel.theme.cookieName) +'"]'); 669 | 670 | if (targetElm) { 671 | targetElm.click(); 672 | 673 | app.color.theme = (getComputedStyle(document.body).getPropertyValue('--bs-theme')).trim(); 674 | app.color.themeRgb = (getComputedStyle(document.body).getPropertyValue('--bs-theme-rgb')).trim(); 675 | 676 | document.dispatchEvent(new Event('theme-reload')); 677 | } 678 | } 679 | } 680 | 681 | 682 | // 12.4 Theme Panel - Background Selector 683 | var elmList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.themeCover.toggleAttr +']')); 684 | elmList.map(function(elm) { 685 | elm.onclick = function(e) { 686 | e.preventDefault(); 687 | 688 | var htmlElm = document.querySelector('html'); 689 | var targetTheme = this.getAttribute(app.themePanel.themeCover.classAttr); 690 | for (var x = 0; x < document.documentElement.classList.length; x++) { 691 | var targetClass = document.documentElement.classList[x]; 692 | if (targetClass.search('bg-cover-') > -1) { 693 | htmlElm.classList.remove(targetClass); 694 | } 695 | } 696 | 697 | if (targetTheme) { 698 | htmlElm.classList.add(targetTheme); 699 | } 700 | 701 | var targetCoverList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.themeCover.toggleAttr +']')); 702 | targetCoverList.map(function(targetElm) { 703 | if (targetElm.getAttribute(app.themePanel.themeCover.toggleAttr) != targetTheme) { 704 | targetElm.closest('.'+ app.themePanel.themeCoverItemClass).classList.remove(app.themePanel.themeCover.activeClass); 705 | } 706 | }); 707 | 708 | this.closest('.'+ app.themePanel.themeCoverItemClass).classList.add(app.themePanel.themeCover.activeClass); 709 | if (Cookies) { 710 | Cookies.set(app.themePanel.themeCover.cookieName, targetTheme); 711 | } 712 | }; 713 | }); 714 | 715 | if (Cookies) { 716 | if (Cookies.get(app.themePanel.themeCover.cookieName)) { 717 | var targetElm = document.querySelector('.'+ app.themePanel.class +' ['+ app.themePanel.themeCover.toggleAttr +']' + '['+ app.themePanel.themeCover.classAttr +'="'+ Cookies.get(app.themePanel.themeCover.cookieName) +'"]'); 718 | if (targetElm) { 719 | targetElm.click(); 720 | } 721 | } 722 | } 723 | }; 724 | 725 | 726 | 727 | /* 12. Application Controller 728 | ------------------------------------------------ */ 729 | var App = function () { 730 | "use strict"; 731 | 732 | return { 733 | //main function 734 | init: function () { 735 | this.initComponent(); 736 | this.initSidebar(); 737 | this.initAppLoad(); 738 | }, 739 | initAppLoad: function() { 740 | document.querySelector('body').classList.add(app.init.class); 741 | }, 742 | initSidebar: function() { 743 | handleSidebarMenu(); 744 | handleSidebarScrollMemory(); 745 | }, 746 | initComponent: function() { 747 | handleScrollbar(); 748 | handleScrollToTopButton(); 749 | handleScrollTo(); 750 | handleCardAction(); 751 | handelTooltipPopoverActivation(); 752 | handleToggleClass(); 753 | handleThemePanel(); 754 | }, 755 | scrollTop: function() { 756 | window.scrollTo({top: 0, behavior: 'smooth'}); 757 | } 758 | }; 759 | }(); 760 | 761 | 762 | 763 | /* 13. Initialise 764 | ------------------------------------------------ */ 765 | document.addEventListener('DOMContentLoaded', function() { 766 | App.init(); 767 | }); 768 | //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["app.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"app.min.js","sourcesContent":["/*\nTemplate Name: HUD - Responsive Bootstrap 5 Admin Template\nVersion: 1.9.0\nAuthor: Sean Ngu\nWebsite: http://www.seantheme.com/hud/\n\t----------------------------\n\t\tAPPS CONTENT TABLE\n\t----------------------------\n\n\t<!-- ======== GLOBAL SCRIPT SETTING ======== -->\n  01. Global Variable\n  02. Handle Scrollbar\n  03. Handle Sidebar Menu\n  04. Handle Sidebar Scroll Memory\n  05. Handle Card Action\n  06. Handle Tooltip & Popover Activation\n  07. Handle Scroll to Top Button\n  08. Handle hexToRgba\n  09. Handle Scroll To\n  10. Handle Toggle Class\n  11. Handle Theme Panel\n  12. Application Controller\n  13. Initialise\n\t\n\t<!-- ======== APPLICATION SETTING ======== -->\n\tApplication Controller\n*/\n\n\n\n/* 01. Global Variable\n------------------------------------------------ */\nvar app = {\n\tid: '#app',\n\tisMobile: ((/Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent)) || window.innerWidth < 992),\n\tbootstrap: {\n\t\ttooltip: {\n\t\t\tattr: 'data-bs-toggle=\"tooltip\"'\n\t\t},\n\t\tpopover: {\n\t\t\tattr: 'data-bs-toggle=\"popover\"'\n\t\t},\n\t\tmodal: {\n\t\t\tattr: 'data-bs-toggle=\"modal\"',\n\t\t\tdismissAttr: 'data-bs-dismiss=\"modal\"',\n\t\t\tevent: {\n\t\t\t\thidden: 'hidden.bs.modal'\n\t\t\t}\n\t\t},\n\t\tnav: {\n\t\t\tclass: 'nav',\n\t\t\ttabs: {\n\t\t\t\tclass: 'nav-tabs',\n\t\t\t\tactiveClass: 'active',\n\t\t\t\titemClass: 'nav-item',\n\t\t\t\titemLinkClass: 'nav-link'\n\t\t\t}\n\t\t}\n\t},\n\theader: {\n\t\tid: '#header',\n\t\tclass: 'app-header',\n\t\thasScrollClass: 'has-scroll'\n\t},\n\tsidebar: {\n\t\tid: '#sidebar',\n\t\tclass: 'app-sidebar',\n\t\tscrollBar: {\n\t\t\tlocalStorage: 'appSidebarScrollPosition',\n\t\t\tdom: ''\n\t\t},\n\t\tmenu: {\n\t\t\tclass: 'menu',\n\t\t\tinitAttr: 'data-init',\n\t\t\tanimationTime: 0,\n\t\t\titemClass: 'menu-item',\n\t\t\titemLinkClass: 'menu-link',\n\t\t\thasSubClass: 'has-sub',\n\t\t\tactiveClass: 'active',\n\t\t\texpandingClass: 'expanding',\n\t\t\texpandClass: 'expand',\n\t\t\tsubmenu: {\n\t\t\t\tclass: 'menu-submenu',\n\t\t\t}\n\t\t},\n\t\tmobile: {\n\t\t\ttoggleAttr: 'data-toggle=\"app-sidebar-mobile\"',\n\t\t\tdismissAttr: 'data-dismiss=\"app-sidebar-mobile\"',\n\t\t\ttoggledClass: 'app-sidebar-mobile-toggled',\n\t\t\tclosedClass: 'app-sidebar-mobile-closed',\n\t\t\tbackdrop: {\n\t\t\t\tclass: 'app-sidebar-mobile-backdrop'\n\t\t\t}\n\t\t},\n\t\tminify: {\n\t\t\ttoggleAttr: 'data-toggle=\"app-sidebar-minify\"',\n\t\t\ttoggledClass: 'app-sidebar-minified',\n\t\t\tcookieName: 'app-sidebar-minified'\n\t\t},\n\t\tfloatSubmenu: {\n\t\t\tid: '#app-sidebar-float-submenu',\n\t\t\tdom: '',\n\t\t\ttimeout: '',\n\t\t\tclass: 'app-sidebar-float-submenu',\n\t\t\tcontainer: {\n\t\t\t\tclass: 'app-sidebar-float-submenu-container'\n\t\t\t},\n\t\t\tarrow: {\n\t\t\t\tid: '#app-sidebar-float-submenu-arrow',\n\t\t\t\tclass: 'app-sidebar-float-submenu-arrow'\n\t\t\t},\n\t\t\tline: {\n\t\t\t\tid: '#app-sidebar-float-submenu-line',\n\t\t\t\tclass: 'app-sidebar-float-submenu-line'\n\t\t\t},\n\t\t\toverflow: {\n\t\t\t\tclass: 'overflow-scroll mh-100vh'\n\t\t\t}\n\t\t},\n\t\tsearch: {\n\t\t\tclass: 'menu-search',\n\t\t\ttoggleAttr: 'data-sidebar-search=\"true\"',\n\t\t\thideClass: 'd-none',\n\t\t\tfoundClass: 'has-text'\n\t\t},\n\t\ttransparent: {\n\t\t\tclass: 'app-sidebar-transparent'\n\t\t}\n\t},\n\tscrollBar: {\n\t\tattr: 'data-scrollbar=\"true\"',\n\t\tskipMobileAttr: 'data-skip-mobile',\n\t\theightAttr: 'data-height',\n\t\twheelPropagationAttr: 'data-wheel-propagation'\n\t},\n\tcontent: {\n\t\tid: '#content',\n\t\tclass: 'app-content',\n\t\tfullHeight: {\n\t\t\tclass: 'app-content-full-height'\n\t\t},\n\t\tfullWidth: {\n\t\t\tclass: 'app-content-full-width'\n\t\t}\n\t},\n\tlayout: {\n\t\tsidebarLight: {\n\t\t\tclass: 'app-with-light-sidebar'\n\t\t},\n\t\tsidebarEnd: {\n\t\t\tclass: 'app-with-end-sidebar'\n\t\t},\n\t\tsidebarWide: {\n\t\t\tclass: 'app-with-wide-sidebar'\n\t\t},\n\t\tsidebarMinified: {\n\t\t\tclass: 'app-sidebar-minified'\n\t\t},\n\t\tsidebarTwo: {\n\t\t\tclass: 'app-with-two-sidebar'\n\t\t},\n\t\twithoutHeader: {\n\t\t\tclass: 'app-without-header'\n\t\t},\n\t\twithoutSidebar: {\n\t\t\tclass: 'app-without-sidebar'\n\t\t},\n\t\ttopMenu: {\n\t\t\tclass: 'app-with-top-menu'\n\t\t},\n\t\tboxedLayout: {\n\t\t\tclass: 'boxed-layout'\n\t\t}\n\t},\n\tscrollToTopBtn: {\n\t\tshowClass: 'show',\n\t\theightShow: 200,\n\t\ttoggleAttr: 'data-toggle=\"scroll-to-top\"',\n\t\tscrollSpeed: 500\n\t},\n\tscrollTo: {\n\t\tattr: 'data-toggle=\"scroll-to\"',\n\t\ttarget: 'data-target',\n\t\tlinkTarget: 'href'\n\t},\n\tthemePanel: {\n\t\tclass: 'app-theme-panel',\n\t\ttoggleAttr: 'data-toggle=\"theme-panel-expand\"',\n\t\tcookieName: 'app-theme-panel-expand',\n\t\tactiveClass: 'active',\n\t\tthemeListCLass: 'app-theme-list',\n\t\tthemeListItemCLass: 'app-theme-list-item',\n\t\tthemeCoverClass: 'app-theme-cover',\n\t\tthemeCoverItemClass: 'app-theme-cover-item',\n\t\ttheme: {\n\t\t\ttoggleAttr: 'data-toggle=\"theme-selector\"',\n\t\t\tclassAttr: 'data-theme-class',\n\t\t\tcookieName: 'app-theme',\n\t\t\tactiveClass: 'active'\n\t\t},\n\t\tthemeCover: {\n\t\t\ttoggleAttr: 'data-toggle=\"theme-cover-selector\"',\n\t\t\tclassAttr: 'data-theme-cover-class',\n\t\t\tcookieName: 'app-theme-cover',\n\t\t\tactiveClass: 'active'\n\t\t}\n\t},\n\tdismissClass: {\n\t\ttoggleAttr: 'data-dismiss-class',\n\t\ttargetAttr: 'data-dismiss-target'\n\t},\n\ttoggleClass: {\n\t\ttoggleAttr: 'data-toggle-class',\n\t\ttargetAttr: 'data-toggle-target'\n\t},\n\tfont: {\n\t\tfamily: (getComputedStyle(document.body).getPropertyValue('--bs-body-font-family')).trim(),\n\t\tsize: (getComputedStyle(document.body).getPropertyValue('--bs-body-font-size')).trim(),\n\t\tweight: (getComputedStyle(document.body).getPropertyValue('--bs-body-font-weight')).trim()\n\t},\n\tcolor: {\n\t\ttheme: (getComputedStyle(document.body).getPropertyValue('--bs-theme')).trim(),\n\t\tblue: (getComputedStyle(document.body).getPropertyValue('--bs-blue')).trim(),\n\t\tgreen: (getComputedStyle(document.body).getPropertyValue('--bs-green')).trim(),\n\t\torange: (getComputedStyle(document.body).getPropertyValue('--bs-orange')).trim(),\n\t\tred: (getComputedStyle(document.body).getPropertyValue('--bs-red')).trim(),\n\t\tcyan: (getComputedStyle(document.body).getPropertyValue('--bs-cyan')).trim(),\n\t\tpurple: (getComputedStyle(document.body).getPropertyValue('--bs-purple')).trim(),\n\t\tyellow: (getComputedStyle(document.body).getPropertyValue('--bs-yellow')).trim(),\n\t\tindigo: (getComputedStyle(document.body).getPropertyValue('--bs-indigo')).trim(),\n\t\tpink: (getComputedStyle(document.body).getPropertyValue('--bs-pink')).trim(),\n\t\tblack: (getComputedStyle(document.body).getPropertyValue('--bs-black')).trim(),\n\t\twhite: (getComputedStyle(document.body).getPropertyValue('--bs-white')).trim(),\n\t\tgray: (getComputedStyle(document.body).getPropertyValue('--bs-gray')).trim(),\n\t\tdark: (getComputedStyle(document.body).getPropertyValue('--bs-dark')).trim(),\n\t\tgray100: (getComputedStyle(document.body).getPropertyValue('--bs-gray-100')).trim(),\n\t\tgray200: (getComputedStyle(document.body).getPropertyValue('--bs-gray-200')).trim(),\n\t\tgray300: (getComputedStyle(document.body).getPropertyValue('--bs-gray-300')).trim(),\n\t\tgray400: (getComputedStyle(document.body).getPropertyValue('--bs-gray-400')).trim(),\n\t\tgray500: (getComputedStyle(document.body).getPropertyValue('--bs-gray-500')).trim(),\n\t\tgray600: (getComputedStyle(document.body).getPropertyValue('--bs-gray-600')).trim(),\n\t\tgray700: (getComputedStyle(document.body).getPropertyValue('--bs-gray-700')).trim(),\n\t\tgray800: (getComputedStyle(document.body).getPropertyValue('--bs-gray-800')).trim(),\n\t\tgray900: (getComputedStyle(document.body).getPropertyValue('--bs-gray-900')).trim(),\n\t\t\n\t\tthemeRgb: (getComputedStyle(document.body).getPropertyValue('--bs-theme-rgb')).trim(),\n\t\tblueRgb: (getComputedStyle(document.body).getPropertyValue('--bs-blue-rgb')).trim(),\n\t\tgreenRgb: (getComputedStyle(document.body).getPropertyValue('--bs-green-rgb')).trim(),\n\t\torangeRgb: (getComputedStyle(document.body).getPropertyValue('--bs-orange-rgb')).trim(),\n\t\tredRgb: (getComputedStyle(document.body).getPropertyValue('--bs-red-rgb')).trim(),\n\t\tcyanRgb: (getComputedStyle(document.body).getPropertyValue('--bs-cyan-rgb')).trim(),\n\t\tpurpleRgb: (getComputedStyle(document.body).getPropertyValue('--bs-purple-rgb')).trim(),\n\t\tyellowRgb: (getComputedStyle(document.body).getPropertyValue('--bs-yellow-rgb')).trim(),\n\t\tindigoRgb: (getComputedStyle(document.body).getPropertyValue('--bs-indigo-rgb')).trim(),\n\t\tpinkRgb: (getComputedStyle(document.body).getPropertyValue('--bs-pink-rgb')).trim(),\n\t\tblackRgb: (getComputedStyle(document.body).getPropertyValue('--bs-black-rgb')).trim(),\n\t\twhiteRgb: (getComputedStyle(document.body).getPropertyValue('--bs-white-rgb')).trim(),\n\t\tgrayRgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-rgb')).trim(),\n\t\tdarkRgb: (getComputedStyle(document.body).getPropertyValue('--bs-dark-rgb')).trim(),\n\t\tgray100Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-100-rgb')).trim(),\n\t\tgray200Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-200-rgb')).trim(),\n\t\tgray300Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-300-rgb')).trim(),\n\t\tgray400Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-400-rgb')).trim(),\n\t\tgray500Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-500-rgb')).trim(),\n\t\tgray600Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-600-rgb')).trim(),\n\t\tgray700Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-700-rgb')).trim(),\n\t\tgray800Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-800-rgb')).trim(),\n\t\tgray900Rgb: (getComputedStyle(document.body).getPropertyValue('--bs-gray-900-rgb')).trim()\n\t},\n\tcard: {\n\t\texpand: {\n\t\t\tstatus: false,\n\t\t\ttoggleAttr: 'data-toggle=\"card-expand\"',\n\t\t\ttoggleTitle: 'Expand / Compress',\n\t\t\tclass: 'card-expand'\n\t\t}\n\t},\n\tinit: {\n\t\tattr: 'data-init',\n\t\tclass: 'app-init'\n\t}\n};\n\n\n\n/* 02. Handle Scrollbar\n------------------------------------------------ */\nvar handleScrollbar = function() {\n\t\"use strict\";\n\tvar elms = document.querySelectorAll('['+ app.scrollBar.attr +']');\n\t\t\n\tfor (var i = 0; i < elms.length; i++) {\n\t\tgenerateScrollbar(elms[i])\n\t}\n};\nvar generateScrollbar = function(elm) {\n  \"use strict\";\n\t\n\tif (elm.scrollbarInit || (app.isMobile && elm.getAttribute(app.scrollBar.skipMobileAttr))) {\n\t\treturn;\n\t}\n\tvar dataHeight = (!elm.getAttribute(app.scrollBar.heightAttr)) ? elm.offsetHeight : elm.getAttribute(app.scrollBar.heightAttr);\n\t\n\telm.style.height = dataHeight;\n\telm.scrollbarInit = true;\n\t\n\tif(app.isMobile) {\n\t\telm.style.overflowX = 'scroll';\n\t} else {\n\t\tvar dataWheelPropagation = (elm.getAttribute(app.scrollBar.wheelPropagationAttr)) ? elm.getAttribute(app.scrollBar.wheelPropagationAttr) : false;\n\t\t\n\t\tif (elm.closest('.'+ app.sidebar.class) && elm.closest('.'+ app.sidebar.class).length !== 0) {\n\t\t\tapp.sidebar.scrollBar.dom = new PerfectScrollbar(elm, {\n\t\t\t\twheelPropagation: dataWheelPropagation\n\t\t\t});\n\t\t} else {\n\t\t\tnew PerfectScrollbar(elm, {\n\t\t\t\twheelPropagation: dataWheelPropagation\n\t\t\t});\n\t\t}\n\t}\n\telm.setAttribute(app.init.attr, true);\n\telm.classList.remove('invisible');\n};\n\n\n\n/* 03. Handle Sidebar Menu\n------------------------------------------------ */\nvar handleSidebarMenuToggle = function(menus) {\n\tmenus.map(function(menu) {\n\t\tmenu.onclick = function(e) {\n\t\t\te.preventDefault();\n\t\t\tvar target = this.nextElementSibling;\n\t\n\t\t\tmenus.map(function(m) {\n\t\t\t\tvar otherTarget = m.nextElementSibling;\n\t\t\t\tif (otherTarget !== target) {\n\t\t\t\t\totherTarget.style.display = 'none';\n\t\t\t\t\totherTarget.closest('.'+ app.sidebar.menu.itemClass).classList.remove(app.sidebar.menu.expandClass);\n\t\t\t\t}\n\t\t\t});\n\t\n\t\t\tvar targetItemElm = target.closest('.'+ app.sidebar.menu.itemClass);\n\n\t\t\tif (targetItemElm.classList.contains(app.sidebar.menu.expandClass) || (targetItemElm.classList.contains(app.sidebar.menu.activeClass) && !target.style.display)) {\n\t\t\t\ttargetItemElm.classList.remove(app.sidebar.menu.expandClass);\n\t\t\t\ttarget.style.display = 'none';\n\t\t\t} else {\n\t\t\t\ttargetItemElm.classList.add(app.sidebar.menu.expandClass);\n\t\t\t\ttarget.style.display = 'block';\n\t\t\t}\n\t\t}\n\t});\n};\nvar handleSidebarMenu = function() {\n\t\"use strict\";\n\t\n\tvar menuBaseSelector = '.'+ app.sidebar.class +' .'+ app.sidebar.menu.class +' > .'+ app.sidebar.menu.itemClass +'.'+ app.sidebar.menu.hasSubClass;\n\tvar submenuBaseSelector = ' > .'+ app.sidebar.menu.submenu.class +' > .'+ app.sidebar.menu.itemClass + '.' + app.sidebar.menu.hasSubClass;\n\t\n\t// menu\n\tvar menuLinkSelector =  menuBaseSelector + ' > .'+ app.sidebar.menu.itemLinkClass;\n\tvar menus = [].slice.call(document.querySelectorAll(menuLinkSelector));\n\thandleSidebarMenuToggle(menus);\n\t\n\t// submenu lvl 1\n\tvar submenuLvl1Selector = menuBaseSelector + submenuBaseSelector;\n\tvar submenusLvl1 = [].slice.call(document.querySelectorAll(submenuLvl1Selector + ' > .' + app.sidebar.menu.itemLinkClass));\n\thandleSidebarMenuToggle(submenusLvl1);\n\t\n\t// submenu lvl 2\n\tvar submenuLvl2Selector = menuBaseSelector + submenuBaseSelector + submenuBaseSelector;\n\tvar submenusLvl2 = [].slice.call(document.querySelectorAll(submenuLvl2Selector + ' > .' + app.sidebar.menu.itemLinkClass));\n\thandleSidebarMenuToggle(submenusLvl2);\n};\n\n\n\n/* 04. Handle Sidebar Scroll Memory\n------------------------------------------------ */\nvar handleSidebarScrollMemory = function() {\n\tif (!app.isMobile) {\n\t\ttry {\n\t\t\tif (typeof(Storage) !== 'undefined' && typeof(localStorage) !== 'undefined') {\n\t\t\t\tvar elm = document.querySelector('.'+ app.sidebar.class +' ['+ app.scrollBar.attr +']');\n\t\t\t\t\n\t\t\t\tif (elm) {\n\t\t\t\t\telm.onscroll = function() {\n\t\t\t\t\t\tlocalStorage.setItem(app.sidebar.scrollBar.localStorage, this.scrollTop);\n\t\t\t\t\t}\n\t\t\t\t\tvar defaultScroll = localStorage.getItem(app.sidebar.scrollBar.localStorage);\n\t\t\t\t\tif (defaultScroll) {\n\t\t\t\t\t\tdocument.querySelector('.'+ app.sidebar.class +' ['+ app.scrollBar.attr +']').scrollTop = defaultScroll;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tconsole.log(error);\n\t\t}\n\t}\n};\n\n\n\n/* 05. Handle Card Action\n------------------------------------------------ */\nvar handleCardAction = function() {\n\t\"use strict\";\n\n\tif (app.card.expand.status) {\n\t\treturn false;\n\t}\n\tapp.card.expand.status = true;\n\n\t// expand\n\tvar expandTogglerList = [].slice.call(document.querySelectorAll('['+ app.card.expand.toggleAttr +']'));\n\tvar expandTogglerTooltipList = expandTogglerList.map(function (expandTogglerEl) {\n\t\texpandTogglerEl.onclick = function(e) {\n\t\t\te.preventDefault();\n\t\t\n\t\t\tvar target = this.closest('.card');\n\t\t\tvar targetClass = app.card.expand.class;\n\t\t\tvar targetTop = 40;\n\n\t\t\tif (document.body.classList.contains(targetClass) && target.classList.contains(targetClass)) {\n\t\t\t\ttarget.removeAttribute('style');\n\t\t\t\ttarget.classList.remove(targetClass);\n\t\t\t\tdocument.body.classList.remove(targetClass);\n\t\t\t} else {\n\t\t\t\tdocument.body.classList.add(targetClass);\n\t\t\t\ttarget.classList.add(targetClass);\n\t\t\t}\n\t\t\n\t\t\twindow.dispatchEvent(new Event('resize'));\n\t\t};\n\t\n\t\treturn new bootstrap.Tooltip(expandTogglerEl, {\n\t\t\ttitle: app.card.expand.toggleTitle,\n\t\t\tplacement: 'bottom',\n\t\t\ttrigger: 'hover',\n\t\t\tcontainer: 'body'\n\t\t});\n\t});\n};\n\n\n\n/* 06. Handle Tooltip & Popover Activation\n------------------------------------------------ */\nvar handelTooltipPopoverActivation = function() {\n\t\"use strict\";\n\t\n\tvar tooltipTriggerList = [].slice.call(document.querySelectorAll('['+ app.bootstrap.tooltip.attr +']'))\n\tvar tooltipList = tooltipTriggerList.map(function (tooltipTriggerEl) {\n\t\treturn new bootstrap.Tooltip(tooltipTriggerEl);\n\t});\n\t\n\tvar popoverTriggerList = [].slice.call(document.querySelectorAll('['+ app.bootstrap.popover.attr +']'))\n\tvar popoverList = popoverTriggerList.map(function (popoverTriggerEl) {\n\t\treturn new bootstrap.Popover(popoverTriggerEl);\n\t});\n};\n\n\n\n/* 07. Handle Scroll to Top Button\n------------------------------------------------ */\nvar handleScrollToTopButton = function() {\n\t\"use strict\";\n\t\n\tvar elmTriggerList = [].slice.call(document.querySelectorAll('['+ app.scrollToTopBtn.toggleAttr +']'));\n\t\n\tdocument.onscroll = function() {\n\t\tvar doc = document.documentElement;\n\t\tvar totalScroll = (window.pageYOffset || doc.scrollTop)  - (doc.clientTop || 0);\n\t\tvar elmList = elmTriggerList.map(function(elm) {\n\t\t\tif (totalScroll >= app.scrollToTopBtn.heightShow) {\n\t\t\t\tif (!elm.classList.contains(app.scrollToTopBtn.showClass)) {\n\t\t\t\t\telm.classList.add(app.scrollToTopBtn.showClass);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\telm.classList.remove(app.scrollToTopBtn.showClass);\n\t\t\t}\n\t\t});\n\t\t\n\t\tvar elm = document.querySelectorAll(app.id)[0];\n\t\n\t\tif (totalScroll > 0) {\n\t\t\telm.classList.add(app.header.hasScrollClass);\n\t\t} else {\n\t\t\telm.classList.remove(app.header.hasScrollClass);\n\t\t}\n\t}\n\t\n\tvar elmList = elmTriggerList.map(function(elm) {\n\t\telm.onclick = function(e) {\n\t\t\te.preventDefault();\n\t\t\t\n\t\t\twindow.scrollTo({top: 0, behavior: 'smooth'});\n\t\t}\n\t});\n};\n\n\n\n/* 08. Handle hexToRgba\n------------------------------------------------ */\nvar hexToRgba = function(hex, transparent = 1) {\n\tvar c;\n\tif(/^#([A-Fa-f0-9]{3}){1,2}$/.test(hex)){\n\t\tc= hex.substring(1).split('');\n\t\tif(c.length== 3){\n\t\t\t\tc= [c[0], c[0], c[1], c[1], c[2], c[2]];\n\t\t}\n\t\tc= '0x'+c.join('');\n\t\treturn 'rgba('+[(c>>16)&255, (c>>8)&255, c&255].join(',')+','+ transparent +')';\n\t}\n  throw new Error('Bad Hex');\n};\n\n\n\n/* 09. Handle Scroll To\n------------------------------------------------ */\nvar handleScrollTo = function() {\n\tvar elmTriggerList = [].slice.call(document.querySelectorAll('['+ app.scrollTo.attr +']'));\n\tvar elmList = elmTriggerList.map(function(elm) {\n\t\telm.onclick = function(e) {\n\t\t\te.preventDefault();\n\t\t\n\t\t\tvar targetAttr = (elm.getAttribute(app.scrollTo.target)) ? this.getAttribute(app.scrollTo.target) : this.getAttribute(app.scrollTo.linkTarget);\n\t\t\tvar targetElm = document.querySelectorAll(targetAttr)[0];\n\t\t\tvar targetHeader = document.querySelectorAll(app.header.id)[0];\n\t\t\tvar targetHeight = targetHeader.offsetHeight;\n\t\t\tif (targetElm) {\n\t\t\t\tvar targetTop = targetElm.offsetTop - targetHeight - 24;\n\t\t\t\twindow.scrollTo({top: targetTop, behavior: 'smooth'});\n\t\t\t}\n\t\t}\n\t});\n};\n\n\n\n/* 10. Handle Toggle Class\n------------------------------------------------ */\nvar handleToggleClass = function() {\n\tvar elmList = [].slice.call(document.querySelectorAll('['+ app.toggleClass.toggleAttr +']'));\n\t\n\telmList.map(function(elm) {\n\t\telm.onclick = function(e) {\n\t\t\te.preventDefault();\n\t\t\t\n\t\t\tvar targetToggleClass = this.getAttribute(app.toggleClass.toggleAttr);\n\t\t\tvar targetDismissClass = this.getAttribute(app.dismissClass.toggleAttr);\n\t\t\tvar targetToggleElm = document.querySelector(this.getAttribute(app.toggleClass.targetAttr));\n\t\t\n\t\t\tif (!targetDismissClass) {\n\t\t\t\tif (targetToggleElm.classList.contains(targetToggleClass)) {\n\t\t\t\t\ttargetToggleElm.classList.remove(targetToggleClass);\n\t\t\t\t} else {\n\t\t\t\t\ttargetToggleElm.classList.add(targetToggleClass);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tif (!targetToggleElm.classList.contains(targetToggleClass) && !targetToggleElm.classList.contains(targetDismissClass)) {\n\t\t\t\t\tif (targetToggleElm.classList.contains(targetToggleClass)) {\n\t\t\t\t\t\ttargetToggleElm.classList.remove(targetToggleClass);\n\t\t\t\t\t} else {\n\t\t\t\t\t\ttargetToggleElm.classList.add(targetToggleClass);\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\tif (targetToggleElm.classList.contains(targetToggleClass)) {\n\t\t\t\t\t\ttargetToggleElm.classList.remove(targetToggleClass);\n\t\t\t\t\t} else {\n\t\t\t\t\t\ttargetToggleElm.classList.add(targetToggleClass);\n\t\t\t\t\t}\n\t\t\t\t\tif (targetToggleElm.classList.contains(targetDismissClass)) {\n\t\t\t\t\t\ttargetToggleElm.classList.remove(targetDismissClass);\n\t\t\t\t\t} else {\n\t\t\t\t\t\ttargetToggleElm.classList.add(targetDismissClass);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t});\n}\n\n\n\n/* 11. Handle Theme Panel\n------------------------------------------------ */\nvar handleThemePanel = function() {\n\t\"use strict\";\n\t\n\t// 12.1 Theme Panel - Toggle / Dismiss\n\tvar elmList = [].slice.call(document.querySelectorAll('['+ app.themePanel.toggleAttr +']'));\n\t\n\telmList.map(function(elm) {\n\t\telm.onclick = function(e) {\n\t\t\te.preventDefault();\n\t\t\t\n\t\t\tvar targetContainer = document.querySelector('.'+ app.themePanel.class);\n\t\t\tvar targetExpand = false;\n\t\t\n\t\t\tif (targetContainer.classList.contains(app.themePanel.activeClass)) {\n\t\t\t\ttargetContainer.classList.remove(app.themePanel.activeClass);\n\t\t\t} else {\n\t\t\t\ttargetContainer.classList.add(app.themePanel.activeClass);\n\t\t\t\ttargetExpand = true;\n\t\t\t}\n\t\t\tif (Cookies) {\n\t\t\t\tCookies.set(app.themePanel.cookieName, targetExpand);\n\t\t\t}\n\t\t}\n\t});\n\t\n\t// 12.2 Theme Panel - Page Load Cookies \n\tif (Cookies) {\n\t\tvar themePanelExpand = Cookies.get(app.themePanel.cookieName);\n\t\t\n\t\tif (themePanelExpand == 'true' || typeof themePanelExpand == 'undefined') {\n\t\t\tvar elm = document.querySelector('['+ app.themePanel.toggleAttr +']');\n\t\t\tif (elm) {\n\t\t\t\telm.click();\n\t\t\t}\n\t\t}\n\t}\n\t\n\t\n\t// 12.3 Theme Panel - Theme Selector\n\tvar elmList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.theme.toggleAttr +']'));\n\telmList.map(function(elm) {\n\t\telm.onclick = function() {\n\t\t\tfor (var x = 0; x < document.body.classList.length; x++) {\n\t\t\t\tvar targetClass = document.body.classList[x];\n\t\t\t\tif (targetClass.search('theme-') > -1) {\n\t\t\t\t\tdocument.body.classList.remove(targetClass);\n\t\t\t\t}\n\t\t\t}\n\t\t\n\t\t\tvar targetTheme = this.getAttribute(app.themePanel.theme.classAttr);\n\t\t\tvar targetThemeList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.theme.toggleAttr +']'));\n\t\t\t\n\t\t\tif (targetTheme) {\n\t\t\t\tdocument.body.classList.add(targetTheme);\n\t\t\t}\n\t\t\ttargetThemeList.map(function(targetElm) {\n\t\t\t\tif (targetElm.getAttribute(app.themePanel.theme.classAttr) != targetTheme) {\n\t\t\t\t\ttargetElm.closest('.'+ app.themePanel.themeListItemCLass).classList.remove(app.themePanel.theme.activeClass);\n\t\t\t\t}\n\t\t\t});\n\t\t\t\n\t\t\tthis.closest('.'+ app.themePanel.themeListItemCLass).classList.add(app.themePanel.theme.activeClass);\n\t\t\t\n\t\t\tif (Cookies) {\n\t\t\t\tCookies.set(app.themePanel.theme.cookieName, targetTheme);\n\t\t\t\tapp.color.theme = (getComputedStyle(document.body).getPropertyValue('--bs-theme')).trim();\n\t\t\t\tapp.color.themeRgb = (getComputedStyle(document.body).getPropertyValue('--bs-theme-rgb')).trim();\n\t\t\t\t\n\t\t\t\tdocument.dispatchEvent(new Event('theme-reload'));\n\t\t\t}\n\t\t}\n\t});\n\t\n\tif (Cookies) {\n\t\tif (Cookies.get(app.themePanel.theme.cookieName)) {\n\t\t\tvar targetElm = document.querySelector('.'+ app.themePanel.class +' ['+ app.themePanel.theme.toggleAttr +']' + '['+ app.themePanel.theme.classAttr +'=\"'+ Cookies.get(app.themePanel.theme.cookieName) +'\"]');\n\t\t\t\n\t\t\tif (targetElm) {\n\t\t\t\ttargetElm.click();\n\t\t\t\n\t\t\t\tapp.color.theme = (getComputedStyle(document.body).getPropertyValue('--bs-theme')).trim();\n\t\t\t\tapp.color.themeRgb = (getComputedStyle(document.body).getPropertyValue('--bs-theme-rgb')).trim();\n\t\t\t\n\t\t\t\tdocument.dispatchEvent(new Event('theme-reload'));\n\t\t\t}\n\t\t}\n\t}\n\t\n\t\n\t// 12.4 Theme Panel - Background Selector\n\tvar elmList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.themeCover.toggleAttr +']'));\n\telmList.map(function(elm) {\n\t\telm.onclick = function(e) {\n\t\t\te.preventDefault();\n\t\t\t\n\t\t\tvar htmlElm = document.querySelector('html');\n\t\t\tvar targetTheme = this.getAttribute(app.themePanel.themeCover.classAttr);\n\t\t\tfor (var x = 0; x < document.documentElement.classList.length; x++) {\n\t\t\t\tvar targetClass = document.documentElement.classList[x];\n\t\t\t\tif (targetClass.search('bg-cover-') > -1) {\n\t\t\t\t\thtmlElm.classList.remove(targetClass);\n\t\t\t\t}\n\t\t\t}\n\t\t\t\n\t\t\tif (targetTheme) {\n\t\t\t\thtmlElm.classList.add(targetTheme);\n\t\t\t}\n\t\t\t\n\t\t\tvar targetCoverList = [].slice.call(document.querySelectorAll('.'+ app.themePanel.class +' ['+ app.themePanel.themeCover.toggleAttr +']'));\n\t\t\ttargetCoverList.map(function(targetElm) {\n\t\t\t\tif (targetElm.getAttribute(app.themePanel.themeCover.toggleAttr) != targetTheme) {\n\t\t\t\t\ttargetElm.closest('.'+ app.themePanel.themeCoverItemClass).classList.remove(app.themePanel.themeCover.activeClass);\n\t\t\t\t}\n\t\t\t});\n\t\t\t\n\t\t\tthis.closest('.'+ app.themePanel.themeCoverItemClass).classList.add(app.themePanel.themeCover.activeClass);\n\t\t\tif (Cookies) {\n\t\t\t\tCookies.set(app.themePanel.themeCover.cookieName, targetTheme);\n\t\t\t}\n\t\t};\n\t});\n\t\n\tif (Cookies) {\n\t\tif (Cookies.get(app.themePanel.themeCover.cookieName)) {\n\t\t\tvar targetElm = document.querySelector('.'+ app.themePanel.class +' ['+ app.themePanel.themeCover.toggleAttr +']' + '['+ app.themePanel.themeCover.classAttr +'=\"'+ Cookies.get(app.themePanel.themeCover.cookieName) +'\"]');\n\t\t\tif (targetElm) {\n\t\t\t\ttargetElm.click();\n\t\t\t}\n\t\t}\n\t}\n};\n\n\n\n/* 12. Application Controller\n------------------------------------------------ */\nvar App = function () {\n\t\"use strict\";\n\t\n\treturn {\n\t\t//main function\n\t\tinit: function () {\n\t\t\tthis.initComponent();\n\t\t\tthis.initSidebar();\n\t\t\tthis.initAppLoad();\n\t\t},\n\t\tinitAppLoad: function() {\n\t\t\tdocument.querySelector('body').classList.add(app.init.class);\n\t\t},\n\t\tinitSidebar: function() {\n\t\t\thandleSidebarMenu();\n\t\t\thandleSidebarScrollMemory();\n\t\t},\n\t\tinitComponent: function() {\n\t\t\thandleScrollbar();\n\t\t\thandleScrollToTopButton();\n\t\t\thandleScrollTo();\n\t\t\thandleCardAction();\n\t\t\thandelTooltipPopoverActivation();\n\t\t\thandleToggleClass();\n\t\t\thandleThemePanel();\n\t\t},\n\t\tscrollTop: function() {\n\t\t\twindow.scrollTo({top: 0, behavior: 'smooth'});\n\t\t}\n\t};\n}();\n\n\n\n/* 13. Initialise\n------------------------------------------------ */\ndocument.addEventListener('DOMContentLoaded', function() {\n\tApp.init();\n});"]} 769 | -------------------------------------------------------------------------------- /static/js/zoom.js: -------------------------------------------------------------------------------- 1 | +function ($) { "use strict"; 2 | 3 | /** 4 | * The zoom service 5 | */ 6 | function ZoomService () { 7 | this._activeZoom = 8 | this._initialScrollPosition = 9 | this._initialTouchPosition = 10 | this._touchMoveListener = null 11 | 12 | this._$document = $(document) 13 | this._$window = $(window) 14 | this._$body = $(document.body) 15 | 16 | this._boundClick = $.proxy(this._clickHandler, this) 17 | } 18 | 19 | ZoomService.prototype.listen = function () { 20 | this._$body.on('click', '[data-action="zoom"]', $.proxy(this._zoom, this)) 21 | } 22 | 23 | ZoomService.prototype._zoom = function (e) { 24 | var target = e.target 25 | 26 | if (!target || target.tagName != 'IMG') return 27 | 28 | if (this._$body.hasClass('zoom-overlay-open')) return 29 | 30 | if (e.metaKey || e.ctrlKey) { 31 | return window.open((e.target.getAttribute('data-original') || e.target.src), '_blank') 32 | } 33 | 34 | if (target.width >= ($(window).width() - Zoom.OFFSET)) return 35 | 36 | this._activeZoomClose(true) 37 | 38 | this._activeZoom = new Zoom(target) 39 | this._activeZoom.zoomImage() 40 | 41 | // todo(fat): probably worth throttling this 42 | this._$window.on('scroll.zoom', $.proxy(this._scrollHandler, this)) 43 | 44 | this._$document.on('keyup.zoom', $.proxy(this._keyHandler, this)) 45 | this._$document.on('touchstart.zoom', $.proxy(this._touchStart, this)) 46 | 47 | // we use a capturing phase here to prevent unintended js events 48 | // sadly no useCapture in jquery api (http://bugs.jquery.com/ticket/14953) 49 | if (document.addEventListener) { 50 | document.addEventListener('click', this._boundClick, true) 51 | } else { 52 | document.attachEvent('onclick', this._boundClick, true) 53 | } 54 | 55 | if ('bubbles' in e) { 56 | if (e.bubbles) e.stopPropagation() 57 | } else { 58 | // Internet Explorer before version 9 59 | e.cancelBubble = true 60 | } 61 | } 62 | 63 | ZoomService.prototype._activeZoomClose = function (forceDispose) { 64 | if (!this._activeZoom) return 65 | 66 | if (forceDispose) { 67 | this._activeZoom.dispose() 68 | } else { 69 | this._activeZoom.close() 70 | } 71 | 72 | this._$window.off('.zoom') 73 | this._$document.off('.zoom') 74 | 75 | document.removeEventListener('click', this._boundClick, true) 76 | 77 | this._activeZoom = null 78 | } 79 | 80 | ZoomService.prototype._scrollHandler = function (e) { 81 | if (this._initialScrollPosition === null) this._initialScrollPosition = $(window).scrollTop() 82 | var deltaY = this._initialScrollPosition - $(window).scrollTop() 83 | if (Math.abs(deltaY) >= 40) this._activeZoomClose() 84 | } 85 | 86 | ZoomService.prototype._keyHandler = function (e) { 87 | if (e.keyCode == 27) this._activeZoomClose() 88 | } 89 | 90 | ZoomService.prototype._clickHandler = function (e) { 91 | if (e.preventDefault) e.preventDefault() 92 | else event.returnValue = false 93 | 94 | if ('bubbles' in e) { 95 | if (e.bubbles) e.stopPropagation() 96 | } else { 97 | // Internet Explorer before version 9 98 | e.cancelBubble = true 99 | } 100 | 101 | this._activeZoomClose() 102 | } 103 | 104 | ZoomService.prototype._touchStart = function (e) { 105 | this._initialTouchPosition = e.touches[0].pageY 106 | $(e.target).on('touchmove.zoom', $.proxy(this._touchMove, this)) 107 | } 108 | 109 | ZoomService.prototype._touchMove = function (e) { 110 | if (Math.abs(e.touches[0].pageY - this._initialTouchPosition) > 10) { 111 | this._activeZoomClose() 112 | $(e.target).off('touchmove.zoom') 113 | } 114 | } 115 | 116 | 117 | /** 118 | * The zoom object 119 | */ 120 | function Zoom (img) { 121 | this._fullHeight = 122 | this._fullWidth = 123 | this._overlay = 124 | this._targetImageWrap = null 125 | 126 | this._targetImage = img 127 | 128 | this._$body = $(document.body) 129 | } 130 | 131 | Zoom.OFFSET = 80 132 | Zoom._MAX_WIDTH = 2560 133 | Zoom._MAX_HEIGHT = 4096 134 | 135 | Zoom.prototype.zoomImage = function () { 136 | var img = document.createElement('img') 137 | img.onload = $.proxy(function () { 138 | this._fullHeight = Number(img.height) 139 | this._fullWidth = Number(img.width) 140 | this._zoomOriginal() 141 | }, this) 142 | img.src = this._targetImage.src 143 | } 144 | 145 | Zoom.prototype._zoomOriginal = function () { 146 | this._targetImageWrap = document.createElement('div') 147 | this._targetImageWrap.className = 'zoom-img-wrap' 148 | 149 | this._targetImage.parentNode.insertBefore(this._targetImageWrap, this._targetImage) 150 | this._targetImageWrap.appendChild(this._targetImage) 151 | 152 | $(this._targetImage) 153 | .addClass('zoom-img') 154 | .attr('data-action', 'zoom-out') 155 | 156 | this._overlay = document.createElement('div') 157 | this._overlay.className = 'zoom-overlay' 158 | 159 | document.body.appendChild(this._overlay) 160 | 161 | this._calculateZoom() 162 | this._triggerAnimation() 163 | } 164 | 165 | Zoom.prototype._calculateZoom = function () { 166 | this._targetImage.offsetWidth // repaint before animating 167 | 168 | var originalFullImageWidth = this._fullWidth 169 | var originalFullImageHeight = this._fullHeight 170 | 171 | var scrollTop = $(window).scrollTop() 172 | 173 | var maxScaleFactor = originalFullImageWidth / this._targetImage.width 174 | 175 | var viewportHeight = ($(window).height() - Zoom.OFFSET) 176 | var viewportWidth = ($(window).width() - Zoom.OFFSET) 177 | 178 | var imageAspectRatio = originalFullImageWidth / originalFullImageHeight 179 | var viewportAspectRatio = viewportWidth / viewportHeight 180 | 181 | if (originalFullImageWidth < viewportWidth && originalFullImageHeight < viewportHeight) { 182 | this._imgScaleFactor = maxScaleFactor 183 | 184 | } else if (imageAspectRatio < viewportAspectRatio) { 185 | this._imgScaleFactor = (viewportHeight / originalFullImageHeight) * maxScaleFactor 186 | 187 | } else { 188 | this._imgScaleFactor = (viewportWidth / originalFullImageWidth) * maxScaleFactor 189 | } 190 | } 191 | 192 | Zoom.prototype._triggerAnimation = function () { 193 | this._targetImage.offsetWidth // repaint before animating 194 | 195 | var imageOffset = $(this._targetImage).offset() 196 | var scrollTop = $(window).scrollTop() 197 | 198 | var viewportY = scrollTop + ($(window).height() / 2) 199 | var viewportX = ($(window).width() / 2) 200 | 201 | var imageCenterY = imageOffset.top + (this._targetImage.height / 2) 202 | var imageCenterX = imageOffset.left + (this._targetImage.width / 2) 203 | 204 | this._translateY = viewportY - imageCenterY 205 | this._translateX = viewportX - imageCenterX 206 | 207 | var targetTransform = 'scale(' + this._imgScaleFactor + ')' 208 | var imageWrapTransform = 'translate(' + this._translateX + 'px, ' + this._translateY + 'px)' 209 | 210 | if ($.support.transition) { 211 | imageWrapTransform += ' translateZ(0)' 212 | } 213 | 214 | $(this._targetImage) 215 | .css({ 216 | '-webkit-transform': targetTransform, 217 | '-ms-transform': targetTransform, 218 | 'transform': targetTransform 219 | }) 220 | 221 | $(this._targetImageWrap) 222 | .css({ 223 | '-webkit-transform': imageWrapTransform, 224 | '-ms-transform': imageWrapTransform, 225 | 'transform': imageWrapTransform 226 | }) 227 | 228 | this._$body.addClass('zoom-overlay-open') 229 | } 230 | 231 | Zoom.prototype.close = function () { 232 | this._$body 233 | .removeClass('zoom-overlay-open') 234 | .addClass('zoom-overlay-transitioning') 235 | 236 | // we use setStyle here so that the correct vender prefix for transform is used 237 | $(this._targetImage) 238 | .css({ 239 | '-webkit-transform': '', 240 | '-ms-transform': '', 241 | 'transform': '' 242 | }) 243 | 244 | $(this._targetImageWrap) 245 | .css({ 246 | '-webkit-transform': '', 247 | '-ms-transform': '', 248 | 'transform': '' 249 | }) 250 | 251 | if (!$.support.transition) { 252 | return this.dispose() 253 | } 254 | 255 | $(this._targetImage) 256 | .one($.support.transition.end, $.proxy(this.dispose, this)) 257 | .emulateTransitionEnd(300) 258 | } 259 | 260 | Zoom.prototype.dispose = function () { 261 | if (this._targetImageWrap && this._targetImageWrap.parentNode) { 262 | $(this._targetImage) 263 | .removeClass('zoom-img') 264 | .attr('data-action', 'zoom') 265 | 266 | this._targetImageWrap.parentNode.replaceChild(this._targetImage, this._targetImageWrap) 267 | this._overlay.parentNode.removeChild(this._overlay) 268 | 269 | this._$body.removeClass('zoom-overlay-transitioning') 270 | } 271 | } 272 | 273 | // wait for dom ready (incase script included before body) 274 | $(function () { 275 | new ZoomService().listen() 276 | }) 277 | 278 | }(jQuery) 279 | -------------------------------------------------------------------------------- /static/webfonts/fa-brands-400.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-brands-400.ttf -------------------------------------------------------------------------------- /static/webfonts/fa-brands-400.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-brands-400.woff2 -------------------------------------------------------------------------------- /static/webfonts/fa-regular-400.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-regular-400.ttf -------------------------------------------------------------------------------- /static/webfonts/fa-regular-400.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-regular-400.woff2 -------------------------------------------------------------------------------- /static/webfonts/fa-solid-900.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-solid-900.ttf -------------------------------------------------------------------------------- /static/webfonts/fa-solid-900.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-solid-900.woff2 -------------------------------------------------------------------------------- /static/webfonts/fa-v4compatibility.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-v4compatibility.ttf -------------------------------------------------------------------------------- /static/webfonts/fa-v4compatibility.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Maks-Ars1/Django-simple-page/b907442acc963b7145b1c981938e2865fb814caa/static/webfonts/fa-v4compatibility.woff2 --------------------------------------------------------------------------------