├── .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 |
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 |
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
--------------------------------------------------------------------------------