Compare commits
	
		
			55 Commits
		
	
	
		
			c00b2b0e52
			...
			nginx-unit
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 6e99404134 | ||
|   | 7b033454b3 | ||
|   | f34200901e | ||
|   | 330a36556c | ||
|   | d7a385fd45 | ||
|   | edc9366a5b | ||
|   | 88fcb17dc5 | ||
|   | 49ca4bccdc | ||
|   | 3985a5635d | ||
|   | 9843299ef6 | ||
|   | 75099ca05e | ||
|   | 1b8d81bd3c | ||
|   | 75e78d606f | ||
|   | f9478f2894 | ||
|   | b235f67be3 | ||
|   | 43e5d71cec | ||
|   | e11ae55ed9 | ||
|   | 5182b2cdb6 | ||
|   | 6a2f89d36e | ||
|   | ed03ea4a1c | ||
|   | 0806b55cbe | ||
|   | 10648a6d0a | ||
|   | e62e54757a | ||
|   | af1bfc06a5 | ||
|   | d56911901b | ||
|   | 017e473b4d | ||
|   | b65ef99935 | ||
|   | aa3151dde6 | ||
|   | 53273fa3c2 | ||
|   | 2887fd852a | ||
|   | acb2ab9e52 | ||
|   | eca48c6cfa | ||
| 4e18118605 | |||
|   | 7a46de1679 | ||
|   | 398088454b | ||
|   | 9f0d46a17a | ||
| 071b82121c | |||
|   | ac1d7fb5e8 | ||
|   | 915d8369bf | ||
|   | 1efa9b2166 | ||
|   | 777e949c9f | ||
|   | 32a2b3e6c1 | ||
|   | 14f14a2f33 | ||
|   | d57f677bd3 | ||
|   | dc1571bcfb | ||
|   | 5a83ccfc07 | ||
|   | f13132d0c7 | ||
|   | cfe706b1f5 | ||
|   | 5d522da15e | ||
| ee7a0bfc32 | |||
| 17be21ec00 | |||
|   | 76dd31c80c | ||
| fce13c85e4 | |||
| 1c616d27d9 | |||
|   | 58d4f27c61 | 
							
								
								
									
										150
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,150 @@ | |||||||
|  | # project specific | ||||||
|  | settings.json | ||||||
|  | db.sqlite3 | ||||||
|  | dummyusers.json | ||||||
|  | *.prof | ||||||
|  | statictest.txt | ||||||
|  | ./test.sh | ||||||
|  |  | ||||||
|  | # IDE | ||||||
|  | .vscode | ||||||
|  |  | ||||||
|  | # Byte-compiled / optimized / DLL files | ||||||
|  | __pycache__/ | ||||||
|  | *.py[cod] | ||||||
|  | *$py.class | ||||||
|  | *.pyc | ||||||
|  |  | ||||||
|  | # C extensions | ||||||
|  | *.so | ||||||
|  |  | ||||||
|  | # Distribution / packaging | ||||||
|  | .Python | ||||||
|  | build/ | ||||||
|  | develop-eggs/ | ||||||
|  | dist/ | ||||||
|  | downloads/ | ||||||
|  | eggs/ | ||||||
|  | .eggs/ | ||||||
|  | lib/ | ||||||
|  | lib64/ | ||||||
|  | parts/ | ||||||
|  | sdist/ | ||||||
|  | var/ | ||||||
|  | wheels/ | ||||||
|  | share/python-wheels/ | ||||||
|  | *.egg-info/ | ||||||
|  | .installed.cfg | ||||||
|  | *.egg | ||||||
|  | MANIFEST | ||||||
|  |  | ||||||
|  | # PyInstaller | ||||||
|  | #  Usually these files are written by a python script from a template | ||||||
|  | #  before PyInstaller builds the exe, so as to inject date/other infos into it. | ||||||
|  | *.manifest | ||||||
|  | *.spec | ||||||
|  |  | ||||||
|  | # Installer logs | ||||||
|  | pip-log.txt | ||||||
|  | pip-delete-this-directory.txt | ||||||
|  |  | ||||||
|  | # Unit test / coverage reports | ||||||
|  | htmlcov/ | ||||||
|  | .tox/ | ||||||
|  | .nox/ | ||||||
|  | .coverage | ||||||
|  | .coverage.* | ||||||
|  | .cache | ||||||
|  | nosetests.xml | ||||||
|  | coverage.xml | ||||||
|  | *.cover | ||||||
|  | *.py,cover | ||||||
|  | .hypothesis/ | ||||||
|  | .pytest_cache/ | ||||||
|  | cover/ | ||||||
|  |  | ||||||
|  | # Translations | ||||||
|  | *.mo | ||||||
|  | *.pot | ||||||
|  |  | ||||||
|  | # Django stuff: | ||||||
|  | *.log | ||||||
|  | local_settings.py | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Flask stuff: | ||||||
|  | instance/ | ||||||
|  | .webassets-cache | ||||||
|  |  | ||||||
|  | # Scrapy stuff: | ||||||
|  | .scrapy | ||||||
|  |  | ||||||
|  | # Sphinx documentation | ||||||
|  | docs/_build/ | ||||||
|  |  | ||||||
|  | # PyBuilder | ||||||
|  | .pybuilder/ | ||||||
|  | target/ | ||||||
|  |  | ||||||
|  | # Jupyter Notebook | ||||||
|  | .ipynb_checkpoints | ||||||
|  |  | ||||||
|  | # IPython | ||||||
|  | profile_default/ | ||||||
|  | ipython_config.py | ||||||
|  |  | ||||||
|  | # pyenv | ||||||
|  | #   For a library or package, you might want to ignore these files since the code is | ||||||
|  | #   intended to run in multiple environments; otherwise, check them in: | ||||||
|  | # .python-version | ||||||
|  |  | ||||||
|  | # pipenv | ||||||
|  | #   According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. | ||||||
|  | #   However, in case of collaboration, if having platform-specific dependencies or dependencies | ||||||
|  | #   having no cross-platform support, pipenv may install dependencies that don't work, or not | ||||||
|  | #   install all needed dependencies. | ||||||
|  | #Pipfile.lock | ||||||
|  |  | ||||||
|  | # PEP 582; used by e.g. github.com/David-OConnor/pyflow | ||||||
|  | __pypackages__/ | ||||||
|  |  | ||||||
|  | # Celery stuff | ||||||
|  | celerybeat-schedule | ||||||
|  | celerybeat.pid | ||||||
|  |  | ||||||
|  | # SageMath parsed files | ||||||
|  | *.sage.py | ||||||
|  |  | ||||||
|  | # Environments | ||||||
|  | .env | ||||||
|  | .venv | ||||||
|  | env/ | ||||||
|  | venv/ | ||||||
|  | ENV/ | ||||||
|  | env.bak/ | ||||||
|  | venv.bak/ | ||||||
|  | watering-env/ | ||||||
|  |  | ||||||
|  | # Spyder project settings | ||||||
|  | .spyderproject | ||||||
|  | .spyproject | ||||||
|  |  | ||||||
|  | # Rope project settings | ||||||
|  | .ropeproject | ||||||
|  |  | ||||||
|  | # mkdocs documentation | ||||||
|  | /site | ||||||
|  |  | ||||||
|  | # mypy | ||||||
|  | .mypy_cache/ | ||||||
|  | .dmypy.json | ||||||
|  | dmypy.json | ||||||
|  |  | ||||||
|  | # Pyre type checker | ||||||
|  | .pyre/ | ||||||
|  |  | ||||||
|  | # pytype static type analyzer | ||||||
|  | .pytype/ | ||||||
|  |  | ||||||
|  | # Cython debug symbols | ||||||
|  | cython_debug/ | ||||||
| @@ -11,43 +11,103 @@ https://docs.djangoproject.com/en/3.0/ref/settings/ | |||||||
| """ | """ | ||||||
|  |  | ||||||
| import os | import os | ||||||
|  | import json | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  | BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) | ||||||
|  |  | ||||||
|  | with open(BASE_DIR + "/settings.json", "r") as userfile: | ||||||
|  |     usersettings = json.load(userfile) | ||||||
|  |     CALIBRE_DIR = os.path.abspath(usersettings["CALIBRE_DIR"]) | ||||||
|  |     SECRET_KEY = usersettings["SECRET_KEY"] | ||||||
|  |     ALLOWED_HOSTS = usersettings["ALLOWED_HOSTS"] | ||||||
|  |     DEBUG = usersettings["DEBUG"] | ||||||
|  |  | ||||||
|  | DOCKER = os.environ.get('AM_DOCKER_INSTANCE', False) | ||||||
|  |  | ||||||
|  |  | ||||||
| # Build paths inside the project like this: os.path.join(BASE_DIR, ...) | # Build paths inside the project like this: os.path.join(BASE_DIR, ...) | ||||||
| BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) |  | ||||||
| CALIBRE_DIR = os.path.abspath( |  | ||||||
|     "C:\\Users\\MassiveAtoms\\Documents\\Calibre Library") |  | ||||||
| EMAIL_BACKEND = 'django.core.mail.backends.console.EmailBackend' | EMAIL_BACKEND = 'django.core.mail.backends.console.EmailBackend' | ||||||
|  |  | ||||||
|  | # optimisation stuff ###############################################3 | ||||||
|  | #                                                                    # | ||||||
|  | CONN_MAX_AGE = 60 * 5 | ||||||
|  |  | ||||||
|  | CACHES = { | ||||||
|  |     'default': { | ||||||
|  |         'BACKEND': 'django.core.cache.backends.locmem.LocMemCache', | ||||||
|  |         'LOCATION': 'unique-snowflake', | ||||||
|  |         "TIMEOUT": 60 * 5, | ||||||
|  |  | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ##                                                                    ## | ||||||
|  | ######################################################################## | ||||||
|  | ##                    STATIC FILES                                    ## | ||||||
|  |  | ||||||
| # Static files (CSS, JavaScript, Images) | # Static files (CSS, JavaScript, Images) | ||||||
| # https://docs.djangoproject.com/en/3.0/howto/static-files/ | # https://docs.djangoproject.com/en/3.0/howto/static-files/ | ||||||
|  |  | ||||||
| STATICFILES_DIRS = [ | STATICFILES_DIRS = [ | ||||||
|     os.path.abspath(CALIBRE_DIR), |     # os.path.abspath(CALIBRE_DIR), | ||||||
|     # '/static/', |     # '/static/', | ||||||
| ] | ] | ||||||
|  |  | ||||||
| STATIC_URL = '/static/' | STATIC_URL = '/static/' | ||||||
|  |  | ||||||
|  | STATIC_ROOT = BASE_DIR + "/static/" | ||||||
|  | ##                                                                     ## | ||||||
|  | ######################################################################### | ||||||
|  | # LOGGING | ||||||
|  |  | ||||||
| # Quick-start development settings - unsuitable for production | # TODO: think about the issue for a bit. no write access to file on first run | ||||||
| # See https://docs.djangoproject.com/en/3.0/howto/deployment/checklist/ | # but startscript doesn't run to give access if it can't start django | ||||||
|  | # if not DOCKER:  | ||||||
| # SECURITY WARNING: keep the secret key used in production secret! | #     logfile = "/app//data//django.log" | ||||||
| SECRET_KEY = 'u(8^+rb%rz5hsx4v^^y(ul7g(4n7a8!db@s*9(m5cs*2_ppy8+' | #     if not os.path.isfile(logfile): | ||||||
|  | #         os.mknod(logfile) | ||||||
|  | #     LOGGING = { | ||||||
|  | #         "version": 1, | ||||||
|  | #         "disable_existing_loggers": False, | ||||||
|  | #         "root": {"level": "INFO", "handlers": ["file"]}, | ||||||
|  | #         "handlers": { | ||||||
|  | #             "file": { | ||||||
|  | #                 "level": "INFO", | ||||||
|  | #                 "class": "logging.FileHandler", | ||||||
|  | #                 "filename": logfile, | ||||||
|  | #                 "formatter": "app", | ||||||
|  | #             }, | ||||||
|  | #         }, | ||||||
|  | #         "loggers": { | ||||||
|  | #             "django": { | ||||||
|  | #                 "handlers": ["file"], | ||||||
|  | #                 "level": "INFO", | ||||||
|  | #                 "propagate": True | ||||||
|  | #             }, | ||||||
|  | #         }, | ||||||
|  | #         "formatters": { | ||||||
|  | #             "app": { | ||||||
|  | #                 "format": ( | ||||||
|  | #                     u"%(asctime)s [%(levelname)-8s] " | ||||||
|  | #                     "(%(module)s.%(funcName)s) %(message)s" | ||||||
|  | #                 ), | ||||||
|  | #                 "datefmt": "%Y-%m-%d %H:%M:%S", | ||||||
|  | #             }, | ||||||
|  | #         }, | ||||||
|  | #     } | ||||||
|  |  | ||||||
|  |  | ||||||
| ALLOWED_HOSTS = ['127.0.0.1', ] | ##                                                                    ## | ||||||
| INTERNAL_IPS = [ | ######################################################################## | ||||||
|     # ... | ##                    DERUG                                           ## | ||||||
|     '127.0.0.1', |  | ||||||
|     # ... |  | ||||||
| ] |  | ||||||
| # Don't change things beyond this |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # SECURITY WARNING: don't run with debug turned on in production! | # SECURITY WARNING: don't run with debug turned on in production! | ||||||
| DEBUG = True |  | ||||||
|  |  | ||||||
| DEBUG_TOOLBAR_PANELS = [ | DEBUG_TOOLBAR_PANELS = [ | ||||||
|     'debug_toolbar.panels.timer.TimerPanel', |     'debug_toolbar.panels.timer.TimerPanel', | ||||||
| @@ -62,9 +122,15 @@ DEBUG_TOOLBAR_PANELS = [ | |||||||
| ] | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ##                                                                    ## | ||||||
|  | ######################################################################## | ||||||
|  | ##                    DERUG                                           ## | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # SILKY_PYTHON_PROFILER = True | ||||||
|  | # SILKY_PYTHON_PROFILER_BINARY = True | ||||||
|  | # SILKY_PYTHON_PROFILER_RESULT_PATH = BASE_DIR + "/profiler" | ||||||
|  | # SILKY_META = True | ||||||
|  |  | ||||||
| LOGIN_REDIRECT_URL = '/books' | LOGIN_REDIRECT_URL = '/books' | ||||||
|  |  | ||||||
| @@ -78,11 +144,15 @@ INSTALLED_APPS = [ | |||||||
|     'django.contrib.messages', |     'django.contrib.messages', | ||||||
|     'django.contrib.staticfiles', |     'django.contrib.staticfiles', | ||||||
|     "library", |     "library", | ||||||
|  |     "django_extensions" | ||||||
|  |     # "silk", # DEBUG/profilling purposes | ||||||
|     # 'debug_toolbar', # DEBUG  purposes |     # 'debug_toolbar', # DEBUG  purposes | ||||||
| ] | ] | ||||||
|  |  | ||||||
| MIDDLEWARE = [ | MIDDLEWARE = [ | ||||||
|  |     # 'silk.middleware.SilkyMiddleware', # DEBUG/profiling purposes | ||||||
|     # 'debug_toolbar.middleware.DebugToolbarMiddleware', # DEBUG purposes |     # 'debug_toolbar.middleware.DebugToolbarMiddleware', # DEBUG purposes | ||||||
|  |     'django.middleware.cache.UpdateCacheMiddleware',  # cache | ||||||
|     'django.middleware.security.SecurityMiddleware', |     'django.middleware.security.SecurityMiddleware', | ||||||
|     'django.contrib.sessions.middleware.SessionMiddleware', |     'django.contrib.sessions.middleware.SessionMiddleware', | ||||||
|     'django.middleware.common.CommonMiddleware', |     'django.middleware.common.CommonMiddleware', | ||||||
| @@ -90,7 +160,11 @@ MIDDLEWARE = [ | |||||||
|     'django.contrib.auth.middleware.AuthenticationMiddleware', |     'django.contrib.auth.middleware.AuthenticationMiddleware', | ||||||
|     'django.contrib.messages.middleware.MessageMiddleware', |     'django.contrib.messages.middleware.MessageMiddleware', | ||||||
|     'django.middleware.clickjacking.XFrameOptionsMiddleware', |     'django.middleware.clickjacking.XFrameOptionsMiddleware', | ||||||
|  |     'django.middleware.cache.FetchFromCacheMiddleware',  # cache | ||||||
| ] | ] | ||||||
|  | ##                                                                    ## | ||||||
|  | ######################################################################## | ||||||
|  | DEFAULT_CHARSET = "utf-8" | ||||||
|  |  | ||||||
| ROOT_URLCONF = 'CalibreWebCompanion.urls' | ROOT_URLCONF = 'CalibreWebCompanion.urls' | ||||||
|  |  | ||||||
| @@ -115,23 +189,33 @@ TEMPLATES = [ | |||||||
|  |  | ||||||
| WSGI_APPLICATION = 'CalibreWebCompanion.wsgi.application' | WSGI_APPLICATION = 'CalibreWebCompanion.wsgi.application' | ||||||
|  |  | ||||||
|  | ##                                                                    ## | ||||||
| # Database | ######################################################################## | ||||||
|  | ##                    DATBASE                                         ## | ||||||
| # https://docs.djangoproject.com/en/3.0/ref/settings/#databases | # https://docs.djangoproject.com/en/3.0/ref/settings/#databases | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if DOCKER: | ||||||
|  |     djangodb_path = "/app/CalibreWebCompanion" | ||||||
|  |     calibredb_path = "/app/content/" | ||||||
|  |      | ||||||
|  | else: | ||||||
|  |     djangodb_path = BASE_DIR | ||||||
|  |     calibredb_path = CALIBRE_DIR | ||||||
|  |  | ||||||
| DATABASES = { | DATABASES = { | ||||||
|     'default': { |     'default': { | ||||||
|         'ENGINE': 'django.db.backends.sqlite3', |         'ENGINE': 'django.db.backends.sqlite3', | ||||||
|         'NAME': os.path.join(BASE_DIR, 'db.sqlite3'), |         'NAME': os.path.join(djangodb_path, 'db.sqlite3'), | ||||||
|     }, |     }, | ||||||
|     'calibre': { |     'calibre': { | ||||||
|         'ENGINE': 'django.db.backends.sqlite3', |         'ENGINE': 'django.db.backends.sqlite3', | ||||||
|         'NAME': os.path.join(CALIBRE_DIR, 'metadata.db'), |         'NAME': os.path.join(calibredb_path, 'metadata.db'), | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| DATABASE_ROUTERS = ["db_routers.DjangoRouter", "db_routers.CalibreRouter"] | DATABASE_ROUTERS = ["db_routers.CalibreRouter", "db_routers.DjangoRouter"] | ||||||
|  |  | ||||||
| # Password validation | # Password validation | ||||||
| # https://docs.djangoproject.com/en/3.0/ref/settings/#auth-password-validators | # https://docs.djangoproject.com/en/3.0/ref/settings/#auth-password-validators | ||||||
|   | |||||||
| @@ -22,6 +22,7 @@ from django.conf import settings | |||||||
| from django.urls import include, path | from django.urls import include, path | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| urlpatterns = [ | urlpatterns = [ | ||||||
|     path('admin/', admin.site.urls), |     path('admin/', admin.site.urls), | ||||||
|     path('accounts/', include('django.contrib.auth.urls')), |     path('accounts/', include('django.contrib.auth.urls')), | ||||||
| @@ -31,7 +32,8 @@ urlpatterns = [ | |||||||
|  |  | ||||||
|  |  | ||||||
| # if settings.DEBUG: # DEBUG purposes | # if settings.DEBUG: # DEBUG purposes | ||||||
| #     import debug_toolbar | #     urlpatterns+= [path('silk/', include('silk.urls', namespace='silk'))] | ||||||
| #     urlpatterns = [ |     # import debug_toolbar | ||||||
| #         path('__debug__/', include(debug_toolbar.urls)), |     # urlpatterns = [ | ||||||
| #     ] + urlpatterns |     #     path('__debug__/', include(debug_toolbar.urls)), | ||||||
|  |     # ] + urlpatterns | ||||||
|   | |||||||
| @@ -1,26 +1,51 @@ | |||||||
|  | import logging | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| class DjangoRouter: | class DjangoRouter: | ||||||
|     """ |     """ | ||||||
|     A router to control all database operations on models in the |     A router to control all database operations on models in the | ||||||
|     auth and contenttypes applications. |     auth and contenttypes applications. | ||||||
|     """ |     """ | ||||||
|     route_app_labels = {'auth', 'contenttypes', "sessions", "sites", "admin", "flatpages"} |     def db_for_read(self, model, **hints): | ||||||
|  |         """ | ||||||
|  |         Attempts to read anything else goes to calibre | ||||||
|  |         """ | ||||||
|  |         return 'default' | ||||||
|  |  | ||||||
|  |     def db_for_write(self, model, **hints):  | ||||||
|  |         """ | ||||||
|  |         Attempts to write auth and contenttypes models go to 'calibre'. | ||||||
|  |         """ | ||||||
|  |         return 'default' | ||||||
|  |  | ||||||
|  |     def allow_relation(self, obj1, obj2, **hints): | ||||||
|  |         """ | ||||||
|  |         Allow relations. | ||||||
|  |         """ | ||||||
|  |         return True | ||||||
|  |  | ||||||
|  |     def allow_migrate(self, db, app_label, model_name=None, **hints):  | ||||||
|  |         """ | ||||||
|  |         Yes | ||||||
|  |         """ | ||||||
|  |         return True | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CalibreRouter: | ||||||
|  |     """ | ||||||
|  |     A router to control all database operations on models in the | ||||||
|  |     auth and contenttypes applications. | ||||||
|  |     """ | ||||||
|  |     route_app_labels = {"library"} | ||||||
|  |  | ||||||
|     def db_for_read(self, model, **hints): |     def db_for_read(self, model, **hints): | ||||||
|         """ |         """ | ||||||
|         Attempts to read auth and contenttypes models go to default. |         Attempts to read auth and contenttypes models go to default. | ||||||
|         """ |         """ | ||||||
|         if model._meta.app_label in self.route_app_labels: |         if model._meta.app_label in self.route_app_labels: | ||||||
|             return 'default' |             return 'calibre' | ||||||
|         return None |  | ||||||
|  |  | ||||||
|     def db_for_write(self, model, **hints): |  | ||||||
|         """ |  | ||||||
|         Attempts to write auth and contenttypes models go to django. |  | ||||||
|         """ |  | ||||||
|         if model._meta.app_label in self.route_app_labels: |  | ||||||
|             return 'default' |  | ||||||
|         return None |         return None | ||||||
|  |  | ||||||
|     def allow_relation(self, obj1, obj2, **hints): |     def allow_relation(self, obj1, obj2, **hints): | ||||||
| @@ -35,42 +60,19 @@ class DjangoRouter: | |||||||
|            return True |            return True | ||||||
|         return None |         return None | ||||||
|  |  | ||||||
|     def allow_migrate(self, db, app_label, model_name=None, **hints): |     # def allow_migrate(self, db, app_label, model_name=None, **hints): | ||||||
|         """ |  | ||||||
|         Make sure the auth and contenttypes apps only appear in the |  | ||||||
|         'django' database. |  | ||||||
|         """ |  | ||||||
|         if app_label in self.route_app_labels: |  | ||||||
|             return db == 'default' |  | ||||||
|         return None |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class CalibreRouter: |  | ||||||
|     """ |  | ||||||
|     A router to control all database operations on models in the |  | ||||||
|     auth and contenttypes applications. |  | ||||||
|     """ |  | ||||||
|     def db_for_read(self, model, **hints): |  | ||||||
|         """ |  | ||||||
|         Attempts to read anything else goes to calibre |  | ||||||
|         """ |  | ||||||
|         return 'calibre' |  | ||||||
|  |  | ||||||
|     # def db_for_write(self, model, **hints): # might be prudent not to allow writes |  | ||||||
|     #     """ |     #     """ | ||||||
|     #     Attempts to write auth and contenttypes models go to 'calibre'. |     #     Make sure the auth and contenttypes apps only appear in the | ||||||
|  |     #     'django' database. | ||||||
|     #     """ |     #     """ | ||||||
|     #     return 'calibre' |     #     if app_label in self.route_app_labels: | ||||||
|  |     #         return db == 'default' | ||||||
|  |     #     return None | ||||||
|  |  | ||||||
|     def allow_relation(self, obj1, obj2, **hints): |     # def db_for_write(self, model, **hints): | ||||||
|         """ |  | ||||||
|         Allow relations. |  | ||||||
|         """ |  | ||||||
|         return True |  | ||||||
|  |  | ||||||
|     # def allow_migrate(self, db, app_label, model_name=None, **hints): # might be prudent not to allow migrations |  | ||||||
|     #     """ |     #     """ | ||||||
|     #     Yes |     #     Attempts to write auth and contenttypes models go to django. | ||||||
|     #     """ |     #     """ | ||||||
|     #     return True |     #     if model._meta.app_label in self.route_app_labels: | ||||||
|  |     #         return 'default' | ||||||
|  |     #     return None | ||||||
|   | |||||||
							
								
								
									
										107
									
								
								CalibreWebCompanion/library/calibredb.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,107 @@ | |||||||
|  | from sqlalchemy import create_engine | ||||||
|  | from sqlalchemy.ext.declarative import declarative_base | ||||||
|  | from sqlalchemy.orm import sessionmaker | ||||||
|  | engine = create_engine('sqlite:///C://Users//MassiveAtoms//Documents//Calibre Library//metadata.db.', echo=True) | ||||||
|  | Base = declarative_base(engine) | ||||||
|  | ######################################################################## | ||||||
|  |  | ||||||
|  | class Author(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'authors' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id, text name, sort  | ||||||
|  |  | ||||||
|  | class Comment(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'comments' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id, int book, text text | ||||||
|  |  | ||||||
|  | # class Data(Base): # maybe | ||||||
|  | #     """""" | ||||||
|  | #     __tablename__ = 'data' | ||||||
|  | #     __table_args__ = {'autoload':True} | ||||||
|  | #     # has int id, int book, text format, text name | ||||||
|  |  | ||||||
|  | class Identifier(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'identifiers' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id, int book, text value | ||||||
|  |  | ||||||
|  | class Publisher(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'publishers' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id, text name | ||||||
|  |  | ||||||
|  | class Rating(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'ratings' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  int rating | ||||||
|  |  | ||||||
|  | class Series(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'series' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  text name | ||||||
|  |  | ||||||
|  | class Tag(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'tags' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  text name | ||||||
|  |  | ||||||
|  | class Book(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'books' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  text title, text sort, time timestamp, time pubdate,  | ||||||
|  |     # float series_index, text path  | ||||||
|  |  | ||||||
|  | class Book_author_link(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'books_authors_link' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  id book, id author | ||||||
|  |  | ||||||
|  | class Book_publisher_link(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'books_publishers_link' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  id book, id publisher | ||||||
|  |  | ||||||
|  | class Book_rating_link(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'books_ratings_link' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  id book, id rating | ||||||
|  |  | ||||||
|  | class Book_series_link(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'books_series_link' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  id book, id series | ||||||
|  |  | ||||||
|  | class Book_tags_link(Base): # needed | ||||||
|  |     """""" | ||||||
|  |     __tablename__ = 'books_tags_link' | ||||||
|  |     __table_args__ = {'autoload':True} | ||||||
|  |     # has int id,  id book, id tag | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #---------------------------------------------------------------------- | ||||||
|  | def loadSession(): | ||||||
|  |     """""" | ||||||
|  |     metadata = Base.metadata | ||||||
|  |     Session = sessionmaker(bind=engine) | ||||||
|  |     session = Session() | ||||||
|  |     return session | ||||||
|  |      | ||||||
|  | if __name__ == "__main__": | ||||||
|  |     session = loadSession() | ||||||
|  |     res = session.query(Book).all() | ||||||
|  |     for i in res: | ||||||
|  |         print(i.id, i.title, i.sort, i.timestamp, i.pubdate, i.series_index, i.path) | ||||||
|  |  | ||||||
| @@ -1,13 +1,24 @@ | |||||||
| from .models import Author, Tag, Publisher, Language, Rating, Series | from .models import Author, Tag, Publisher, Language, Rating, Series | ||||||
|  | from django.db.models import Count | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| def filters(request): | def filters(request): | ||||||
|     unique_authors = Author.objects.all().order_by('sort') |     # unique_authors = Author.objects.all().order_by('sort') | ||||||
|     unique_tags = Tag.objects.all().order_by('name') |     # unique_tags = Tag.objects.all().order_by('name') | ||||||
|     unique_publishers = Publisher.objects.all().order_by('name') |     # unique_publishers = Publisher.objects.all().order_by('name') | ||||||
|     unique_languages = Language.objects.all() |     # unique_ratings = Rating.objects.all().order_by('rating') | ||||||
|     unique_ratings = Rating.objects.all().order_by('rating') |     # unique_languages = Language.objects.all() | ||||||
|     unique_series = Series.objects.all().order_by('sort') |     # unique_series = Series.objects.all().order_by('sort') | ||||||
|  |     # unique_authors = Author.objects.annotate(num_books=Count('book')).order_by('sort') | ||||||
|  |     unique_authors = Author.objects.only('name', "id").annotate(num_books=Count('book')).order_by('name') | ||||||
|  |     unique_tags = Tag.objects.annotate(num_books=Count('book')).order_by('name') | ||||||
|  |     unique_publishers = Publisher.objects.annotate(num_books=Count('book')).order_by('name') | ||||||
|  |     unique_languages = Language.objects.annotate(num_books=Count('book')).order_by('lang_code') | ||||||
|  |     unique_ratings = Rating.objects.annotate(num_books=Count('book')) | ||||||
|  |     unique_series = Series.objects.annotate(num_books=Count('book')).order_by('sort') | ||||||
|  |      | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|         "unique_authors": unique_authors, |         "unique_authors": unique_authors, | ||||||
|   | |||||||
| @@ -1,12 +1,15 @@ | |||||||
| from django import forms | from django import forms | ||||||
| from django.contrib.auth.forms import UserCreationForm | from django.contrib.auth.forms import UserCreationForm | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| class SearchForm(forms.Form): | class SearchForm(forms.Form): | ||||||
|     title = forms.CharField(label="Title", max_length=200) |     title = forms.CharField(label="Title", max_length=200) | ||||||
|     author = forms.CharField(label='Author', max_length=100) |     author = forms.CharField(label='Author', max_length=100) | ||||||
|     # identifier = forms.CharField(label='Identifier(ISBN, Google-id, amazon id)', max_length=20) |     identifier = forms.CharField(label='Identifier(ISBN, Google-id, amazon id)', max_length=20) | ||||||
|  |     generic = forms.CharField(label='All', max_length=100, required=False) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,221 @@ | |||||||
|  | # Generated by Django 3.0.7 on 2020-07-11 16:22 | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('library', '0001_initial'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Author', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('name', models.TextField()), | ||||||
|  |                 ('sort', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('link', models.TextField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'authors', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Book', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('title', models.TextField()), | ||||||
|  |                 ('sort', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('timestamp', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('pubdate', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('series_index', models.FloatField()), | ||||||
|  |                 ('author_sort', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('isbn', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('lccn', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('path', models.TextField()), | ||||||
|  |                 ('flags', models.IntegerField()), | ||||||
|  |                 ('uuid', models.TextField(blank=True, null=True)), | ||||||
|  |                 ('has_cover', models.BooleanField(blank=True, null=True)), | ||||||
|  |                 ('last_modified', models.TextField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'books', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='BookAuthorLink', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'books_authors_link', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='BookLanguageLink', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('item_order', models.IntegerField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'books_languages_link', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='BookPublisherLink', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'books_publishers_link', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='BookRatingLink', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'books_ratings_link', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='BookSeriesLink', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'books_series_link', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='BookTagLink', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'books_tags_link', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Comment', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('text', models.TextField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'comments', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Identifier', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('book', models.IntegerField()), | ||||||
|  |                 ('type', models.TextField()), | ||||||
|  |                 ('val', models.TextField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'identifiers', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Language', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('lang_code', models.TextField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'languages', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Publisher', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('name', models.TextField()), | ||||||
|  |                 ('sort', models.TextField(blank=True, null=True)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'publishers', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Rating', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('rating', models.IntegerField(blank=True, null=True)), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'ratings', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.CreateModel( | ||||||
|  |             name='Tag', | ||||||
|  |             fields=[ | ||||||
|  |                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||||
|  |                 ('name', models.TextField()), | ||||||
|  |             ], | ||||||
|  |             options={ | ||||||
|  |                 'db_table': 'tags', | ||||||
|  |                 'managed': False, | ||||||
|  |             }, | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Authors', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Books', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='BooksAuthorsLink', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='BooksLanguagesLink', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='BooksPublishersLink', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='BooksRatingsLink', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='BooksSeriesLink', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='BooksTagsLink', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Comments', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Identifiers', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Languages', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Publishers', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Ratings', | ||||||
|  |         ), | ||||||
|  |         migrations.DeleteModel( | ||||||
|  |             name='Tags', | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
| @@ -7,7 +7,10 @@ | |||||||
| # Feel free to rename the models, but don't rename db_table values or field names. | # Feel free to rename the models, but don't rename db_table values or field names. | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| class Author(models.Model): | class Author(models.Model): | ||||||
|     name = models.TextField() |     name = models.TextField() | ||||||
| @@ -27,12 +30,11 @@ class Author(models.Model): | |||||||
|         db_table = 'authors' |         db_table = 'authors' | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Comment(models.Model): | class Comment(models.Model): | ||||||
|     book = models.ForeignKey("Book", db_column="book", on_delete=models.CASCADE) |     book = models.ForeignKey("Book", db_column="book", | ||||||
|  |                              on_delete=models.CASCADE) | ||||||
|     text = models.TextField() |     text = models.TextField() | ||||||
|  |  | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         managed = False |         managed = False | ||||||
|         db_table = 'comments' |         db_table = 'comments' | ||||||
| @@ -41,7 +43,6 @@ class Comment(models.Model): | |||||||
|         ] |         ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Data(models.Model): | class Data(models.Model): | ||||||
|     book = models.IntegerField() |     book = models.IntegerField() | ||||||
|     format = models.TextField() |     format = models.TextField() | ||||||
| @@ -58,7 +59,7 @@ class Data(models.Model): | |||||||
|  |  | ||||||
|  |  | ||||||
| class Identifier(models.Model): | class Identifier(models.Model): | ||||||
|     book = models.IntegerField() |     book = models.ForeignKey("Book", db_column="book", on_delete=models.CASCADE) | ||||||
|     type = models.TextField() |     type = models.TextField() | ||||||
|     val = models.TextField() |     val = models.TextField() | ||||||
|  |  | ||||||
| @@ -98,7 +99,7 @@ class Publisher(models.Model): | |||||||
|         through='BookPublisherLink', |         through='BookPublisherLink', | ||||||
|         through_fields=('publisher', 'book'), |         through_fields=('publisher', 'book'), | ||||||
|         related_name="released" |         related_name="released" | ||||||
|         ) |     ) | ||||||
|  |  | ||||||
|     def get_absolute_url(self): |     def get_absolute_url(self): | ||||||
|         """Returns the url to access a particular instance of MyModelName.""" |         """Returns the url to access a particular instance of MyModelName.""" | ||||||
| @@ -118,6 +119,7 @@ class Publisher(models.Model): | |||||||
|  |  | ||||||
| class Rating(models.Model): | class Rating(models.Model): | ||||||
|     rating = models.IntegerField(blank=True, null=True) |     rating = models.IntegerField(blank=True, null=True) | ||||||
|  |  | ||||||
|     def get_absolute_url(self): |     def get_absolute_url(self): | ||||||
|         """Returns the url to access a particular instance of MyModelName.""" |         """Returns the url to access a particular instance of MyModelName.""" | ||||||
|         return reverse('rating-detail-view', args=[str(self.id)]) |         return reverse('rating-detail-view', args=[str(self.id)]) | ||||||
| @@ -153,6 +155,7 @@ class Series(models.Model): | |||||||
|  |  | ||||||
| class Tag(models.Model): | class Tag(models.Model): | ||||||
|     name = models.TextField() |     name = models.TextField() | ||||||
|  |  | ||||||
|     def get_absolute_url(self): |     def get_absolute_url(self): | ||||||
|         """Returns the url to access a particular instance of MyModelName.""" |         """Returns the url to access a particular instance of MyModelName.""" | ||||||
|         return reverse('tag-detail-view', args=[str(self.id)]) |         return reverse('tag-detail-view', args=[str(self.id)]) | ||||||
| @@ -173,9 +176,9 @@ class Book(models.Model): | |||||||
|     title = models.TextField() |     title = models.TextField() | ||||||
|     sort = models.TextField(blank=True, null=True) |     sort = models.TextField(blank=True, null=True) | ||||||
|     # This field type is a guess. |     # This field type is a guess. | ||||||
|     timestamp = models.TextField(blank=True, null=True) |     timestamp = models.DateTimeField(blank=True, null=True) | ||||||
|     # This field type is a guess. |     # This field type is a guess. | ||||||
|     pubdate = models.TextField(blank=True, null=True) |     pubdate = models.DateTimeField(blank=True, null=True) | ||||||
|     series_index = models.FloatField() |     series_index = models.FloatField() | ||||||
|     author_sort = models.TextField(blank=True, null=True) |     author_sort = models.TextField(blank=True, null=True) | ||||||
|     isbn = models.TextField(blank=True, null=True) |     isbn = models.TextField(blank=True, null=True) | ||||||
| @@ -184,7 +187,7 @@ class Book(models.Model): | |||||||
|     flags = models.IntegerField() |     flags = models.IntegerField() | ||||||
|     uuid = models.TextField(blank=True, null=True) |     uuid = models.TextField(blank=True, null=True) | ||||||
|     has_cover = models.BooleanField(blank=True, null=True) |     has_cover = models.BooleanField(blank=True, null=True) | ||||||
|     last_modified = models.TextField()  # This field type is a guess. |     last_modified = models.DateTimeField()  # This field type is a guess. | ||||||
|     authors = models.ManyToManyField( |     authors = models.ManyToManyField( | ||||||
|         Author, |         Author, | ||||||
|         through='BookAuthorLink', |         through='BookAuthorLink', | ||||||
| @@ -193,14 +196,28 @@ class Book(models.Model): | |||||||
|         Language, |         Language, | ||||||
|         through='BookLanguageLink', |         through='BookLanguageLink', | ||||||
|         through_fields=('book', 'lang_code')) |         through_fields=('book', 'lang_code')) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def language(self): | ||||||
|  |         return self.languages.first() | ||||||
|  |  | ||||||
|     publishers = models.ManyToManyField( |     publishers = models.ManyToManyField( | ||||||
|         Publisher, |         Publisher, | ||||||
|         through='BookPublisherLink', |         through='BookPublisherLink', | ||||||
|         through_fields=('book', 'publisher')) |         through_fields=('book', 'publisher')) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def publisher(self): | ||||||
|  |         return self.publishers.first() | ||||||
|  |  | ||||||
|     series = models.ManyToManyField( |     series = models.ManyToManyField( | ||||||
|         Series, |         Series, | ||||||
|         through='BookSeriesLink', |         through='BookSeriesLink', | ||||||
|         through_fields=('book', 'series')) |         through_fields=('book', 'series')) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def serie(self): | ||||||
|  |         return self.series.first() | ||||||
|     tags = models.ManyToManyField( |     tags = models.ManyToManyField( | ||||||
|         Tag, |         Tag, | ||||||
|         through='BookTagLink', |         through='BookTagLink', | ||||||
| @@ -209,6 +226,11 @@ class Book(models.Model): | |||||||
|         Rating, |         Rating, | ||||||
|         through='BookRatingLink', |         through='BookRatingLink', | ||||||
|         through_fields=('book', 'rating')) |         through_fields=('book', 'rating')) | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def rating(self): | ||||||
|  |         return self.ratings.first() | ||||||
|  |  | ||||||
|     def get_absolute_url(self): |     def get_absolute_url(self): | ||||||
|         """Returns the url to access a particular instance of MyModelName.""" |         """Returns the url to access a particular instance of MyModelName.""" | ||||||
|         return reverse('book-detail-view', args=[str(self.id)]) |         return reverse('book-detail-view', args=[str(self.id)]) | ||||||
| @@ -363,25 +385,3 @@ class BookTagLink(models.Model): | |||||||
| #     class Meta: | #     class Meta: | ||||||
| #         managed = False | #         managed = False | ||||||
| #         db_table = 'feeds' | #         db_table = 'feeds' | ||||||
| # |  | ||||||
| # |  | ||||||
| # class LastReadPositions(models.Model): |  | ||||||
| #     book = models.IntegerField() |  | ||||||
| #     format = models.TextField() |  | ||||||
| #     user = models.TextField() |  | ||||||
| #     device = models.TextField() |  | ||||||
| #     cfi = models.TextField() |  | ||||||
| #     epoch = models.FloatField() |  | ||||||
| #     pos_frac = models.FloatField() |  | ||||||
| # |  | ||||||
| #     class Meta: |  | ||||||
| #         managed = False |  | ||||||
| #         db_table = 'last_read_positions' |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # class MetadataDirtied(models.Model): |  | ||||||
| #     book = models.IntegerField() |  | ||||||
|  |  | ||||||
| #     class Meta: |  | ||||||
| #         managed = False |  | ||||||
| #         db_table = 'metadata_dirtied' |  | ||||||
|   | |||||||
| @@ -1,112 +0,0 @@ | |||||||
| /* sidenav */ |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /* Fixed sidenav, full height */ |  | ||||||
| .sidenav { |  | ||||||
|   height: 100%; |  | ||||||
|   width: 200px; |  | ||||||
|   position: fixed; |  | ||||||
|   z-index: 1; |  | ||||||
|   top: 0; |  | ||||||
|   left: 0; |  | ||||||
|   background-color: #111; |  | ||||||
|   overflow-x: hidden; |  | ||||||
|   padding-top: 20px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* Style the sidenav links and the dropdown button */ |  | ||||||
| .sidenav a, |  | ||||||
| .dropdown-btn { |  | ||||||
|   padding: 6px 8px 6px 16px; |  | ||||||
|   text-decoration: none; |  | ||||||
|   font-size: 20px; |  | ||||||
|   color: #818181; |  | ||||||
|   display: block; |  | ||||||
|   border: none; |  | ||||||
|   background: none; |  | ||||||
|   width: 100%; |  | ||||||
|   text-align: left; |  | ||||||
|   cursor: pointer; |  | ||||||
|   outline: none; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* On mouse-over */ |  | ||||||
| .sidenav a:hover, |  | ||||||
| .dropdown-btn:hover { |  | ||||||
|   color: #f1f1f1; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* Main content */ |  | ||||||
| .main { |  | ||||||
|   margin-left: 200px; |  | ||||||
|   /* Same as the width of the sidenav */ |  | ||||||
|   font-size: 20px; |  | ||||||
|   /* Increased text to enable scrolling */ |  | ||||||
|   padding: 0px 10px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* Add an active class to the active dropdown button */ |  | ||||||
| .active { |  | ||||||
|   background-color: green; |  | ||||||
|   color: white; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* Dropdown container (hidden by default). Optional: add a lighter background color and some left padding to change the design of the dropdown content */ |  | ||||||
| .dropdown-container { |  | ||||||
|   display: none; |  | ||||||
|   background-color: #262626; |  | ||||||
|   padding-left: 8px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /* Optional: Style the caret down icon */ |  | ||||||
| .fa-caret-down { |  | ||||||
|   float: right; |  | ||||||
|   padding-right: 8px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ul.topnav { |  | ||||||
|   list-style-type: none; |  | ||||||
|   margin: 0; |  | ||||||
|   padding: 0; |  | ||||||
|   overflow: hidden; |  | ||||||
|   background-color: #333; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .topnav li { |  | ||||||
|   float: right; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .topnav li a { |  | ||||||
|   display: block; |  | ||||||
|   color: white; |  | ||||||
|   text-align: center; |  | ||||||
|   padding: 14px 16px; |  | ||||||
|   text-decoration: none; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .topnav li a:hover { |  | ||||||
|   background-color: #111; |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| /* Table */ |  | ||||||
| table { |  | ||||||
|   border-spacing: 0; |  | ||||||
|   width: 100%; |  | ||||||
|   border: 1px solid #ddd; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| th { |  | ||||||
|   cursor: pointer; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| th, |  | ||||||
| td { |  | ||||||
|   text-align: left; |  | ||||||
|   padding: 16px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| tr:nth-child(even) { |  | ||||||
|   background-color: #f2f2f2 |  | ||||||
| } |  | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| var dropdown = document.getElementsByClassName("dropdown-btn"); |  | ||||||
| var i; |  | ||||||
|  |  | ||||||
| for (i = 0; i < dropdown.length; i++) { |  | ||||||
|   dropdown[i].addEventListener("click", function() { |  | ||||||
|     this.classList.toggle("active"); |  | ||||||
|     var dropdownContent = this.nextElementSibling; |  | ||||||
|     if (dropdownContent.style.display === "block") { |  | ||||||
|       dropdownContent.style.display = "none"; |  | ||||||
|     } else { |  | ||||||
|       dropdownContent.style.display = "block"; |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| }  |  | ||||||
| @@ -5,97 +5,171 @@ | |||||||
|   {% block title %}<title>Local Library</title>{% endblock %} |   {% block title %}<title>Local Library</title>{% endblock %} | ||||||
|   <meta charset="utf-8"> |   <meta charset="utf-8"> | ||||||
|   <meta name="viewport" content="width=device-width, initial-scale=1"> |   <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||||
|   <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.1.3/css/bootstrap.min.css" |   <script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script> | ||||||
|     integrity="sha384-MCw98/SFnGE8fJT3GXwEOngsV7Zt27NXFoaoApmYm81iuXoPkFOJwJ8ERdknLPMO" crossorigin="anonymous"> |   <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet"> | ||||||
|   <!-- Add additional CSS in static file --> |   <!-- Compiled and minified CSS --> | ||||||
|   {% load static %} |   <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/materialize/1.0.0-beta/css/materialize.min.css"> | ||||||
|   <link rel="stylesheet" href="{% static 'css/styles.css' %}"> |   <!-- Compiled and minified JavaScript --> | ||||||
|  |   <script src="https://cdnjs.cloudflare.com/ajax/libs/materialize/1.0.0-beta/js/materialize.min.js"></script> | ||||||
|  |   <style> | ||||||
|  |     p.count { | ||||||
|  |       color: #FFFFFF; | ||||||
|  |       background-color: #515151; | ||||||
|  |       border: 1px #303030; | ||||||
|  |       border-radius: 0.5rem; | ||||||
|  |       padding: .2rem .25rem; | ||||||
|  |       margin: 0.1rem 0.1rem .1rem; | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     table { | ||||||
|  |       width: 100%; | ||||||
|  |       table-layout: fixed; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     th { | ||||||
|  |       text-align: center; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .title { | ||||||
|  |       width: 30%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .author { | ||||||
|  |       width: 20%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .rating { | ||||||
|  |       width: 5%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .tags { | ||||||
|  |       width: 25%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .added { | ||||||
|  |       width: 10%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .published { | ||||||
|  |       width: 10%; | ||||||
|  |     } | ||||||
|  |   </style> | ||||||
| </head> | </head> | ||||||
|  |  | ||||||
| <body> | <body> | ||||||
|   {% block topnav%} |  | ||||||
|   <ul class="topnav"> |  | ||||||
|  |  | ||||||
|     {% if user.is_authenticated %} |  | ||||||
|     <li class="active"><a href="{{user.get_absolute_url}}"> {{ user.get_username }}</a></li> |  | ||||||
|     <li><a href="{% url 'logout'%}?next={{request.path}}">Logout</a></li> |  | ||||||
|     {% else %} |  | ||||||
|     <li><a href="{% url 'sign-up'%}?next={{request.path}}">Sign up</a></li> |  | ||||||
|     <li><a href="{% url 'login'%}?next={{request.path}}">Login</a></li> |  | ||||||
|     {% endif %} |  | ||||||
|   </ul> |  | ||||||
|   {% endblock %} |  | ||||||
|  |  | ||||||
|   {% if user.is_authenticated %} |  | ||||||
|   <div class="container-fluid"> |  | ||||||
|     <div class="row"> |  | ||||||
|       <div class="col-sm-2"> |  | ||||||
|         {% block sidebar %} |  | ||||||
|         <div class="sidenav"> |  | ||||||
|           <a href="{% url 'search' %}">Search</a> |  | ||||||
|           <a href="{% url 'books' %}">Books</a> |  | ||||||
|           <button class="dropdown-btn">Authors |  | ||||||
|             <i class="fa fa-caret-down"></i> |  | ||||||
|           </button> |  | ||||||
|           <div class="dropdown-container"> |  | ||||||
|             {% for author in unique_authors %} |  | ||||||
|             <a href="{{author.get_absolute_url}}">{{author}}</a> |  | ||||||
|  |  | ||||||
|             {% endfor %} |   <div class="navbar-fixed"> | ||||||
|           </div> |     <nav> | ||||||
|  |       <div class="nav-wrapper row green darken-1"> | ||||||
|  |  | ||||||
|           <button class="dropdown-btn">Publishers |         <ul class="right"> | ||||||
|             <i class="fa fa-caret-down"></i> |           {% if user.is_authenticated %} | ||||||
|           </button> |           <li class="active"><a href="{{user.get_absolute_url}}"> {{ user.get_username }}</a></li> | ||||||
|           <div class="dropdown-container"> |           <li><a href="{% url 'logout'%}?next={{request.path}}">Logout</a></li> | ||||||
|             {% for publisher in unique_publishers %} |         </ul> | ||||||
|             <a href="{{publisher.get_absolute_url}}">{{publisher}}</a> |         {% load cache %} | ||||||
|  |         {% cache 500 sidebar request.user.username %} | ||||||
|  |         <!--Maybe i'm retarded but this is not caching versions per user--> | ||||||
|  |         <ul class="left"> | ||||||
|  |           <li><a href="{% url 'books' %}">Books</a></li> | ||||||
|  |           <li><a class="dropdown-trigger" href={% url 'authors' %} data-target="dropdown-authors">Authors<i | ||||||
|  |                 class="material-icons right">arrow_drop_down</i></a></li> | ||||||
|  |           <li><a class="dropdown-trigger" href={% url "ratings" %} data-target="dropdown-ratings">Ratings<i | ||||||
|  |                 class="material-icons right">arrow_drop_down</i></a></li> | ||||||
|  |           <li><a class="dropdown-trigger" href={% url "tags" %} data-target="dropdown-tags">Tags<i | ||||||
|  |                 class="material-icons right">arrow_drop_down</i></a></li> | ||||||
|  |  | ||||||
|             {% endfor %} |           <li><a class="dropdown-trigger" href={% url "series" %} data-target="dropdown-series">Series<i | ||||||
|           </div> |                 class="material-icons right">arrow_drop_down</i></a></li> | ||||||
|  |  | ||||||
|           <button class="dropdown-btn">Ratings |           <li><a class="dropdown-trigger" href={% url "publishers" %} data-target="dropdown-pubishers">Publishers<i | ||||||
|             <i class="fa fa-caret-down"></i> |                 class="material-icons right">arrow_drop_down</i></a></li> | ||||||
|           </button> |  | ||||||
|           <div class="dropdown-container"> |  | ||||||
|             {% for rating in unique_ratings %} |  | ||||||
|             <a href="{{rating.get_absolute_url}}">{{rating}}</a> |  | ||||||
|  |  | ||||||
|             {% endfor %} |           <li><a href="{% url 'search' %}">Advanced search</a></li> | ||||||
|           </div> |           <li> | ||||||
|           <button class="dropdown-btn">Tags |             <!-- stefan, this div. can we have this int the navbar? --> | ||||||
|             <i class="fa fa-caret-down"></i> |             <div class="container"> | ||||||
|           </button> |               <form action="{% url 'results' %}" method="get" style="padding-top:2em"> | ||||||
|           <div class="dropdown-container"> |                 <label for="generic"></label> | ||||||
|             {% for tag in unique_tags %} |                 <input id="generic" type="text" name="generic" value=""> | ||||||
|             <a href="{{tag.get_absolute_url}}">{{tag}}</a> |                 <button class="waves-effect waves-light btn green accent-4" type="submit">search</button> | ||||||
|  |               </form> | ||||||
|  |             </div> | ||||||
|  |             <!-- this is the end of the div, stefan --> | ||||||
|  |           </li> | ||||||
|  |         </ul> | ||||||
|  |         <ul id="dropdown-authors" class="dropdown-content"> | ||||||
|  |           {% for author in unique_authors %} | ||||||
|  |           <li><a href="{{author.get_absolute_url}}">{{author}}</a></li> | ||||||
|  |           {% endfor %} | ||||||
|  |         </ul> | ||||||
|  |         <ul id="dropdown-ratings" class="dropdown-content"> | ||||||
|  |           {% for rating in unique_ratings %} | ||||||
|  |           <li><a href="{{rating.get_absolute_url}}">{{rating}}</a></li> | ||||||
|  |           {% endfor %} | ||||||
|  |         </ul> | ||||||
|  |         <ul id="dropdown-tags" class="dropdown-content"> | ||||||
|  |           {% for tag in unique_tags %} | ||||||
|  |           <li><a href="{{tag.get_absolute_url}}">{{tag}} ({{tag.num_books}})</a></li> | ||||||
|  |           {% endfor %} | ||||||
|  |         </ul> | ||||||
|  |  | ||||||
|             {% endfor %} |         <ul id="dropdown-series" class="dropdown-content"> | ||||||
|           </div> |           {% for tag in unique_series %} | ||||||
|  |           <li><a href="{{tag.get_absolute_url}}">{{tag}} ({{tag.num_books}})</a></li> | ||||||
|         </div> |           {% endfor %} | ||||||
|         {% endblock %} |         </ul> | ||||||
|  |         <ul id="dropdown-pubishers" class="dropdown-content"> | ||||||
|  |           {% for pub in unique_publishers %} | ||||||
|  |           <!-- stefan here's my shit count  --> | ||||||
|  |           <li><a href="{{pub.get_absolute_url}}">{{pub}} <p class="count">{{pub.num_books}}</p> </a> </li> | ||||||
|  |           {% endfor %} | ||||||
|  |         </ul> | ||||||
|  |         {% endcache %} | ||||||
|  |         {% else %} | ||||||
|  |         <li><a href="{% url 'sign-up'%}?next={{request.path}}">Sign up</a></li> | ||||||
|  |         <li><a href="{% url 'login'%}?next={{request.path}}">Login</a></li> | ||||||
|  |         </ul> | ||||||
|  |         {% endif %} | ||||||
|       </div> |       </div> | ||||||
|       <div class="col-sm-10 ">{% block content %}{% endblock %}</div> |  | ||||||
|     </div> |     </nav> | ||||||
|   </div> |   </div> | ||||||
|   <script> |   <script> | ||||||
|     var dropdown = document.getElementsByClassName("dropdown-btn"); |     $(".dropdown-trigger").dropdown({ | ||||||
|     var i; |       hover: true, | ||||||
|  |       constrainWidth: false, | ||||||
|     for (i = 0; i < dropdown.length; i++) { |       coverTrigger: false | ||||||
|       dropdown[i].addEventListener("click", function () { |     }); | ||||||
|         this.classList.toggle("active"); |  | ||||||
|         var dropdownContent = this.nextElementSibling; |  | ||||||
|         if (dropdownContent.style.display === "block") { |  | ||||||
|           dropdownContent.style.display = "none"; |  | ||||||
|         } else { |  | ||||||
|           dropdownContent.style.display = "block"; |  | ||||||
|         } |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|   </script> |   </script> | ||||||
|  |  | ||||||
|  |   {% if user.is_authenticated %} | ||||||
|  |   {% block content %} {% endblock %} | ||||||
|  |   {% else %} | ||||||
|  |   <div class="valign-wrapper" style="width:100%;height:100%;position: absolute;"> | ||||||
|  |     <div class="valign" style="width:100%;"> | ||||||
|  |       <div class="container"> | ||||||
|  |         <div class="row"> | ||||||
|  |           <div class="col s12 m6 offset-m3"> | ||||||
|  |             <div class="card"> | ||||||
|  |               <div class="card-content center"> | ||||||
|  |                 <p>You don't have permission to view this.</p> | ||||||
|  |               </div> | ||||||
|  |               <div class="card-action center"> | ||||||
|  |                 <a class="waves-effect waves-light btn-large green accent-4" | ||||||
|  |                   href="{% url 'login'%}?next={{request.path}}">Login</a> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  |   {% endif %} | ||||||
|  |  | ||||||
|   <script> |   <script> | ||||||
|     function sortTable(n) { |     function sortTable(n) { | ||||||
|       var table, rows, switching, i, x, y, shouldSwitch, dir, switchcount = 0; |       var table, rows, switching, i, x, y, shouldSwitch, dir, switchcount = 0; | ||||||
| @@ -153,17 +227,6 @@ | |||||||
|     } |     } | ||||||
|   </script> |   </script> | ||||||
|  |  | ||||||
|   {% else %} |  | ||||||
|   <h1>You don't have permission to view this.</h1> |  | ||||||
|   <li><a href="{% url 'login'%}?next={{request.path}}">Login</a></li> |  | ||||||
|   or  |  | ||||||
|   <li><a href="{% url 'sign-up'%}?next={{request.path}}">Sign up</a></li> |  | ||||||
|   {% endif %} |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| </body> | </body> | ||||||
|  |  | ||||||
| </html> | </html> | ||||||
| @@ -1,6 +1,22 @@ | |||||||
|    {% if user.is_authenticated %} |    {% if user.is_authenticated %} | ||||||
|     |     | ||||||
|    {% else %} |    {% else %} | ||||||
|    <h1>You don't have permission to view this.</h1> |     <div class="valign-wrapper" style="width:100%;height:100%;position: absolute;"> | ||||||
|      <li><a href="{% url 'login'%}?next={{request.path}}">Login</a></li>    |       <div class="valign" style="width:100%;"> | ||||||
|  |         <div class="container"> | ||||||
|  |           <div class="row"> | ||||||
|  |             <div class="col s12 m6 offset-m3"> | ||||||
|  |               <div class="card"> | ||||||
|  |                 <div class="card-content center"> | ||||||
|  |                   <p>You don't have permission to view this.</p>  | ||||||
|  |                 </div> | ||||||
|  |                 <div class="card-action center"> | ||||||
|  |                   <a class="waves-effect waves-light btn-large green accent-4" href="{% url 'login'%}?next={{request.path}}">Login</a> | ||||||
|  |                 </div> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|    {% endif %}  |    {% endif %}  | ||||||
| @@ -1,23 +1,24 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
| <h1>{{author}}</h1> | <h1>{{author}}</h1> | ||||||
|  |  | ||||||
| <table id="books"> | <table id="books" class="highlight centered"> | ||||||
|     <tr> |     <tr> | ||||||
|         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> |         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|         <th onclick="sortTable(0)">Title</th> |         <th class="title" onclick="sortTable(0)">Title</th> | ||||||
|         <th onclick="sortTable(1)">Author</th> |         <th class="author" onclick="sortTable(1)">Author</th> | ||||||
|         <th onclick="sortTable(2)">Rating</th> |         <th class="rating" onclick="sortTable(2)">Rating</th> | ||||||
|         <th onclick="sortTable(3)">Tags</th> |         <th class="tags" onclick="sortTable(3)">Tags</th> | ||||||
|         <th onclick="sortTable(4)">Added</th> |         <th class="added" onclick="sortTable(4)">Added</th> | ||||||
|     </tr> |     </tr> | ||||||
|     {% for book in books %} |     {% for book in books %} | ||||||
|     <tr> |     <tr> | ||||||
|         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> |         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> | ||||||
|         <td>{{book.author_sort}}</td> |         <td>{{book.author_sort}}</td> | ||||||
|         <td> {% for rating in book.ratings.all %} |         <td>  | ||||||
|  |             {% for rating in book.ratings.all %} | ||||||
|             {{rating}} |             {{rating}} | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </td> |         </td> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
|   <h1>Author List</h1> |   <h1>Author List</h1> | ||||||
|   {% if author_list %} |   {% if author_list %} | ||||||
|   <ul> |   <ul> | ||||||
|   | |||||||
| @@ -1,50 +1,62 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  | {% block title %}<title>{{book.title}}</title>{% endblock %} | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} | {% load static %} | ||||||
| <h1>{{book.title}} by |  | ||||||
|   {% if book.authors %} |  | ||||||
|   {% for author in book.authors.all %} |  | ||||||
|   <a href="{{author.get_absolute_url}}">{{author.name}}</a> |  | ||||||
|   {%endfor%} |  | ||||||
|   {% else %} |  | ||||||
|  |  | ||||||
|   {{book.author_sort}} | <div class="col s12 m7"> | ||||||
|   {%endif%} |   <div class="card z-depth-0 horizontal"> | ||||||
|  |     <div class="card-image"> | ||||||
|  |       <a style="padding-top:15%" href=" /content/{{download}}"><img src=" /content/{{imgpath}}" | ||||||
|  |           alt="download" srcset=""></a> | ||||||
|  |     </div> | ||||||
|  |     <div class="card-stacked"> | ||||||
|  |       <div class="card-content"> | ||||||
|  |         <h1> {{book.title}}</h1> | ||||||
|  |         <h4> by | ||||||
|  |           {% if book.authors %} | ||||||
|  |           {% for author in book.authors.all %} | ||||||
|  |           <a href="{{author.get_absolute_url}}">{{author.name}}</a> | ||||||
|  |           {%endfor%} | ||||||
|  |           {% else %} | ||||||
|  |           {{book.author_sort}} | ||||||
|  |           {%endif%} | ||||||
|  |           <br> | ||||||
|  |           Published by | ||||||
|  |           {% if book.publishers %} | ||||||
|  |           {% for pub in book.publishers.all %} | ||||||
|  |           <a href="{{pub.get_absolute_url}}">{{pub.name}}</a> | ||||||
|  |           {%endfor%} | ||||||
|  |           {% else %} | ||||||
|  |           Unknown | ||||||
|  |           {%endif%} | ||||||
|  |           <br> | ||||||
|  |           Tags: | ||||||
|  |           {% if book.tags %} | ||||||
|  |           {% for tag in book.tags.all %} | ||||||
|  |           <a href="{{tag.get_absolute_url}}">{{tag.name}}</a>, | ||||||
|  |           {%endfor%} | ||||||
|  |           {% else %} | ||||||
|  |           {%endif%} | ||||||
|  |           <br> | ||||||
|  |           Rating: | ||||||
|  |           {% if book.ratings %} | ||||||
|  |           {% for rating in book.ratings.all %} | ||||||
|  |           <a href="{{rating.get_absolute_url}}">{{rating}}</a> | ||||||
|  |           {%endfor%} | ||||||
|  |           {% else %} | ||||||
|  |           {%endif%} | ||||||
|  |           <br> | ||||||
|  |           <a href="{{book.publisher.get_absolute_url}}">{{book.publisher}}</a> | ||||||
|  |         </h4> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|   Published by  | <div class="container"> | ||||||
|   {% if book.publishers %} |   {% autoescape off %} | ||||||
|   {% for pub in book.publishers.all %} |   {{comment}} | ||||||
|   <a href="{{pub.get_absolute_url}}">{{pub.name}}</a> |   {% endautoescape %} | ||||||
|   {%endfor%} | </div> | ||||||
|   {% else %} |  | ||||||
|   Unknown |  | ||||||
|  |  | ||||||
|   {%endif%} |  | ||||||
|  |  | ||||||
|   Tags:    |  | ||||||
|   {% if book.tags %} |  | ||||||
|   {% for tag in book.tags.all %} |  | ||||||
|   <a href="{{tag.get_absolute_url}}">{{tag.name}}</a>, |  | ||||||
|   {%endfor%} |  | ||||||
|   {% else %} |  | ||||||
|   {%endif%} |  | ||||||
|  |  | ||||||
|   Rating:    |  | ||||||
|   {% if book.ratings %} |  | ||||||
|   {% for rating in book.ratings.all %} |  | ||||||
|   <a href="{{rating.get_absolute_url}}">{{rating}}</a> |  | ||||||
|   {%endfor%} |  | ||||||
|   {% else %} |  | ||||||
|   {%endif%} |  | ||||||
|  |  | ||||||
|   <a href="{{book.publisher.get_absolute_url}}">{{book.publisher}}</a> |  | ||||||
| </h1> |  | ||||||
|  |  | ||||||
| <a href="{% static "" %}{{download}}"><img src="{% static "" %}{{imgpath}}" alt="download" srcset=""></a> |  | ||||||
|  |  | ||||||
| {% autoescape off %} |  | ||||||
| {{comment}} |  | ||||||
| {% endautoescape %} |  | ||||||
|  |  | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -1,40 +1,68 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
| <h1>Book List</h1> |  | ||||||
|  |  | ||||||
| <table id="books"> | <style> | ||||||
|   <tr> |   /* stefan, this is my tag style */ | ||||||
|     <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> |   .tags a { | ||||||
|     <th onclick="sortTable(0)">Title</th> |     color: #FFFFFF; | ||||||
|     <th onclick="sortTable(1)">Author</th> |     background-color: #43A047; | ||||||
|     <th onclick="sortTable(2)">Rating</th> |     text-transform: uppercase; | ||||||
|     <th onclick="sortTable(3)">Tags</th> |     font-size: .66rem; | ||||||
|     <th onclick="sortTable(4)">Added</th> |     white-space: nowrap; | ||||||
|   </tr> |     border-radius: 2rem; | ||||||
|   {% for book in book_list %} |     padding: .25rem .85rem .25rem; | ||||||
|   <tr> |     line-height: 2; | ||||||
|     <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> |     margin: 0.1rem 0.1rem .1rem; | ||||||
|     <td>{{book.author_sort}}</td> |   } | ||||||
|     <td> {% for rating in book.ratings.all %} |  | ||||||
|       {{rating}} |   .tags { | ||||||
|  |     width: 25vw; | ||||||
|  |     padding: .5rem 0 1rem; | ||||||
|  |     line-height: 2; | ||||||
|  |     display: flex; | ||||||
|  |     flex-flow: row wrap; | ||||||
|  |  | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  | <h1 class="center">Book List</h1> | ||||||
|  |  | ||||||
|  | <div class="row"> | ||||||
|  |   <div class="col s1 m0"> | ||||||
|  |   </div> | ||||||
|  |   <div class="col s10 m12"> | ||||||
|  |     <table id="books" class="highlight centered"> | ||||||
|  |       <tr> | ||||||
|  |         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|  |         <th class="title" onclick="sortTable(0)">Title</th> | ||||||
|  |         <th class="author" onclick="sortTable(1)">Author</th> | ||||||
|  |         <th class="rating" onclick="sortTable(2)">Rating</th> | ||||||
|  |         <th class="tags" onclick="sortTable(3)">Tags</th> | ||||||
|  |         <th class="added" onclick="sortTable(4)">Added</th> | ||||||
|  |         <th class="published" onclick="sortTable(5)">Published</th> | ||||||
|  |       </tr> | ||||||
|  |       {% for book in book_list %} | ||||||
|  |       <tr> | ||||||
|  |         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> | ||||||
|  |         <td>{{book.author_sort}}</td> | ||||||
|  |         <td> {% for rating in book.ratings.all %} | ||||||
|  |           {{rating}} | ||||||
|  |           {% endfor %} | ||||||
|  |         </td> | ||||||
|  |         <td class="tags"> | ||||||
|  |           <!-- stefan  --> | ||||||
|  |           {% for tag in book.tags.all %} | ||||||
|  |           <a href={{tag.get_absolute_url}} rel="tag">{{tag}}</a> | ||||||
|  |           {% endfor %} | ||||||
|  |         </td> | ||||||
|  |         <td>{{book.timestamp | date:"d/m/Y" }}</td> | ||||||
|  |         <td>{{book.pubdate.year}}</td> | ||||||
|  |       </tr> | ||||||
|       {% endfor %} |       {% endfor %} | ||||||
|     </td> |     </table> | ||||||
|     <td> |     <div class="col s1 m0"> | ||||||
|       {% for tag in book.tags.all %} |     </div> | ||||||
|       {{tag}}, |   </div> | ||||||
|       {% endfor %} |  | ||||||
|     </td> |  | ||||||
|     <td>{{book.timestamp}}</td> |  | ||||||
|   </tr> |  | ||||||
|   {% endfor %} |  | ||||||
| </table> |  | ||||||
|  |  | ||||||
|  |   {% endblock %} | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| {% endblock %} |  | ||||||
| @@ -1,24 +1,24 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
| <h1>{{publisher}} </h1> | <h1>{{publisher}} </h1> | ||||||
|  |  | ||||||
| {% if publisher.released %} | {% if publisher.released %} | ||||||
| <table id="books"> | <table id="books" class="highlight centered"> | ||||||
|     <tr> |     <tr> | ||||||
|         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> |         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|         <th onclick="sortTable(0)">Title</th> |         <th class="title" onclick="sortTable(0)">Title</th> | ||||||
|         <th onclick="sortTable(1)">Author</th> |         <th class="author" onclick="sortTable(1)">Author</th> | ||||||
|         <th onclick="sortTable(2)">Rating</th> |         <th class="rating" onclick="sortTable(2)">Rating</th> | ||||||
|         <th onclick="sortTable(3)">Tags</th> |         <th class="tags" onclick="sortTable(3)">Tags</th> | ||||||
|         <th onclick="sortTable(4)">Added</th> |         <th class="added" onclick="sortTable(4)">Added</th> | ||||||
|     </tr> |     </tr> | ||||||
|     {% for book in books %} |     {% for book in books %} | ||||||
|     <tr> |     <tr> | ||||||
|         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> |         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> | ||||||
|         <td>{{book.author_sort}}</td> |         <td>{{book.author_sort}}</td> | ||||||
|         <td> {% for rating in book.rating.all %} |         <td> {% for rating in book.ratings.all %} | ||||||
|             {{rating}} |             {{rating}} | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </td> |         </td> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
|   <h1>Publishers List</h1> |   <h1>Publishers List</h1> | ||||||
|   {% if publisher_list %} |   {% if publisher_list %} | ||||||
|   <ul> |   <ul> | ||||||
|   | |||||||
| @@ -1,26 +1,24 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
|  |  | ||||||
| <h1>{{rating}}</h1> | <h1 class="left">Books with rating: {{rating}}</h1> | ||||||
| {% if books %} | {% if books %} | ||||||
| <table id="books"> | <table id="books" class="highlight centered"> | ||||||
|     <tr> |     <tr> | ||||||
|         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> |         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|         <th onclick="sortTable(0)">Title</th> |         <th class="title" onclick="sortTable(0)">Title</th> | ||||||
|         <th onclick="sortTable(1)">Author</th> |         <th class="author" onclick="sortTable(1)">Author</th> | ||||||
|         <th onclick="sortTable(2)">Rating</th> |         <th class="rating" onclick="sortTable(2)">Rating</th> | ||||||
|         <th onclick="sortTable(3)">Tags</th> |         <th class="tags" onclick="sortTable(3)">Tags</th> | ||||||
|         <th onclick="sortTable(4)">Added</th> |         <th class="added" onclick="sortTable(4)">Added</th> | ||||||
|     </tr> |     </tr> | ||||||
|     {% for book in books %} |     {% for book in books %} | ||||||
|     <tr> |     <tr> | ||||||
|         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> |         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> | ||||||
|         <td>{{book.author_sort}}</td> |         <td>{{book.author_sort}}</td> | ||||||
|         <td> {% for rating in book.rating.all %} |         <td>  | ||||||
|             {{rating}} |             {{rating}} | ||||||
|             {% endfor %} |  | ||||||
|         </td> |         </td> | ||||||
|         <td> |         <td> | ||||||
|             {% for tag in book.tags.all %} |             {% for tag in book.tags.all %} | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
|   <h1>Ratings List</h1> |   <h1>Ratings List</h1> | ||||||
|   {% if rating_list %} |   {% if rating_list %} | ||||||
|   <ul> |   <ul> | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								CalibreWebCompanion/library/templates/library/results.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  | {% load static %} | ||||||
|  | <h1>Results</h1> | ||||||
|  |  | ||||||
|  | <table id="books"> | ||||||
|  |   <tr> | ||||||
|  |     <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|  |     <th onclick="sortTable(0)">Title</th> | ||||||
|  |     <th onclick="sortTable(1)">Author</th> | ||||||
|  |     <th onclick="sortTable(2)">Rating</th> | ||||||
|  |     <th onclick="sortTable(3)">Tags</th> | ||||||
|  |     <th onclick="sortTable(4)">Added</th> | ||||||
|  |   </tr> | ||||||
|  |   {% for book in book_list %} | ||||||
|  |   <tr> | ||||||
|  |     <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> | ||||||
|  |     <td>{{book.author_sort}}</td> | ||||||
|  |     <td> {% for rating in book.ratings.all %} | ||||||
|  |       {{rating}} | ||||||
|  |       {% endfor %} | ||||||
|  |     </td> | ||||||
|  |     <td> | ||||||
|  |       {% for tag in book.tags.all %} | ||||||
|  |       {{tag}}, | ||||||
|  |       {% endfor %} | ||||||
|  |     </td> | ||||||
|  |     <td>{{book.timestamp}}</td> | ||||||
|  |   </tr> | ||||||
|  |   {% endfor %} | ||||||
|  | </table> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | {% endblock %} | ||||||
| @@ -0,0 +1,38 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  |  | ||||||
|  | <h1>{{Series}} </h1> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <table id="books" class="highlight centered"> | ||||||
|  |     <tr> | ||||||
|  |         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|  |         <th class="title" onclick="sortTable(0)">Title</th> | ||||||
|  |         <th class="author" onclick="sortTable(1)">Author</th> | ||||||
|  |         <th class="rating" onclick="sortTable(2)">Rating</th> | ||||||
|  |         <th class="tags" onclick="sortTable(3)">Tags</th> | ||||||
|  |         <th class="added" onclick="sortTable(4)">Added</th> | ||||||
|  |     </tr> | ||||||
|  |     {% for book in books %} | ||||||
|  |     <tr> | ||||||
|  |         <td><a href="{{ book.get_absolute_url }}">{{ book.title }}</a></td> | ||||||
|  |         <td>{{book.author_sort}}</td> | ||||||
|  |         <td> {% for rating in book.ratings.all %} | ||||||
|  |             {{rating}} | ||||||
|  |             {% endfor %} | ||||||
|  |         </td> | ||||||
|  |         <td> | ||||||
|  |             {% for tag in book.tags.all %} | ||||||
|  |             {{tag}}, | ||||||
|  |             {% endfor %} | ||||||
|  |         </td> | ||||||
|  |         <td>{{book.timestamp}}</td> | ||||||
|  |     </tr> | ||||||
|  |     {% endfor %} | ||||||
|  | </table> | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | {% endblock %} | ||||||
| @@ -0,0 +1,17 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  |  | ||||||
|  |   <h1>Author List</h1> | ||||||
|  |   {% if series_list %} | ||||||
|  |   <ul> | ||||||
|  |     {% for series in series_list %} | ||||||
|  |       <li> | ||||||
|  |         <a href="{{ series.get_absolute_url }}">{{ series.name }}</a> | ||||||
|  |       </li> | ||||||
|  |     {% endfor %} | ||||||
|  |   </ul> | ||||||
|  |   {% else %} | ||||||
|  |     <p>There are no series in the library.</p> | ||||||
|  |   {% endif %}        | ||||||
|  | {% endblock %} | ||||||
| @@ -1,18 +1,17 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
|  |  | ||||||
| <h1>{{tag}}</h1> | <h1>{{tag}}</h1> | ||||||
| {% if books %} | {% if books %} | ||||||
| <table id="books"> | <table id="books" class="highlight centered"> | ||||||
|     <tr> |     <tr> | ||||||
|         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> |         <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|         <th onclick="sortTable(0)">Title</th> |         <th class="title" onclick="sortTable(0)">Title</th> | ||||||
|         <th onclick="sortTable(1)">Author</th> |         <th class="author" onclick="sortTable(1)">Author</th> | ||||||
|         <th onclick="sortTable(2)">Rating</th> |         <th class="rating" onclick="sortTable(2)">Rating</th> | ||||||
|         <th onclick="sortTable(3)">Tags</th> |         <th class="tags" onclick="sortTable(3)">Tags</th> | ||||||
|         <th onclick="sortTable(4)">Added</th> |         <th class="added" onclick="sortTable(4)">Added</th> | ||||||
|     </tr> |     </tr> | ||||||
|     {% for book in books %} |     {% for book in books %} | ||||||
|     <tr> |     <tr> | ||||||
| @@ -23,7 +22,7 @@ | |||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </td> |         </td> | ||||||
|         <td> |         <td> | ||||||
|             {% for tag in book.tag.all %} |             {% for tag in book.tags.all %} | ||||||
|             {{tag}}, |             {{tag}}, | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </td> |         </td> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
|   <h1>Tags List</h1> |   <h1>Tags List</h1> | ||||||
|   {% if tag_list %} |   {% if tag_list %} | ||||||
|   <ul> |   <ul> | ||||||
|   | |||||||
| @@ -2,19 +2,6 @@ | |||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
|  |  | ||||||
|   {% if form.errors %} |  | ||||||
|     <p>Your username and password didn't match. Please try again.</p> |  | ||||||
|   {% endif %} |  | ||||||
|    |  | ||||||
|   {% if next %} |  | ||||||
|     {% if user.is_authenticated %} |  | ||||||
|       <p>Your account doesn't have access to this page. To proceed, |  | ||||||
|       please login with an account that has access.</p> |  | ||||||
|     {% else %} |  | ||||||
|       <p>Please login to see this page.</p> |  | ||||||
|     {% endif %} |  | ||||||
|   {% endif %} |  | ||||||
|    |  | ||||||
|   <form method="post" action="{% url 'login' %}"> |   <form method="post" action="{% url 'login' %}"> | ||||||
|     {% csrf_token %} |     {% csrf_token %} | ||||||
|     <table> |     <table> | ||||||
| @@ -27,7 +14,23 @@ | |||||||
|         <td>{{ form.password }}</td> |         <td>{{ form.password }}</td> | ||||||
|       </tr> |       </tr> | ||||||
|     </table> |     </table> | ||||||
|     <input type="submit" value="login" /> |  | ||||||
|  |     <div style="color:#f44336"> | ||||||
|  |     {% if form.errors %} | ||||||
|  |     <p>Your username and password didn't match. Please try again.</p> | ||||||
|  |     {% endif %} | ||||||
|  |    | ||||||
|  |     {% if next %} | ||||||
|  |     {% if user.is_authenticated %} | ||||||
|  |     <p>Your account doesn't have access to this page. To proceed, | ||||||
|  |     please login with an account that has access.</p> | ||||||
|  |     {% else %} | ||||||
|  |     <p>Please login to see this page.</p> | ||||||
|  |     {% endif %} | ||||||
|  |     {% endif %} | ||||||
|  |     </div> | ||||||
|  |  | ||||||
|  |     <button class="waves-effect waves-light btn green accent-4" type="submit">login</button> | ||||||
|     <input type="hidden" name="next" value="{{ next }}" /> |     <input type="hidden" name="next" value="{{ next }}" /> | ||||||
|   </form> |   </form> | ||||||
|    |    | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ | |||||||
| {% block content %} | {% block content %} | ||||||
|  |  | ||||||
| <form method="post"> | <form method="post"> | ||||||
|     {% csrf_token %} {{form.as_p}} <input type="submit" value="Sign up"></form> |     {% csrf_token %} {{form.as_p}} <button class="waves-effect waves-light btn green accent-4" type="submit">sign up</button></form> | ||||||
|    |    | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,17 +1,17 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
| <h1>Results</h1> | <h1>Results</h1> | ||||||
|  |  | ||||||
| <table id="books"> | <table id="books"> | ||||||
|   <tr> |   <tr> | ||||||
|     <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> |     <!--When a header is clicked, run the sortTable function, with a parameter, 0 for sorting by names, 1 for sorting by country:--> | ||||||
|     <th onclick="sortTable(0)">Title</th> |     <th class="title" onclick="sortTable(0)">Title</th> | ||||||
|     <th onclick="sortTable(1)">Author</th> |     <th class="author" onclick="sortTable(1)">Author</th> | ||||||
|     <th onclick="sortTable(2)">Rating</th> |     <th class="rating" onclick="sortTable(2)">Rating</th> | ||||||
|     <th onclick="sortTable(3)">Tags</th> |     <th class="tags" onclick="sortTable(3)">Tags</th> | ||||||
|     <th onclick="sortTable(4)">Added</th> |     <th class="added" onclick="sortTable(4)">Added</th> | ||||||
|   </tr> |   </tr> | ||||||
|   {% for book in book_list %} |   {% for book in book_list %} | ||||||
|   <tr> |   <tr> | ||||||
|   | |||||||
| @@ -1,15 +1,17 @@ | |||||||
| {% extends "base.html" %} | {% extends "base.html" %} | ||||||
|  |  | ||||||
| {% block content %} | {% block content %} | ||||||
| {% load static %} |  | ||||||
| <form action="{% url 'results' %}" method="get"> | <div class="container"> | ||||||
|  | <form action="{% url 'results' %}" method="get" style="padding-top:2em"> | ||||||
|     <label for="title">Title: </label> |     <label for="title">Title: </label> | ||||||
|     <input id="title" type="text" name="title" value=""> |     <input id="title" type="text" name="title" value=""> | ||||||
|     <label for="author">Author: </label> |     <label for="author">Author: </label> | ||||||
|     <input id="author" type="text" name="author" value=""> |     <input id="author" type="text" name="author" value=""> | ||||||
|     <input type="submit" value="search"> |     <label for="author">Identifier: </label> | ||||||
|  |     <input id="identifier" type="text" name="identifier" value=""> | ||||||
|  |     <button class="waves-effect waves-light btn green accent-4" type="submit">search</button> | ||||||
| </form> | </form> | ||||||
|  | </div> | ||||||
|  |  | ||||||
| {% endblock %} | {% endblock %} | ||||||
| @@ -5,36 +5,39 @@ | |||||||
|   {% block title %}<title>Local Library</title>{% endblock %} |   {% block title %}<title>Local Library</title>{% endblock %} | ||||||
|   <meta charset="utf-8"> |   <meta charset="utf-8"> | ||||||
|   <meta name="viewport" content="width=device-width, initial-scale=1"> |   <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||||
|   <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.1.3/css/bootstrap.min.css" |  | ||||||
|     integrity="sha384-MCw98/SFnGE8fJT3GXwEOngsV7Zt27NXFoaoApmYm81iuXoPkFOJwJ8ERdknLPMO" crossorigin="anonymous"> |   <script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script> | ||||||
|   <!-- Add additional CSS in static file --> |   <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet"> | ||||||
|   {% load static %} |   <!-- Compiled and minified CSS --> | ||||||
|   <link rel="stylesheet" href="{% static 'css/styles.css' %}"> |   <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/materialize/1.0.0-beta/css/materialize.min.css"> | ||||||
|  |   <!-- Compiled and minified JavaScript --> | ||||||
|  |   <script src="https://cdnjs.cloudflare.com/ajax/libs/materialize/1.0.0-beta/js/materialize.min.js"></script> | ||||||
| </head> | </head> | ||||||
|  |  | ||||||
| <body> | <body> | ||||||
|   {% block topnav%} |  | ||||||
|   <ul class="topnav"> |  | ||||||
|  |  | ||||||
|     {% if user.is_authenticated %} | <div class="navbar-fixed"> | ||||||
|      <li class="active"><a href="{{user.get_absolute_url}}"></a> {{ user.get_username }}</a></li> |   <nav> | ||||||
|      <li><a href="{% url 'logout'%}?next={{request.path}}">Logout</a></li>    |     <div class="nav-wrapper row green darken-1"> | ||||||
|    {% else %} |  | ||||||
|    <li><a href="{% url 'sign-up'%}?next={{request.path}}">Sign up</a></li>  |  | ||||||
|      <li><a href="{% url 'login'%}?next={{request.path}}">Login</a></li>    |  | ||||||
|    {% endif %}  |  | ||||||
|   </ul> |  | ||||||
|   {% endblock %} |  | ||||||
|  |  | ||||||
|    |       <ul class="right"> | ||||||
|  |       {% if user.is_authenticated %} | ||||||
|   <div class="container-fluid"> |         <li class="active"><a href="{{user.get_absolute_url}}"> {{ user.get_username }}</a></li> | ||||||
|     <div class="row"> |         <li><a href="{% url 'logout'%}?next={{request.path}}">Logout</a></li> | ||||||
|  |       </ul> | ||||||
|       <div class="col-sm-10 ">{% block content %}{% endblock %}</div> |       {% else %} | ||||||
|  |         <li><a href="{% url 'sign-up'%}?next={{request.path}}">Sign up</a></li> | ||||||
|  |         <li><a href="{% url 'login'%}?next={{request.path}}">Login</a></li> | ||||||
|  |       </ul> | ||||||
|  |       {% endif %} | ||||||
|     </div> |     </div> | ||||||
|  |   </nav> | ||||||
|  | </div> | ||||||
|  | <div class="container"> | ||||||
|  |   <div class="row"> | ||||||
|  |     <div class="col s10 ">{% block content %}{% endblock %}</div> | ||||||
|   </div> |   </div> | ||||||
|  | </div> | ||||||
|  |  | ||||||
| </body> | </body> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,3 +1,166 @@ | |||||||
| from django.test import TestCase | from django.test import Client, TestCase | ||||||
|  | from pprint import pprint | ||||||
|  | from django.test.utils import setup_test_environment | ||||||
|  | from .models import Book, Author, Publisher, Series, Rating, Tag, Identifier | ||||||
|  | from django.db.models import Count | ||||||
|  |  | ||||||
| # Create your tests here. |  | ||||||
|  | client = Client() | ||||||
|  | client.login(username="testuser", password="dumbeasypassword") | ||||||
|  |  | ||||||
|  | def booklisttest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     res = c.get("/books/") | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |     context = dict(res.context) | ||||||
|  |     assert sorted(context["book_list"], key=lambda x: x.id)== sorted(Book.objects.all(), key=lambda x: x.id) | ||||||
|  |  | ||||||
|  | def authorlisttest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     res = c.get("/authors/") | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |     context = dict(res.context) | ||||||
|  |     assert sorted(context["author_list"], key=lambda x: x.id)== sorted(Author.objects.all(), key=lambda x: x.id) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def publisherlisttest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     res = c.get("/publishers/") | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |     context = dict(res.context) | ||||||
|  |     assert sorted(context["publisher_list"], key=lambda x: x.id)== sorted(Publisher.objects.all(), key=lambda x: x.id) | ||||||
|  |  | ||||||
|  | def serieslisttest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     res = c.get("/series/") | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |     context = dict(res.context) | ||||||
|  |     assert sorted(context["series_list"], key=lambda x: x.id)== sorted(Series.objects.all(), key=lambda x: x.id) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def ratinglisttest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     res = c.get("/ratings/") | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |     context = dict(res.context) | ||||||
|  |     assert sorted(context["rating_list"], key=lambda x: x.id)== sorted(Rating.objects.all(), key=lambda x: x.id) | ||||||
|  |  | ||||||
|  | def taglisttest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     res = c.get("/tags/") | ||||||
|  |     assert res.status_code == 200 | ||||||
|  |     context = dict(res.context) | ||||||
|  |     assert sorted(context["tag_list"], key=lambda x: x.id)== sorted(Tag.objects.all(), key=lambda x: x.id) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def bookdetailtest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     ids = [i.id for i in Book.objects.all()][:10] | ||||||
|  |     for i in ids: | ||||||
|  |         res = c.get(f"/book/{i}") | ||||||
|  |         assert res.status_code == 200 | ||||||
|  |         context = dict(res.context) | ||||||
|  |         assert context["book"] == Book.objects.get(id=i) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def authordetailtest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     ids = [i.id for i in Author.objects.all()][:10] | ||||||
|  |     for i in ids: | ||||||
|  |         res = c.get(f"/author/{i}") | ||||||
|  |         assert res.status_code == 200 | ||||||
|  |         context = dict(res.context) | ||||||
|  |         assert context["author"] == Author.objects.get(id=i) | ||||||
|  |  | ||||||
|  | def publisherdetailtest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     ids = [i.id for i in Publisher.objects.all()][:10] | ||||||
|  |     for i in ids: | ||||||
|  |         res = c.get(f"/publisher/{i}") | ||||||
|  |         assert res.status_code == 200 | ||||||
|  |         context = dict(res.context) | ||||||
|  |         assert context["publisher"] == Publisher.objects.get(id=i) | ||||||
|  |  | ||||||
|  | def seriesdetailtest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     ids = [i.id for i in Series.objects.all()][:10] | ||||||
|  |     for i in ids: | ||||||
|  |         res = c.get(f"/series/{i}") | ||||||
|  |         assert res.status_code == 200 | ||||||
|  |         context = dict(res.context) | ||||||
|  |         assert context["series"] == Series.objects.get(id=i) | ||||||
|  |  | ||||||
|  | def ratingdetailtest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     ids = [i.id for i in Rating.objects.all()][:10] | ||||||
|  |     for i in ids: | ||||||
|  |         res = c.get(f"/rating/{i}") | ||||||
|  |         assert res.status_code == 200 | ||||||
|  |         context = dict(res.context) | ||||||
|  |         assert context["rating"] == Rating.objects.get(id=i)         | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def tagdetailtest(): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     ids = [i.id for i in Tag.objects.all()][:10] | ||||||
|  |     for i in ids: | ||||||
|  |         res = c.get(f"/tag/{i}") | ||||||
|  |         assert res.status_code == 200 | ||||||
|  |         context = dict(res.context) | ||||||
|  |         assert context["tag"] == Tag.objects.get(id=i)  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def search_partial(key, value, book=None): | ||||||
|  |     c = Client() | ||||||
|  |     c.login(username="testuser", password="dumbeasypassword") | ||||||
|  |     res = c.get("/results/", {key : value}) | ||||||
|  |     if not book: | ||||||
|  |         return dict(res.context)["book_list"] | ||||||
|  |     return book in dict(res.context)["book_list"] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def searchtest(): | ||||||
|  |     books = [i for i in Book.objects.all()][:10] | ||||||
|  |     for i in books: | ||||||
|  |         assert search_partial("title", i.title, i) | ||||||
|  |         assert search_partial("generic", i.title, i) | ||||||
|  |         assert search_partial("author", i.author_sort, i) | ||||||
|  |         author = i.authors.first() | ||||||
|  |         if author: | ||||||
|  |             assert search_partial("author", author.name, i) | ||||||
|  |             assert search_partial("generic", author.name, i) | ||||||
|  |  | ||||||
|  |         assert search_partial("generic", i.author_sort, i) | ||||||
|  |         id = Identifier.objects.filter(book=i.id).first() | ||||||
|  |         if id: | ||||||
|  |             assert search_partial("identifier", id, i) | ||||||
|  |             assert search_partial("generic", id, i) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |      | ||||||
|  |  | ||||||
|  | booklisttest() | ||||||
|  | bookdetailtest() | ||||||
|  | authorlisttest() | ||||||
|  | authordetailtest() | ||||||
|  | publisherdetailtest() | ||||||
|  | publisherlisttest() | ||||||
|  | seriesdetailtest() | ||||||
|  | serieslisttest() | ||||||
|  | ratingdetailtest() | ||||||
|  | ratinglisttest() | ||||||
|  | tagdetailtest() | ||||||
|  | taglisttest() | ||||||
|  | searchtest() | ||||||
| @@ -1,7 +1,9 @@ | |||||||
| from django.urls import path | from django.urls import path | ||||||
| from . import views | from . import views | ||||||
|  | from django.views.decorators.cache import cache_page | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| urlpatterns = [ | urlpatterns = [ | ||||||
|     path('authors/', views.AuthorListView.as_view(), name='authors'), |     path('authors/', views.AuthorListView.as_view(), name='authors'), | ||||||
| @@ -9,17 +11,21 @@ urlpatterns = [ | |||||||
|     path('publishers/', views.PublisherListView.as_view(), name='publishers'), |     path('publishers/', views.PublisherListView.as_view(), name='publishers'), | ||||||
|     path('ratings/', views.RatingListView.as_view(), name='ratings'), |     path('ratings/', views.RatingListView.as_view(), name='ratings'), | ||||||
|     path('tags/', views.TagListView.as_view(), name='tags'), |     path('tags/', views.TagListView.as_view(), name='tags'), | ||||||
|  |     path('series/', views.SeriesListView.as_view(), name='series'), | ||||||
|     path('author/<int:pk>', views.AuthorDetailView.as_view(), name='author-detail-view'), |     path('author/<int:pk>', views.AuthorDetailView.as_view(), | ||||||
|  |          name='author-detail-view'), | ||||||
|     path('book/<int:pk>', views.BookDetailView.as_view(), name='book-detail-view'), |     path('book/<int:pk>', views.BookDetailView.as_view(), name='book-detail-view'), | ||||||
|     path('publisher/<int:pk>', views.PublisherDetailView.as_view(), name='publisher-detail-view'), |     path('publisher/<int:pk>', views.PublisherDetailView.as_view(), | ||||||
|     path('rating/<int:pk>', views.RatingDetailView.as_view(), name='rating-detail-view'), |          name='publisher-detail-view'), | ||||||
|  |     path('rating/<int:pk>', views.RatingDetailView.as_view(), | ||||||
|  |          name='rating-detail-view'), | ||||||
|  |     path('series/<int:pk>', views.SeriesDetailView.as_view(), | ||||||
|  |          name='series-detail-view'), | ||||||
|     path('tag/<int:pk>', views.TagDetailView.as_view(), name='tag-detail-view'), |     path('tag/<int:pk>', views.TagDetailView.as_view(), name='tag-detail-view'), | ||||||
|  |  | ||||||
|     path('results/', views.ResultsView.as_view(), name='results'), |     path('results/', views.ResultsView.as_view(), name='results'), | ||||||
|     path('search/', views.SearchView.as_view(), name='search'), |     path('search/', views.SearchView.as_view(), name='search'), | ||||||
|  |     path('accounts/sign_up/', views.sign_up, name="sign-up") | ||||||
|     path('accounts/sign_up/',views.sign_up,name="sign-up") |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ] | ] | ||||||
| @@ -1,6 +1,7 @@ | |||||||
|  | from django.utils.decorators import method_decorator | ||||||
| from django.shortcuts import render | from django.shortcuts import render | ||||||
| from django.views import generic | from django.views import generic | ||||||
| from .models import Author, Book, Comment, Rating, BookAuthorLink, Publisher, Tag, BookTagLink, BookRatingLink, Data | from .models import Author, Book, Comment, Rating, BookAuthorLink, Publisher, Tag, BookTagLink, BookRatingLink, Data, Identifier, Series | ||||||
| from django.http import HttpResponseRedirect | from django.http import HttpResponseRedirect | ||||||
| from .forms import SearchForm, UserCreationForm | from .forms import SearchForm, UserCreationForm | ||||||
| from django.db import models | from django.db import models | ||||||
| @@ -8,6 +9,11 @@ from django.db.models import Q | |||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.contrib.auth import login | from django.contrib.auth import login | ||||||
| from django.contrib.auth.decorators import login_required | from django.contrib.auth.decorators import login_required | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  | # might be helpful for vary headers later | ||||||
|  |  | ||||||
|  |  | ||||||
| @login_required | @login_required | ||||||
| @@ -30,6 +36,9 @@ def sign_up(request): | |||||||
| class SearchView(generic.TemplateView): | class SearchView(generic.TemplateView): | ||||||
|     template_name = 'search.html' |     template_name = 'search.html' | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(SearchView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ResultsView(generic.ListView):  # no clue if this is secure. | class ResultsView(generic.ListView):  # no clue if this is secure. | ||||||
|     # according to this https://stackoverflow.com/questions/13574043/how-do-django-forms-sanitize-text-input-to-prevent-sql-injection-xss-etc |     # according to this https://stackoverflow.com/questions/13574043/how-do-django-forms-sanitize-text-input-to-prevent-sql-injection-xss-etc | ||||||
| @@ -37,23 +46,59 @@ class ResultsView(generic.ListView):  # no clue if this is secure. | |||||||
|     model = Book |     model = Book | ||||||
|     template_name = 'results.html' |     template_name = 'results.html' | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(ResultsView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_queryset(self):  # new |     def get_queryset(self):  # new | ||||||
|         title = self.request.GET.get('title') |         title = self.request.GET.get('title') | ||||||
|         author = self.request.GET.get('author') |         author = self.request.GET.get('author') | ||||||
|  |         identifier = self.request.GET.get("identifier") | ||||||
|  |         generic = self.request.GET.get("generic") | ||||||
|         books = Book.objects.prefetch_related("tags", "ratings") |         books = Book.objects.prefetch_related("tags", "ratings") | ||||||
|         if title: |         if title: | ||||||
|             books =books.filter(sort__icontains=title) |             books = books.filter(sort__icontains=title) | ||||||
|         if author: |         if author: | ||||||
|             books = books.filter(author_sort__icontains=author) |             # authors are stored as author_sort and author, needs to be slightly more complex | ||||||
|  |             author_obj = Author.objects.filter(name__icontains=author).first() | ||||||
|  |             if not author_obj: | ||||||
|  |                 author_id = -1 | ||||||
|  |             else: | ||||||
|  |                 author_id = author_obj.id | ||||||
|  |  | ||||||
|  |             books = books.filter( | ||||||
|  |                 Q(author_sort__icontains=author) | | ||||||
|  |                 Q(authors__id=author_id) | ||||||
|  |             ) | ||||||
|  |         if identifier: | ||||||
|  |             books = books.filter(identifier__val=identifier) | ||||||
|  |         if generic: | ||||||
|  |             author_obj = Author.objects.filter(name__icontains=generic).first() | ||||||
|  |             if not author_obj: | ||||||
|  |                 author_id = -1 | ||||||
|  |             else: | ||||||
|  |                 author_id = author_obj.id | ||||||
|  |             books = books.filter( | ||||||
|  |                 Q(sort__icontains=generic) | | ||||||
|  |                 Q(author_sort__icontains=generic) | | ||||||
|  |                 Q(authors__id=author_id) | | ||||||
|  |                 Q(identifier__val=generic) | ||||||
|  |             ) | ||||||
|         return books |         return books | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthorListView(generic.ListView): | class AuthorListView(generic.ListView): | ||||||
|     model = Author |     model = Author | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(AuthorListView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class BookListView(generic.ListView): | class BookListView(generic.ListView): | ||||||
|     model = Book |     model = Book | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(BookListView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_queryset(self): |     def get_queryset(self): | ||||||
|         # Annotate the books with ratings, tags, etc |         # Annotate the books with ratings, tags, etc | ||||||
|         # books = Book.objects.annotate( |         # books = Book.objects.annotate( | ||||||
| @@ -64,18 +109,37 @@ class BookListView(generic.ListView): | |||||||
| class PublisherListView(generic.ListView): | class PublisherListView(generic.ListView): | ||||||
|     model = Publisher |     model = Publisher | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(PublisherListView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class RatingListView(generic.ListView): | class RatingListView(generic.ListView): | ||||||
|     model = Rating |     model = Rating | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(RatingListView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SeriesListView(generic.ListView):  # make url entry and template, sometime | ||||||
|  |     model = Series | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(SeriesListView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TagListView(generic.ListView): | class TagListView(generic.ListView): | ||||||
|     model = Tag |     model = Tag | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(TagListView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthorDetailView(generic.DetailView): | class AuthorDetailView(generic.DetailView): | ||||||
|     model = Author |     model = Author | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(AuthorDetailView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         # Call the base implementation first to get the context |         # Call the base implementation first to get the context | ||||||
|         context = super(AuthorDetailView, self).get_context_data(**kwargs) |         context = super(AuthorDetailView, self).get_context_data(**kwargs) | ||||||
| @@ -89,6 +153,9 @@ class AuthorDetailView(generic.DetailView): | |||||||
| class BookDetailView(generic.DetailView): | class BookDetailView(generic.DetailView): | ||||||
|     model = Book |     model = Book | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(BookDetailView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         # Call the base implementation first to get the context |         # Call the base implementation first to get the context | ||||||
|         context = super(BookDetailView, self).get_context_data(**kwargs) |         context = super(BookDetailView, self).get_context_data(**kwargs) | ||||||
| @@ -100,13 +167,16 @@ class BookDetailView(generic.DetailView): | |||||||
|             pass |             pass | ||||||
|         context["imgpath"] = context["object"].path + "/cover.jpg" |         context["imgpath"] = context["object"].path + "/cover.jpg" | ||||||
|         download = Data.objects.get(book=context["object"].id) |         download = Data.objects.get(book=context["object"].id) | ||||||
|         context["download"] = f"{context['object'].path}/{download.name}.{download.format}" |         context["download"] = f"{context['object'].path}/{download.name}.{download.format.lower()}" | ||||||
|         return context |         return context | ||||||
|  |  | ||||||
|  |  | ||||||
| class PublisherDetailView(generic.DetailView): | class PublisherDetailView(generic.DetailView): | ||||||
|     model = Publisher |     model = Publisher | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(PublisherDetailView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         # Call the base implementation first to get the context |         # Call the base implementation first to get the context | ||||||
|         context = super(PublisherDetailView, self).get_context_data(**kwargs) |         context = super(PublisherDetailView, self).get_context_data(**kwargs) | ||||||
| @@ -120,6 +190,9 @@ class PublisherDetailView(generic.DetailView): | |||||||
| class RatingDetailView(generic.DetailView): | class RatingDetailView(generic.DetailView): | ||||||
|     model = Rating |     model = Rating | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(RatingDetailView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         # Call the base implementation first to get the context |         # Call the base implementation first to get the context | ||||||
|         context = super(RatingDetailView, self).get_context_data(**kwargs) |         context = super(RatingDetailView, self).get_context_data(**kwargs) | ||||||
| @@ -133,6 +206,9 @@ class RatingDetailView(generic.DetailView): | |||||||
| class TagDetailView(generic.DetailView): | class TagDetailView(generic.DetailView): | ||||||
|     model = Tag |     model = Tag | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(TagDetailView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|     def get_context_data(self, **kwargs): |     def get_context_data(self, **kwargs): | ||||||
|         # Call the base implementation first to get the context |         # Call the base implementation first to get the context | ||||||
|         context = super(TagDetailView, self).get_context_data(**kwargs) |         context = super(TagDetailView, self).get_context_data(**kwargs) | ||||||
| @@ -141,3 +217,19 @@ class TagDetailView(generic.DetailView): | |||||||
|         books = books.filter(tags=context["object"].id) |         books = books.filter(tags=context["object"].id) | ||||||
|         context['books'] = sorted(books,  key=lambda x: x.title) |         context['books'] = sorted(books,  key=lambda x: x.title) | ||||||
|         return context |         return context | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SeriesDetailView(generic.DetailView): | ||||||
|  |     model = Series | ||||||
|  |  | ||||||
|  |     def dispatch(self, *args, **kwargs): | ||||||
|  |         return super(SeriesDetailView, self).dispatch(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def get_context_data(self, **kwargs): | ||||||
|  |         # Call the base implementation first to get the context | ||||||
|  |         context = super(SeriesDetailView, self).get_context_data(**kwargs) | ||||||
|  |         # Create any data and add it to the context | ||||||
|  |         books = Book.objects.prefetch_related("tags", "ratings") | ||||||
|  |         books = books.filter(series=context["object"].id) | ||||||
|  |         context['books'] = sorted(books,  key=lambda x: x.title) | ||||||
|  |         return context | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								CalibreWebCompanion/manage.py
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
							
								
								
									
										13
									
								
								CalibreWebCompanion/settings.json.bak
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | { | ||||||
|  |     "CALIBRE_DIR": "/app/content", | ||||||
|  |     "LOGFOLDER" : "/usr/src/app/data/logs/", | ||||||
|  |     "SECRET_KEY": "u(8^+rb%rz5hsx4v^^y(ul7g(4n7a8!db@s*9(m5cs*2_ppy8+", | ||||||
|  |     "ALLOWED_HOSTS": [ | ||||||
|  |         "*" | ||||||
|  |     ], | ||||||
|  |     "INTERNAL_IPS": [ | ||||||
|  |         "127.0.0.1", "localhost" | ||||||
|  |     ], | ||||||
|  |     "DEBUG" : false | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										260
									
								
								CalibreWebCompanion/static/admin/css/autocomplete.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,260 @@ | |||||||
|  | select.admin-autocomplete { | ||||||
|  |     width: 20em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container { | ||||||
|  |     min-height: 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--single, | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple { | ||||||
|  |     min-height: 30px; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--focus .select2-selection, | ||||||
|  | .select2-container--admin-autocomplete.select2-container--open .select2-selection { | ||||||
|  |     border-color: #999; | ||||||
|  |     min-height: 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--focus .select2-selection.select2-selection--single, | ||||||
|  | .select2-container--admin-autocomplete.select2-container--open .select2-selection.select2-selection--single { | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--focus .select2-selection.select2-selection--multiple, | ||||||
|  | .select2-container--admin-autocomplete.select2-container--open .select2-selection.select2-selection--multiple { | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--single { | ||||||
|  |     background-color: #fff; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-radius: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--single .select2-selection__rendered { | ||||||
|  |     color: #444; | ||||||
|  |     line-height: 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--single .select2-selection__clear { | ||||||
|  |     cursor: pointer; | ||||||
|  |     float: right; | ||||||
|  |     font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--single .select2-selection__placeholder { | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--single .select2-selection__arrow { | ||||||
|  |     height: 26px; | ||||||
|  |     position: absolute; | ||||||
|  |     top: 1px; | ||||||
|  |     right: 1px; | ||||||
|  |     width: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--single .select2-selection__arrow b { | ||||||
|  |     border-color: #888 transparent transparent transparent; | ||||||
|  |     border-style: solid; | ||||||
|  |     border-width: 5px 4px 0 4px; | ||||||
|  |     height: 0; | ||||||
|  |     left: 50%; | ||||||
|  |     margin-left: -4px; | ||||||
|  |     margin-top: -2px; | ||||||
|  |     position: absolute; | ||||||
|  |     top: 50%; | ||||||
|  |     width: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete[dir="rtl"] .select2-selection--single .select2-selection__clear { | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete[dir="rtl"] .select2-selection--single .select2-selection__arrow { | ||||||
|  |     left: 1px; | ||||||
|  |     right: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--disabled .select2-selection--single { | ||||||
|  |     background-color: #eee; | ||||||
|  |     cursor: default; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--disabled .select2-selection--single .select2-selection__clear { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--open .select2-selection--single .select2-selection__arrow b { | ||||||
|  |     border-color: transparent transparent #888 transparent; | ||||||
|  |     border-width: 0 4px 5px 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple { | ||||||
|  |     background-color: white; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     cursor: text; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple .select2-selection__rendered { | ||||||
|  |     box-sizing: border-box; | ||||||
|  |     list-style: none; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0 5px; | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple .select2-selection__rendered li { | ||||||
|  |     list-style: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple .select2-selection__placeholder { | ||||||
|  |     color: #999; | ||||||
|  |     margin-top: 5px; | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple .select2-selection__clear { | ||||||
|  |     cursor: pointer; | ||||||
|  |     float: right; | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin: 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple .select2-selection__choice { | ||||||
|  |     background-color: #e4e4e4; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     cursor: default; | ||||||
|  |     float: left; | ||||||
|  |     margin-right: 5px; | ||||||
|  |     margin-top: 5px; | ||||||
|  |     padding: 0 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple .select2-selection__choice__remove { | ||||||
|  |     color: #999; | ||||||
|  |     cursor: pointer; | ||||||
|  |     display: inline-block; | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-right: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-selection--multiple .select2-selection__choice__remove:hover { | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete[dir="rtl"] .select2-selection--multiple .select2-selection__choice, .select2-container--admin-autocomplete[dir="rtl"] .select2-selection--multiple .select2-selection__placeholder, .select2-container--admin-autocomplete[dir="rtl"] .select2-selection--multiple .select2-search--inline { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete[dir="rtl"] .select2-selection--multiple .select2-selection__choice { | ||||||
|  |     margin-left: 5px; | ||||||
|  |     margin-right: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete[dir="rtl"] .select2-selection--multiple .select2-selection__choice__remove { | ||||||
|  |     margin-left: 2px; | ||||||
|  |     margin-right: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--focus .select2-selection--multiple { | ||||||
|  |     border: solid #999 1px; | ||||||
|  |     outline: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--disabled .select2-selection--multiple { | ||||||
|  |     background-color: #eee; | ||||||
|  |     cursor: default; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--disabled .select2-selection__choice__remove { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--open.select2-container--above .select2-selection--single, .select2-container--admin-autocomplete.select2-container--open.select2-container--above .select2-selection--multiple { | ||||||
|  |     border-top-left-radius: 0; | ||||||
|  |     border-top-right-radius: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete.select2-container--open.select2-container--below .select2-selection--single, .select2-container--admin-autocomplete.select2-container--open.select2-container--below .select2-selection--multiple { | ||||||
|  |     border-bottom-left-radius: 0; | ||||||
|  |     border-bottom-right-radius: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-search--dropdown .select2-search__field { | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-search--inline .select2-search__field { | ||||||
|  |     background: transparent; | ||||||
|  |     border: none; | ||||||
|  |     outline: 0; | ||||||
|  |     box-shadow: none; | ||||||
|  |     -webkit-appearance: textfield; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results > .select2-results__options { | ||||||
|  |     max-height: 200px; | ||||||
|  |     overflow-y: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option[role=group] { | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option[aria-disabled=true] { | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option[aria-selected=true] { | ||||||
|  |     background-color: #ddd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option .select2-results__option { | ||||||
|  |     padding-left: 1em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option .select2-results__option .select2-results__group { | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |     margin-left: -1em; | ||||||
|  |     padding-left: 2em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |     margin-left: -2em; | ||||||
|  |     padding-left: 3em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |     margin-left: -3em; | ||||||
|  |     padding-left: 4em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |     margin-left: -4em; | ||||||
|  |     padding-left: 5em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |     margin-left: -5em; | ||||||
|  |     padding-left: 6em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__option--highlighted[aria-selected] { | ||||||
|  |     background-color: #79aec8; | ||||||
|  |     color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .select2-container--admin-autocomplete .select2-results__group { | ||||||
|  |     cursor: default; | ||||||
|  |     display: block; | ||||||
|  |     padding: 6px; | ||||||
|  | } | ||||||
							
								
								
									
										987
									
								
								CalibreWebCompanion/static/admin/css/base.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,987 @@ | |||||||
|  | /* | ||||||
|  |     DJANGO Admin styles | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | @import url(fonts.css); | ||||||
|  |  | ||||||
|  | body { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     font-size: 14px; | ||||||
|  |     font-family: "Roboto","Lucida Grande","DejaVu Sans","Bitstream Vera Sans",Verdana,Arial,sans-serif; | ||||||
|  |     color: #333; | ||||||
|  |     background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* LINKS */ | ||||||
|  |  | ||||||
|  | a:link, a:visited { | ||||||
|  |     color: #447e9b; | ||||||
|  |     text-decoration: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a:focus, a:hover { | ||||||
|  |     color: #036; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a:focus { | ||||||
|  |     text-decoration: underline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a img { | ||||||
|  |     border: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.section:link, a.section:visited { | ||||||
|  |     color: #fff; | ||||||
|  |     text-decoration: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.section:focus, a.section:hover { | ||||||
|  |     text-decoration: underline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* GLOBAL DEFAULTS */ | ||||||
|  |  | ||||||
|  | p, ol, ul, dl { | ||||||
|  |     margin: .2em 0 .8em 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | p { | ||||||
|  |     padding: 0; | ||||||
|  |     line-height: 140%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h1,h2,h3,h4,h5 { | ||||||
|  |     font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h1 { | ||||||
|  |     margin: 0 0 20px; | ||||||
|  |     font-weight: 300; | ||||||
|  |     font-size: 20px; | ||||||
|  |     color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h2 { | ||||||
|  |     font-size: 16px; | ||||||
|  |     margin: 1em 0 .5em 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h2.subhead { | ||||||
|  |     font-weight: normal; | ||||||
|  |     margin-top: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h3 { | ||||||
|  |     font-size: 14px; | ||||||
|  |     margin: .8em 0 .3em 0; | ||||||
|  |     color: #666; | ||||||
|  |     font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h4 { | ||||||
|  |     font-size: 12px; | ||||||
|  |     margin: 1em 0 .8em 0; | ||||||
|  |     padding-bottom: 3px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | h5 { | ||||||
|  |     font-size: 10px; | ||||||
|  |     margin: 1.5em 0 .5em 0; | ||||||
|  |     color: #666; | ||||||
|  |     text-transform: uppercase; | ||||||
|  |     letter-spacing: 1px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul li { | ||||||
|  |     list-style-type: square; | ||||||
|  |     padding: 1px 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | li ul { | ||||||
|  |     margin-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | li, dt, dd { | ||||||
|  |     font-size: 13px; | ||||||
|  |     line-height: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | dt { | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-top: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | dd { | ||||||
|  |     margin-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     border: none; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | blockquote { | ||||||
|  |     font-size: 11px; | ||||||
|  |     color: #777; | ||||||
|  |     margin-left: 2px; | ||||||
|  |     padding-left: 10px; | ||||||
|  |     border-left: 5px solid #ddd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | code, pre { | ||||||
|  |     font-family: "Bitstream Vera Sans Mono", Monaco, "Courier New", Courier, monospace; | ||||||
|  |     color: #666; | ||||||
|  |     font-size: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pre.literal-block { | ||||||
|  |     margin: 10px; | ||||||
|  |     background: #eee; | ||||||
|  |     padding: 6px 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | code strong { | ||||||
|  |     color: #930; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | hr { | ||||||
|  |     clear: both; | ||||||
|  |     color: #eee; | ||||||
|  |     background-color: #eee; | ||||||
|  |     height: 1px; | ||||||
|  |     border: none; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     font-size: 1px; | ||||||
|  |     line-height: 1px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* TEXT STYLES & MODIFIERS */ | ||||||
|  |  | ||||||
|  | .small { | ||||||
|  |     font-size: 11px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .tiny { | ||||||
|  |     font-size: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | p.tiny { | ||||||
|  |     margin-top: -2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .mini { | ||||||
|  |     font-size: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | p.mini { | ||||||
|  |     margin-top: -3px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .help, p.help, form p.help, div.help, form div.help, div.help li { | ||||||
|  |     font-size: 11px; | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.help ul { | ||||||
|  |      margin-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .help-tooltip { | ||||||
|  |     cursor: help; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | p img, h1 img, h2 img, h3 img, h4 img, td img { | ||||||
|  |     vertical-align: middle; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .quiet, a.quiet:link, a.quiet:visited { | ||||||
|  |     color: #999; | ||||||
|  |     font-weight: normal; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .float-right { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .float-left { | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .clear { | ||||||
|  |     clear: both; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .align-left { | ||||||
|  |     text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .align-right { | ||||||
|  |     text-align: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .example { | ||||||
|  |     margin: 10px 0; | ||||||
|  |     padding: 5px 10px; | ||||||
|  |     background: #efefef; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .nowrap { | ||||||
|  |     white-space: nowrap; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* TABLES */ | ||||||
|  |  | ||||||
|  | table { | ||||||
|  |     border-collapse: collapse; | ||||||
|  |     border-color: #ccc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | td, th { | ||||||
|  |     font-size: 13px; | ||||||
|  |     line-height: 16px; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  |     vertical-align: top; | ||||||
|  |     padding: 8px; | ||||||
|  |     font-family: "Roboto", "Lucida Grande", Verdana, Arial, sans-serif; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | th { | ||||||
|  |     font-weight: 600; | ||||||
|  |     text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th, | ||||||
|  | tfoot td { | ||||||
|  |     color: #666; | ||||||
|  |     padding: 5px 10px; | ||||||
|  |     font-size: 11px; | ||||||
|  |     background: #fff; | ||||||
|  |     border: none; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tfoot td { | ||||||
|  |     border-bottom: none; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th.required { | ||||||
|  |     color: #000; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tr.alt { | ||||||
|  |     background: #f6f6f6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .row1 { | ||||||
|  |     background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .row2 { | ||||||
|  |     background: #f9f9f9; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* SORTABLE TABLES */ | ||||||
|  |  | ||||||
|  | thead th { | ||||||
|  |     padding: 5px 10px; | ||||||
|  |     line-height: normal; | ||||||
|  |     text-transform: uppercase; | ||||||
|  |     background: #f6f6f6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th a:link, thead th a:visited { | ||||||
|  |     color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th.sorted { | ||||||
|  |     background: #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th.sorted .text { | ||||||
|  |     padding-right: 42px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th .text span { | ||||||
|  |     padding: 8px 10px; | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th .text a { | ||||||
|  |     display: block; | ||||||
|  |     cursor: pointer; | ||||||
|  |     padding: 8px 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th .text a:focus, table thead th .text a:hover { | ||||||
|  |     background: #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th.sorted a.sortremove { | ||||||
|  |     visibility: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted:hover a.sortremove { | ||||||
|  |     visibility: visible; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions { | ||||||
|  |     display: block; | ||||||
|  |     padding: 9px 5px 0 5px; | ||||||
|  |     float: right; | ||||||
|  |     text-align: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortpriority { | ||||||
|  |     font-size: .8em; | ||||||
|  |     min-width: 12px; | ||||||
|  |     text-align: center; | ||||||
|  |     vertical-align: 3px; | ||||||
|  |     margin-left: 2px; | ||||||
|  |     margin-right: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a { | ||||||
|  |     position: relative; | ||||||
|  |     width: 14px; | ||||||
|  |     height: 14px; | ||||||
|  |     display: inline-block; | ||||||
|  |     background: url(../img/sorting-icons.svg) 0 0 no-repeat; | ||||||
|  |     background-size: 14px auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.sortremove { | ||||||
|  |     background-position: 0 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.sortremove:after { | ||||||
|  |     content: '\\'; | ||||||
|  |     position: absolute; | ||||||
|  |     top: -6px; | ||||||
|  |     left: 3px; | ||||||
|  |     font-weight: 200; | ||||||
|  |     font-size: 18px; | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.sortremove:focus:after, | ||||||
|  | table thead th.sorted .sortoptions a.sortremove:hover:after { | ||||||
|  |     color: #447e9b; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.sortremove:focus, | ||||||
|  | table thead th.sorted .sortoptions a.sortremove:hover { | ||||||
|  |     background-position: 0 -14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.ascending { | ||||||
|  |     background-position: 0 -28px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.ascending:focus, | ||||||
|  | table thead th.sorted .sortoptions a.ascending:hover { | ||||||
|  |     background-position: 0 -42px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.descending { | ||||||
|  |     top: 1px; | ||||||
|  |     background-position: 0 -56px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions a.descending:focus, | ||||||
|  | table thead th.sorted .sortoptions a.descending:hover { | ||||||
|  |     background-position: 0 -70px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* FORM DEFAULTS */ | ||||||
|  |  | ||||||
|  | input, textarea, select, .form-row p, form .button { | ||||||
|  |     margin: 2px 0; | ||||||
|  |     padding: 2px 3px; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     font-family: "Roboto", "Lucida Grande", Verdana, Arial, sans-serif; | ||||||
|  |     font-weight: normal; | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  | .form-row div.help { | ||||||
|  |     padding: 2px 3px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | textarea { | ||||||
|  |     vertical-align: top; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | input[type=text], input[type=password], input[type=email], input[type=url], | ||||||
|  | input[type=number], input[type=tel], textarea, select, .vTextField { | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     padding: 5px 6px; | ||||||
|  |     margin-top: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | input[type=text]:focus, input[type=password]:focus, input[type=email]:focus, | ||||||
|  | input[type=url]:focus, input[type=number]:focus, input[type=tel]:focus, | ||||||
|  | textarea:focus, select:focus, .vTextField:focus { | ||||||
|  |     border-color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | select { | ||||||
|  |     height: 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | select[multiple] { | ||||||
|  |     /* Allow HTML size attribute to override the height in the rule above. */ | ||||||
|  |     height: auto; | ||||||
|  |     min-height: 150px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* FORM BUTTONS */ | ||||||
|  |  | ||||||
|  | .button, input[type=submit], input[type=button], .submit-row input, a.button { | ||||||
|  |     background: #79aec8; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     border: none; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     color: #fff; | ||||||
|  |     cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.button { | ||||||
|  |     padding: 4px 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button:active, input[type=submit]:active, input[type=button]:active, | ||||||
|  | .button:focus, input[type=submit]:focus, input[type=button]:focus, | ||||||
|  | .button:hover, input[type=submit]:hover, input[type=button]:hover { | ||||||
|  |     background: #609ab6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button[disabled], input[type=submit][disabled], input[type=button][disabled] { | ||||||
|  |     opacity: 0.4; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button.default, input[type=submit].default, .submit-row input.default { | ||||||
|  |     float: right; | ||||||
|  |     border: none; | ||||||
|  |     font-weight: 400; | ||||||
|  |     background: #417690; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button.default:active, input[type=submit].default:active, | ||||||
|  | .button.default:focus, input[type=submit].default:focus, | ||||||
|  | .button.default:hover, input[type=submit].default:hover { | ||||||
|  |     background: #205067; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button[disabled].default, | ||||||
|  | input[type=submit][disabled].default, | ||||||
|  | input[type=button][disabled].default { | ||||||
|  |     opacity: 0.4; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | /* MODULES */ | ||||||
|  |  | ||||||
|  | .module { | ||||||
|  |     border: none; | ||||||
|  |     margin-bottom: 30px; | ||||||
|  |     background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module p, .module ul, .module h3, .module h4, .module dl, .module pre { | ||||||
|  |     padding-left: 10px; | ||||||
|  |     padding-right: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module blockquote { | ||||||
|  |     margin-left: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module ul, .module ol { | ||||||
|  |     margin-left: 1.5em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module h3 { | ||||||
|  |     margin-top: .6em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module h2, .module caption, .inline-group h2 { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 8px; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 13px; | ||||||
|  |     text-align: left; | ||||||
|  |     background: #79aec8; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module caption, | ||||||
|  | .inline-group h2 { | ||||||
|  |     font-size: 12px; | ||||||
|  |     letter-spacing: 0.5px; | ||||||
|  |     text-transform: uppercase; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module table { | ||||||
|  |     border-collapse: collapse; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* MESSAGES & ERRORS */ | ||||||
|  |  | ||||||
|  | ul.messagelist { | ||||||
|  |     padding: 0; | ||||||
|  |     margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.messagelist li { | ||||||
|  |     display: block; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 13px; | ||||||
|  |     padding: 10px 10px 10px 65px; | ||||||
|  |     margin: 0 0 10px 0; | ||||||
|  |     background: #dfd url(../img/icon-yes.svg) 40px 12px no-repeat; | ||||||
|  |     background-size: 16px auto; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.messagelist li.warning { | ||||||
|  |     background: #ffc url(../img/icon-alert.svg) 40px 14px no-repeat; | ||||||
|  |     background-size: 14px auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.messagelist li.error { | ||||||
|  |     background: #ffefef url(../img/icon-no.svg) 40px 12px no-repeat; | ||||||
|  |     background-size: 16px auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .errornote { | ||||||
|  |     font-size: 14px; | ||||||
|  |     font-weight: 700; | ||||||
|  |     display: block; | ||||||
|  |     padding: 10px 12px; | ||||||
|  |     margin: 0 0 10px 0; | ||||||
|  |     color: #ba2121; | ||||||
|  |     border: 1px solid #ba2121; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     background-color: #fff; | ||||||
|  |     background-position: 5px 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.errorlist { | ||||||
|  |     margin: 0 0 4px; | ||||||
|  |     padding: 0; | ||||||
|  |     color: #ba2121; | ||||||
|  |     background: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.errorlist li { | ||||||
|  |     font-size: 13px; | ||||||
|  |     display: block; | ||||||
|  |     margin-bottom: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.errorlist li:first-child { | ||||||
|  |     margin-top: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.errorlist li a { | ||||||
|  |     color: inherit; | ||||||
|  |     text-decoration: underline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | td ul.errorlist { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | td ul.errorlist li { | ||||||
|  |     margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-row.errors { | ||||||
|  |     margin: 0; | ||||||
|  |     border: none; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  |     background: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-row.errors ul.errorlist li { | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .errors input, .errors select, .errors textarea { | ||||||
|  |     border: 1px solid #ba2121; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.system-message { | ||||||
|  |     background: #ffc; | ||||||
|  |     margin: 10px; | ||||||
|  |     padding: 6px 8px; | ||||||
|  |     font-size: .8em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.system-message p.system-message-title { | ||||||
|  |     padding: 4px 5px 4px 25px; | ||||||
|  |     margin: 0; | ||||||
|  |     color: #c11; | ||||||
|  |     background: #ffefef url(../img/icon-no.svg) 5px 5px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .description { | ||||||
|  |     font-size: 12px; | ||||||
|  |     padding: 5px 0 0 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* BREADCRUMBS */ | ||||||
|  |  | ||||||
|  | div.breadcrumbs { | ||||||
|  |     background: #79aec8; | ||||||
|  |     padding: 10px 40px; | ||||||
|  |     border: none; | ||||||
|  |     font-size: 14px; | ||||||
|  |     color: #c4dce8; | ||||||
|  |     text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.breadcrumbs a { | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.breadcrumbs a:focus, div.breadcrumbs a:hover { | ||||||
|  |     color: #c4dce8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* ACTION ICONS */ | ||||||
|  |  | ||||||
|  | .viewlink, .inlineviewlink { | ||||||
|  |     padding-left: 16px; | ||||||
|  |     background: url(../img/icon-viewlink.svg) 0 1px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .addlink { | ||||||
|  |     padding-left: 16px; | ||||||
|  |     background: url(../img/icon-addlink.svg) 0 1px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .changelink, .inlinechangelink { | ||||||
|  |     padding-left: 16px; | ||||||
|  |     background: url(../img/icon-changelink.svg) 0 1px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .deletelink { | ||||||
|  |     padding-left: 16px; | ||||||
|  |     background: url(../img/icon-deletelink.svg) 0 1px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.deletelink:link, a.deletelink:visited { | ||||||
|  |     color: #CC3434; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.deletelink:focus, a.deletelink:hover { | ||||||
|  |     color: #993333; | ||||||
|  |     text-decoration: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* OBJECT TOOLS */ | ||||||
|  |  | ||||||
|  | .object-tools { | ||||||
|  |     font-size: 10px; | ||||||
|  |     font-weight: bold; | ||||||
|  |     padding-left: 0; | ||||||
|  |     float: right; | ||||||
|  |     position: relative; | ||||||
|  |     margin-top: -48px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-row .object-tools { | ||||||
|  |     margin-top: 5px; | ||||||
|  |     margin-bottom: 5px; | ||||||
|  |     float: none; | ||||||
|  |     height: 2em; | ||||||
|  |     padding-left: 3.5em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools li { | ||||||
|  |     display: block; | ||||||
|  |     float: left; | ||||||
|  |     margin-left: 5px; | ||||||
|  |     height: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools a { | ||||||
|  |     border-radius: 15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools a:link, .object-tools a:visited { | ||||||
|  |     display: block; | ||||||
|  |     float: left; | ||||||
|  |     padding: 3px 12px; | ||||||
|  |     background: #999; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 11px; | ||||||
|  |     text-transform: uppercase; | ||||||
|  |     letter-spacing: 0.5px; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools a:focus, .object-tools a:hover { | ||||||
|  |     background-color: #417690; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools a:focus{ | ||||||
|  |     text-decoration: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools a.viewsitelink, .object-tools a.golink,.object-tools a.addlink { | ||||||
|  |     background-repeat: no-repeat; | ||||||
|  |     background-position: right 7px center; | ||||||
|  |     padding-right: 26px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools a.viewsitelink, .object-tools a.golink { | ||||||
|  |     background-image: url(../img/tooltag-arrowright.svg); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools a.addlink { | ||||||
|  |     background-image: url(../img/tooltag-add.svg); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* OBJECT HISTORY */ | ||||||
|  |  | ||||||
|  | table#change-history { | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table#change-history tbody th { | ||||||
|  |     width: 16em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* PAGE STRUCTURE */ | ||||||
|  |  | ||||||
|  | #container { | ||||||
|  |     position: relative; | ||||||
|  |     width: 100%; | ||||||
|  |     min-width: 980px; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content { | ||||||
|  |     padding: 20px 40px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .dashboard #content { | ||||||
|  |     width: 600px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-main { | ||||||
|  |     float: left; | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related { | ||||||
|  |     float: right; | ||||||
|  |     width: 260px; | ||||||
|  |     position: relative; | ||||||
|  |     margin-right: -300px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #footer { | ||||||
|  |     clear: both; | ||||||
|  |     padding: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* COLUMN TYPES */ | ||||||
|  |  | ||||||
|  | .colMS { | ||||||
|  |     margin-right: 300px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .colSM { | ||||||
|  |     margin-left: 300px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .colSM #content-related { | ||||||
|  |     float: left; | ||||||
|  |     margin-right: 0; | ||||||
|  |     margin-left: -300px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .colSM #content-main { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .popup .colM { | ||||||
|  |     width: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* HEADER */ | ||||||
|  |  | ||||||
|  | #header { | ||||||
|  |     width: auto; | ||||||
|  |     height: auto; | ||||||
|  |     display: flex; | ||||||
|  |     justify-content: space-between; | ||||||
|  |     align-items: center; | ||||||
|  |     padding: 10px 40px; | ||||||
|  |     background: #417690; | ||||||
|  |     color: #ffc; | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #header a:link, #header a:visited { | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #header a:focus , #header a:hover { | ||||||
|  |     text-decoration: underline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #branding { | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #branding h1 { | ||||||
|  |     padding: 0; | ||||||
|  |     margin: 0 20px 0 0; | ||||||
|  |     font-weight: 300; | ||||||
|  |     font-size: 24px; | ||||||
|  |     color: #f5dd5d; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #branding h1, #branding h1 a:link, #branding h1 a:visited { | ||||||
|  |     color: #f5dd5d; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #branding h2 { | ||||||
|  |     padding: 0 10px; | ||||||
|  |     font-size: 14px; | ||||||
|  |     margin: -8px 0 8px 0; | ||||||
|  |     font-weight: normal; | ||||||
|  |     color: #ffc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #branding a:hover { | ||||||
|  |     text-decoration: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #user-tools { | ||||||
|  |     float: right; | ||||||
|  |     padding: 0; | ||||||
|  |     margin: 0 0 0 20px; | ||||||
|  |     font-weight: 300; | ||||||
|  |     font-size: 11px; | ||||||
|  |     letter-spacing: 0.5px; | ||||||
|  |     text-transform: uppercase; | ||||||
|  |     text-align: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #user-tools a { | ||||||
|  |     border-bottom: 1px solid rgba(255, 255, 255, 0.25); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #user-tools a:focus, #user-tools a:hover { | ||||||
|  |     text-decoration: none; | ||||||
|  |     border-bottom-color: #79aec8; | ||||||
|  |     color: #79aec8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* SIDEBAR */ | ||||||
|  |  | ||||||
|  | #content-related { | ||||||
|  |     background: #f8f8f8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related .module { | ||||||
|  |     background: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related h3 { | ||||||
|  |     font-size: 14px; | ||||||
|  |     color: #666; | ||||||
|  |     padding: 0 16px; | ||||||
|  |     margin: 0 0 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related h4 { | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related p { | ||||||
|  |     padding-left: 16px; | ||||||
|  |     padding-right: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related .actionlist { | ||||||
|  |     padding: 0; | ||||||
|  |     margin: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related .actionlist li { | ||||||
|  |     line-height: 1.2; | ||||||
|  |     margin-bottom: 10px; | ||||||
|  |     padding-left: 18px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related .module h2 { | ||||||
|  |     background: none; | ||||||
|  |     padding: 16px; | ||||||
|  |     margin-bottom: 16px; | ||||||
|  |     border-bottom: 1px solid #eaeaea; | ||||||
|  |     font-size: 18px; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-confirmation form input[type="submit"] { | ||||||
|  |     background: #ba2121; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-confirmation form input[type="submit"]:active, | ||||||
|  | .delete-confirmation form input[type="submit"]:focus, | ||||||
|  | .delete-confirmation form input[type="submit"]:hover { | ||||||
|  |     background: #a41515; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-confirmation form .cancel-link { | ||||||
|  |     display: inline-block; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     height: 15px; | ||||||
|  |     line-height: 15px; | ||||||
|  |     background: #ddd; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     color: #333; | ||||||
|  |     margin: 0 0 0 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .delete-confirmation form .cancel-link:active, | ||||||
|  | .delete-confirmation form .cancel-link:focus, | ||||||
|  | .delete-confirmation form .cancel-link:hover { | ||||||
|  |     background: #ccc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* POPUP */ | ||||||
|  | .popup #content { | ||||||
|  |     padding: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .popup #container { | ||||||
|  |     min-width: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .popup #header { | ||||||
|  |     padding: 10px 20px; | ||||||
|  | } | ||||||
							
								
								
									
										345
									
								
								CalibreWebCompanion/static/admin/css/changelists.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,345 @@ | |||||||
|  | /* CHANGELISTS */ | ||||||
|  |  | ||||||
|  | #changelist { | ||||||
|  |     position: relative; | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist table { | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list .hiddenfields { display:none; } | ||||||
|  |  | ||||||
|  | .change-list .filtered table { | ||||||
|  |     border-right: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list .filtered { | ||||||
|  |     min-height: 400px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list .filtered .results, .change-list .filtered .paginator, | ||||||
|  | .filtered #toolbar, .filtered div.xfull { | ||||||
|  |     margin-right: 280px; | ||||||
|  |     width: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list .filtered table tbody th { | ||||||
|  |     padding-right: 1em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-form .results { | ||||||
|  |   overflow-x: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .toplinks { | ||||||
|  |     border-bottom: 1px solid #ddd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .paginator { | ||||||
|  |     color: #666; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  |     background: #fff; | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* CHANGELIST TABLES */ | ||||||
|  |  | ||||||
|  | #changelist table thead th { | ||||||
|  |     padding: 0; | ||||||
|  |     white-space: nowrap; | ||||||
|  |     vertical-align: middle; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist table thead th.action-checkbox-column { | ||||||
|  |     width: 1.5em; | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist table tbody td.action-checkbox { | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist table tfoot { | ||||||
|  |     color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* TOOLBAR */ | ||||||
|  |  | ||||||
|  | #changelist #toolbar { | ||||||
|  |     padding: 8px 10px; | ||||||
|  |     margin-bottom: 15px; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist #toolbar form input { | ||||||
|  |     border-radius: 4px; | ||||||
|  |     font-size: 14px; | ||||||
|  |     padding: 5px; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist #toolbar form #searchbar { | ||||||
|  |     height: 19px; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     padding: 2px 5px; | ||||||
|  |     margin: 0; | ||||||
|  |     vertical-align: top; | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist #toolbar form #searchbar:focus { | ||||||
|  |     border-color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist #toolbar form input[type="submit"] { | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     font-size: 13px; | ||||||
|  |     padding: 4px 8px; | ||||||
|  |     margin: 0; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     background: #fff; | ||||||
|  |     box-shadow: 0 -15px 20px -10px rgba(0, 0, 0, 0.15) inset; | ||||||
|  |     cursor: pointer; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist #toolbar form input[type="submit"]:focus, | ||||||
|  | #changelist #toolbar form input[type="submit"]:hover { | ||||||
|  |     border-color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist #changelist-search img { | ||||||
|  |     vertical-align: middle; | ||||||
|  |     margin-right: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* FILTER COLUMN */ | ||||||
|  |  | ||||||
|  | #changelist-filter { | ||||||
|  |     position: absolute; | ||||||
|  |     top: 0; | ||||||
|  |     right: 0; | ||||||
|  |     z-index: 1000; | ||||||
|  |     width: 240px; | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     border-left: none; | ||||||
|  |     margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter h2 { | ||||||
|  |     font-size: 14px; | ||||||
|  |     text-transform: uppercase; | ||||||
|  |     letter-spacing: 0.5px; | ||||||
|  |     padding: 5px 15px; | ||||||
|  |     margin-bottom: 12px; | ||||||
|  |     border-bottom: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter h3 { | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 14px; | ||||||
|  |     padding: 0 15px; | ||||||
|  |     margin-bottom: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter ul { | ||||||
|  |     margin: 5px 0; | ||||||
|  |     padding: 0 15px 15px; | ||||||
|  |     border-bottom: 1px solid #eaeaea; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter ul:last-child { | ||||||
|  |     border-bottom: none; | ||||||
|  |     padding-bottom: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter li { | ||||||
|  |     list-style-type: none; | ||||||
|  |     margin-left: 0; | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter a { | ||||||
|  |     display: block; | ||||||
|  |     color: #999; | ||||||
|  |     text-overflow: ellipsis; | ||||||
|  |     overflow-x: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter li.selected { | ||||||
|  |     border-left: 5px solid #eaeaea; | ||||||
|  |     padding-left: 10px; | ||||||
|  |     margin-left: -15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter li.selected a { | ||||||
|  |     color: #5b80b2; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter a:focus, #changelist-filter a:hover, | ||||||
|  | #changelist-filter li.selected a:focus, | ||||||
|  | #changelist-filter li.selected a:hover { | ||||||
|  |     color: #036; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* DATE DRILLDOWN */ | ||||||
|  |  | ||||||
|  | .change-list ul.toplinks { | ||||||
|  |     display: block; | ||||||
|  |     float: left; | ||||||
|  |     padding: 0; | ||||||
|  |     margin: 0; | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list ul.toplinks li { | ||||||
|  |     padding: 3px 6px; | ||||||
|  |     font-weight: bold; | ||||||
|  |     list-style-type: none; | ||||||
|  |     display: inline-block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list ul.toplinks .date-back a { | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list ul.toplinks .date-back a:focus, | ||||||
|  | .change-list ul.toplinks .date-back a:hover { | ||||||
|  |     color: #036; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* PAGINATOR */ | ||||||
|  |  | ||||||
|  | .paginator { | ||||||
|  |     font-size: 13px; | ||||||
|  |     padding-top: 10px; | ||||||
|  |     padding-bottom: 10px; | ||||||
|  |     line-height: 22px; | ||||||
|  |     margin: 0; | ||||||
|  |     border-top: 1px solid #ddd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .paginator a:link, .paginator a:visited { | ||||||
|  |     padding: 2px 6px; | ||||||
|  |     background: #79aec8; | ||||||
|  |     text-decoration: none; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .paginator a.showall { | ||||||
|  |     padding: 0; | ||||||
|  |     border: none; | ||||||
|  |     background: none; | ||||||
|  |     color: #5b80b2; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .paginator a.showall:focus, .paginator a.showall:hover { | ||||||
|  |     background: none; | ||||||
|  |     color: #036; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .paginator .end { | ||||||
|  |     margin-right: 6px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .paginator .this-page { | ||||||
|  |     padding: 2px 6px; | ||||||
|  |     font-weight: bold; | ||||||
|  |     font-size: 13px; | ||||||
|  |     vertical-align: top; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .paginator a:focus, .paginator a:hover { | ||||||
|  |     color: white; | ||||||
|  |     background: #036; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* ACTIONS */ | ||||||
|  |  | ||||||
|  | .filtered .actions { | ||||||
|  |     margin-right: 280px; | ||||||
|  |     border-right: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist table input { | ||||||
|  |     margin: 0; | ||||||
|  |     vertical-align: baseline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist table tbody tr.selected { | ||||||
|  |     background-color: #FFFFCC; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions { | ||||||
|  |     padding: 10px; | ||||||
|  |     background: #fff; | ||||||
|  |     border-top: none; | ||||||
|  |     border-bottom: none; | ||||||
|  |     line-height: 24px; | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions.selected { | ||||||
|  |     background: #fffccf; | ||||||
|  |     border-top: 1px solid #fffee8; | ||||||
|  |     border-bottom: 1px solid #edecd6; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions span.all, | ||||||
|  | #changelist .actions span.action-counter, | ||||||
|  | #changelist .actions span.clear, | ||||||
|  | #changelist .actions span.question { | ||||||
|  |     font-size: 13px; | ||||||
|  |     margin: 0 0.5em; | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions:last-child { | ||||||
|  |     border-bottom: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions select { | ||||||
|  |     vertical-align: top; | ||||||
|  |     height: 24px; | ||||||
|  |     background: none; | ||||||
|  |     color: #000; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     font-size: 14px; | ||||||
|  |     padding: 0 0 0 4px; | ||||||
|  |     margin: 0; | ||||||
|  |     margin-left: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions select:focus { | ||||||
|  |     border-color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions label { | ||||||
|  |     display: inline-block; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions .button { | ||||||
|  |     font-size: 13px; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     background: #fff; | ||||||
|  |     box-shadow: 0 -15px 20px -10px rgba(0, 0, 0, 0.15) inset; | ||||||
|  |     cursor: pointer; | ||||||
|  |     height: 24px; | ||||||
|  |     line-height: 1; | ||||||
|  |     padding: 4px 8px; | ||||||
|  |     margin: 0; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist .actions .button:focus, #changelist .actions .button:hover { | ||||||
|  |     border-color: #999; | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								CalibreWebCompanion/static/admin/css/dashboard.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | |||||||
|  | /* DASHBOARD */ | ||||||
|  |  | ||||||
|  | .dashboard .module table th { | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .dashboard .module table td { | ||||||
|  |     white-space: nowrap; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .dashboard .module table td a { | ||||||
|  |     display: block; | ||||||
|  |     padding-right: .6em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* RECENT ACTIONS MODULE */ | ||||||
|  |  | ||||||
|  | .module ul.actionlist { | ||||||
|  |     margin-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.actionlist li { | ||||||
|  |     list-style-type: none; | ||||||
|  |     overflow: hidden; | ||||||
|  |     text-overflow: ellipsis; | ||||||
|  |     -o-text-overflow: ellipsis; | ||||||
|  | } | ||||||
							
								
								
									
										20
									
								
								CalibreWebCompanion/static/admin/css/fonts.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | |||||||
|  | @font-face { | ||||||
|  |     font-family: 'Roboto'; | ||||||
|  |     src: url('../fonts/Roboto-Bold-webfont.woff'); | ||||||
|  |     font-weight: 700; | ||||||
|  |     font-style: normal; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |     font-family: 'Roboto'; | ||||||
|  |     src: url('../fonts/Roboto-Regular-webfont.woff'); | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-style: normal; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @font-face { | ||||||
|  |     font-family: 'Roboto'; | ||||||
|  |     src: url('../fonts/Roboto-Light-webfont.woff'); | ||||||
|  |     font-weight: 300; | ||||||
|  |     font-style: normal; | ||||||
|  | } | ||||||
							
								
								
									
										532
									
								
								CalibreWebCompanion/static/admin/css/forms.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,532 @@ | |||||||
|  | @import url('widgets.css'); | ||||||
|  |  | ||||||
|  | /* FORM ROWS */ | ||||||
|  |  | ||||||
|  | .form-row { | ||||||
|  |     overflow: hidden; | ||||||
|  |     padding: 10px; | ||||||
|  |     font-size: 13px; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-row img, .form-row input { | ||||||
|  |     vertical-align: middle; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .form-row label input[type="checkbox"] { | ||||||
|  |     margin-top: 0; | ||||||
|  |     vertical-align: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .form-row p { | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hidden { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* FORM LABELS */ | ||||||
|  |  | ||||||
|  | label { | ||||||
|  |     font-weight: normal; | ||||||
|  |     color: #666; | ||||||
|  |     font-size: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .required label, label.required { | ||||||
|  |     font-weight: bold; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* RADIO BUTTONS */ | ||||||
|  |  | ||||||
|  | form ul.radiolist li { | ||||||
|  |     list-style-type: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form ul.radiolist label { | ||||||
|  |     float: none; | ||||||
|  |     display: inline; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form ul.radiolist input[type="radio"] { | ||||||
|  |     margin: -2px 4px 0 0; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form ul.inline { | ||||||
|  |     margin-left: 0; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form ul.inline li { | ||||||
|  |     float: left; | ||||||
|  |     padding-right: 7px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* ALIGNED FIELDSETS */ | ||||||
|  |  | ||||||
|  | .aligned label { | ||||||
|  |     display: block; | ||||||
|  |     padding: 4px 10px 0 0; | ||||||
|  |     float: left; | ||||||
|  |     width: 160px; | ||||||
|  |     word-wrap: break-word; | ||||||
|  |     line-height: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned label:not(.vCheckboxLabel):after { | ||||||
|  |     content: ''; | ||||||
|  |     display: inline-block; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     height: 26px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned label + p, .aligned label + div.help, .aligned label + div.readonly { | ||||||
|  |     padding: 6px 0; | ||||||
|  |     margin-top: 0; | ||||||
|  |     margin-bottom: 0; | ||||||
|  |     margin-left: 170px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned ul label { | ||||||
|  |     display: inline; | ||||||
|  |     float: none; | ||||||
|  |     width: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned .form-row input { | ||||||
|  |     margin-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .colMS .aligned .vLargeTextField, .colMS .aligned .vXMLLargeTextField { | ||||||
|  |     width: 350px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned ul { | ||||||
|  |     margin-left: 160px; | ||||||
|  |     padding-left: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned ul.radiolist { | ||||||
|  |     display: inline-block; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned p.help, | ||||||
|  | form .aligned div.help { | ||||||
|  |     clear: left; | ||||||
|  |     margin-top: 0; | ||||||
|  |     margin-left: 160px; | ||||||
|  |     padding-left: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned label + p.help, | ||||||
|  | form .aligned label + div.help { | ||||||
|  |     margin-left: 0; | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned p.help:last-child, | ||||||
|  | form .aligned div.help:last-child { | ||||||
|  |     margin-bottom: 0; | ||||||
|  |     padding-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned input + p.help, | ||||||
|  | form .aligned textarea + p.help, | ||||||
|  | form .aligned select + p.help, | ||||||
|  | form .aligned input + div.help, | ||||||
|  | form .aligned textarea + div.help, | ||||||
|  | form .aligned select + div.help { | ||||||
|  |     margin-left: 160px; | ||||||
|  |     padding-left: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned ul li { | ||||||
|  |     list-style: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned table p { | ||||||
|  |     margin-left: 0; | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned .vCheckboxLabel { | ||||||
|  |     float: none; | ||||||
|  |     width: auto; | ||||||
|  |     display: inline-block; | ||||||
|  |     vertical-align: -3px; | ||||||
|  |     padding: 0 0 5px 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned .vCheckboxLabel + p.help, | ||||||
|  | .aligned .vCheckboxLabel + div.help { | ||||||
|  |     margin-top: -4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .colM .aligned .vLargeTextField, .colM .aligned .vXMLLargeTextField { | ||||||
|  |     width: 610px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .checkbox-row p.help, | ||||||
|  | .checkbox-row div.help { | ||||||
|  |     margin-left: 0; | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset .fieldBox { | ||||||
|  |     float: left; | ||||||
|  |     margin-right: 20px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* WIDE FIELDSETS */ | ||||||
|  |  | ||||||
|  | .wide label { | ||||||
|  |     width: 200px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .wide p, | ||||||
|  | form .wide input + p.help, | ||||||
|  | form .wide input + div.help { | ||||||
|  |     margin-left: 200px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .wide p.help, | ||||||
|  | form .wide div.help { | ||||||
|  |     padding-left: 38px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form div.help ul { | ||||||
|  |     padding-left: 0; | ||||||
|  |     margin-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .colM fieldset.wide .vLargeTextField, .colM fieldset.wide .vXMLLargeTextField { | ||||||
|  |     width: 450px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* COLLAPSED FIELDSETS */ | ||||||
|  |  | ||||||
|  | fieldset.collapsed * { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset.collapsed h2, fieldset.collapsed { | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset.collapsed { | ||||||
|  |     border: 1px solid #eee; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset.collapsed h2 { | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset .collapse-toggle { | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset.collapsed .collapse-toggle { | ||||||
|  |     background: transparent; | ||||||
|  |     display: inline; | ||||||
|  |     color: #447e9b; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* MONOSPACE TEXTAREAS */ | ||||||
|  |  | ||||||
|  | fieldset.monospace textarea { | ||||||
|  |     font-family: "Bitstream Vera Sans Mono", Monaco, "Courier New", Courier, monospace; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* SUBMIT ROW */ | ||||||
|  |  | ||||||
|  | .submit-row { | ||||||
|  |     padding: 12px 14px; | ||||||
|  |     margin: 0 0 20px; | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     border: 1px solid #eee; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     text-align: right; | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | body.popup .submit-row { | ||||||
|  |     overflow: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row input { | ||||||
|  |     height: 35px; | ||||||
|  |     line-height: 15px; | ||||||
|  |     margin: 0 0 0 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row input.default { | ||||||
|  |     margin: 0 0 0 8px; | ||||||
|  |     text-transform: uppercase; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row p { | ||||||
|  |     margin: 0.3em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row p.deletelink-box { | ||||||
|  |     float: left; | ||||||
|  |     margin: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row a.deletelink { | ||||||
|  |     display: block; | ||||||
|  |     background: #ba2121; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     height: 15px; | ||||||
|  |     line-height: 15px; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row a.closelink { | ||||||
|  |     display: inline-block; | ||||||
|  |     background: #bbbbbb; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     padding: 10px 15px; | ||||||
|  |     height: 15px; | ||||||
|  |     line-height: 15px; | ||||||
|  |     margin: 0 0 0 5px; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row a.deletelink:focus, | ||||||
|  | .submit-row a.deletelink:hover, | ||||||
|  | .submit-row a.deletelink:active { | ||||||
|  |     background: #a41515; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row a.closelink:focus, | ||||||
|  | .submit-row a.closelink:hover, | ||||||
|  | .submit-row a.closelink:active { | ||||||
|  |     background: #aaaaaa; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* CUSTOM FORM FIELDS */ | ||||||
|  |  | ||||||
|  | .vSelectMultipleField { | ||||||
|  |     vertical-align: top; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vCheckboxField { | ||||||
|  |     border: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vDateField, .vTimeField { | ||||||
|  |     margin-right: 2px; | ||||||
|  |     margin-bottom: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vDateField { | ||||||
|  |     min-width: 6.85em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vTimeField { | ||||||
|  |     min-width: 4.7em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vURLField { | ||||||
|  |     width: 30em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vLargeTextField, .vXMLLargeTextField { | ||||||
|  |     width: 48em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .flatpages-flatpage #id_content { | ||||||
|  |     height: 40.2em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module table .vPositiveSmallIntegerField { | ||||||
|  |     width: 2.2em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vTextField, .vUUIDField { | ||||||
|  |     width: 20em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vIntegerField { | ||||||
|  |     width: 5em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vBigIntegerField { | ||||||
|  |     width: 10em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vForeignKeyRawIdAdminField { | ||||||
|  |     width: 5em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* INLINES */ | ||||||
|  |  | ||||||
|  | .inline-group { | ||||||
|  |     padding: 0; | ||||||
|  |     margin: 0 0 30px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group thead th { | ||||||
|  |     padding: 8px 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group .aligned label { | ||||||
|  |     width: 160px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related { | ||||||
|  |     position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related h3 { | ||||||
|  |     margin: 0; | ||||||
|  |     color: #666; | ||||||
|  |     padding: 5px; | ||||||
|  |     font-size: 13px; | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related h3 span.delete { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related h3 span.delete label { | ||||||
|  |     margin-left: 2px; | ||||||
|  |     font-size: 11px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related fieldset { | ||||||
|  |     margin: 0; | ||||||
|  |     background: #fff; | ||||||
|  |     border: none; | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related fieldset.module h3 { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 2px 5px 3px 5px; | ||||||
|  |     font-size: 11px; | ||||||
|  |     text-align: left; | ||||||
|  |     font-weight: bold; | ||||||
|  |     background: #bcd; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group .tabular fieldset.module { | ||||||
|  |     border: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related.tabular fieldset.module table { | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .last-related fieldset { | ||||||
|  |     border: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group .tabular tr.has_original td { | ||||||
|  |     padding-top: 2em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group .tabular tr td.original { | ||||||
|  |     padding: 2px 0 0 0; | ||||||
|  |     width: 0; | ||||||
|  |     _position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group .tabular th.original { | ||||||
|  |     width: 0px; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group .tabular td.original p { | ||||||
|  |     position: absolute; | ||||||
|  |     left: 0; | ||||||
|  |     height: 1.1em; | ||||||
|  |     padding: 2px 9px; | ||||||
|  |     overflow: hidden; | ||||||
|  |     font-size: 9px; | ||||||
|  |     font-weight: bold; | ||||||
|  |     color: #666; | ||||||
|  |     _width: 700px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group ul.tools { | ||||||
|  |     padding: 0; | ||||||
|  |     margin: 0; | ||||||
|  |     list-style: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group ul.tools li { | ||||||
|  |     display: inline; | ||||||
|  |     padding: 0 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group div.add-row, | ||||||
|  | .inline-group .tabular tr.add-row td { | ||||||
|  |     color: #666; | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     padding: 8px 10px; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group .tabular tr.add-row td { | ||||||
|  |     padding: 8px 10px; | ||||||
|  |     border-bottom: 1px solid #eee; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-group ul.tools a.add, | ||||||
|  | .inline-group div.add-row a, | ||||||
|  | .inline-group .tabular tr.add-row td a { | ||||||
|  |     background: url(../img/icon-addlink.svg) 0 1px no-repeat; | ||||||
|  |     padding-left: 16px; | ||||||
|  |     font-size: 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .empty-form { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* RELATED FIELD ADD ONE / LOOKUP */ | ||||||
|  |  | ||||||
|  | .add-another, .related-lookup { | ||||||
|  |     margin-left: 5px; | ||||||
|  |     display: inline-block; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     background-repeat: no-repeat; | ||||||
|  |     background-size: 14px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .add-another { | ||||||
|  |     width: 16px; | ||||||
|  |     height: 16px; | ||||||
|  |     background-image: url(../img/icon-addlink.svg); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .related-lookup { | ||||||
|  |     width: 16px; | ||||||
|  |     height: 16px; | ||||||
|  |     background-image: url(../img/search.svg); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .related-widget-wrapper ul { | ||||||
|  |     display: inline-block; | ||||||
|  |     margin-left: 0; | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .clearable-file-input input { | ||||||
|  |     margin-top: 0; | ||||||
|  | } | ||||||
							
								
								
									
										79
									
								
								CalibreWebCompanion/static/admin/css/login.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,79 @@ | |||||||
|  | /* LOGIN FORM */ | ||||||
|  |  | ||||||
|  | body.login { | ||||||
|  |     background: #f8f8f8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login #header { | ||||||
|  |     height: auto; | ||||||
|  |     padding: 15px 16px; | ||||||
|  |     justify-content: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login #header h1 { | ||||||
|  |     font-size: 18px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login #header h1 a { | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login #content { | ||||||
|  |     padding: 20px 20px 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login #container { | ||||||
|  |     background: #fff; | ||||||
|  |     border: 1px solid #eaeaea; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     overflow: hidden; | ||||||
|  |     width: 28em; | ||||||
|  |     min-width: 300px; | ||||||
|  |     margin: 100px auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login #content-main { | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login .form-row { | ||||||
|  |     padding: 4px 0; | ||||||
|  |     float: left; | ||||||
|  |     width: 100%; | ||||||
|  |     border-bottom: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login .form-row label { | ||||||
|  |     padding-right: 0.5em; | ||||||
|  |     line-height: 2em; | ||||||
|  |     font-size: 1em; | ||||||
|  |     clear: both; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login .form-row #id_username, .login .form-row #id_password { | ||||||
|  |     clear: both; | ||||||
|  |     padding: 8px; | ||||||
|  |     width: 100%; | ||||||
|  |     -webkit-box-sizing: border-box; | ||||||
|  |        -moz-box-sizing: border-box; | ||||||
|  |             box-sizing: border-box; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login span.help { | ||||||
|  |     font-size: 10px; | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login .submit-row { | ||||||
|  |     clear: both; | ||||||
|  |     padding: 1em 0 0 9.4em; | ||||||
|  |     margin: 0; | ||||||
|  |     border: none; | ||||||
|  |     background: none; | ||||||
|  |     text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login .password-reset-link { | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
							
								
								
									
										996
									
								
								CalibreWebCompanion/static/admin/css/responsive.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,996 @@ | |||||||
|  | /* Tablets */ | ||||||
|  |  | ||||||
|  | input[type="submit"], button { | ||||||
|  |     -webkit-appearance: none; | ||||||
|  |     appearance: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @media (max-width: 1024px) { | ||||||
|  |     /* Basic */ | ||||||
|  |  | ||||||
|  |     html { | ||||||
|  |         -webkit-text-size-adjust: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     td, th { | ||||||
|  |         padding: 10px; | ||||||
|  |         font-size: 14px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .small { | ||||||
|  |         font-size: 12px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Layout */ | ||||||
|  |  | ||||||
|  |     #container { | ||||||
|  |         min-width: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #content { | ||||||
|  |         padding: 20px 30px 30px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     div.breadcrumbs { | ||||||
|  |         padding: 10px 30px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Header */ | ||||||
|  |  | ||||||
|  |     #header { | ||||||
|  |         flex-direction: column; | ||||||
|  |         padding: 15px 30px; | ||||||
|  |         justify-content: flex-start; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #branding h1 { | ||||||
|  |         margin: 0 0 8px; | ||||||
|  |         font-size: 20px; | ||||||
|  |         line-height: 1.2; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #user-tools { | ||||||
|  |         margin: 0; | ||||||
|  |         font-weight: 400; | ||||||
|  |         line-height: 1.85; | ||||||
|  |         text-align: left; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #user-tools a { | ||||||
|  |         display: inline-block; | ||||||
|  |         line-height: 1.4; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Dashboard */ | ||||||
|  |  | ||||||
|  |     .dashboard #content { | ||||||
|  |         width: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #content-related { | ||||||
|  |         margin-right: -290px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .colSM #content-related { | ||||||
|  |         margin-left: -290px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .colMS { | ||||||
|  |         margin-right: 290px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .colSM { | ||||||
|  |         margin-left: 290px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .dashboard .module table td a { | ||||||
|  |         padding-right: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     td .changelink, td .addlink { | ||||||
|  |         font-size: 13px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Changelist */ | ||||||
|  |  | ||||||
|  |     #changelist #toolbar { | ||||||
|  |         border: none; | ||||||
|  |         padding: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist-search > div { | ||||||
|  |         display: -webkit-flex; | ||||||
|  |         display: flex; | ||||||
|  |         -webkit-flex-wrap: wrap; | ||||||
|  |         flex-wrap: wrap; | ||||||
|  |         max-width: 480px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist-search label { | ||||||
|  |         line-height: 22px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist #toolbar form #searchbar { | ||||||
|  |         -webkit-flex: 1 0 auto; | ||||||
|  |         flex: 1 0 auto; | ||||||
|  |         width: 0; | ||||||
|  |         height: 22px; | ||||||
|  |         margin: 0 10px 0 6px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist-search .quiet { | ||||||
|  |         width: 100%; | ||||||
|  |         margin: 5px 0 0 25px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions { | ||||||
|  |         display: flex; | ||||||
|  |         flex-wrap: wrap; | ||||||
|  |         padding: 15px 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions.selected { | ||||||
|  |         border: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions label { | ||||||
|  |         display: flex; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions select { | ||||||
|  |         background: #fff; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions .button { | ||||||
|  |         min-width: 48px; | ||||||
|  |         margin: 0 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions span.all, | ||||||
|  |     #changelist .actions span.clear, | ||||||
|  |     #changelist .actions span.question, | ||||||
|  |     #changelist .actions span.action-counter { | ||||||
|  |         font-size: 11px; | ||||||
|  |         margin: 0 10px 0 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist-filter { | ||||||
|  |         width: 200px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .change-list .filtered .results, | ||||||
|  |     .change-list .filtered .paginator, | ||||||
|  |     .filtered #toolbar, | ||||||
|  |     .filtered .actions, | ||||||
|  |     .filtered div.xfull { | ||||||
|  |         margin-right: 230px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .paginator { | ||||||
|  |         border-top-color: #eee; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .results + .paginator { | ||||||
|  |         border-top: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Forms */ | ||||||
|  |  | ||||||
|  |     label { | ||||||
|  |         font-size: 14px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .form-row input[type=text], | ||||||
|  |     .form-row input[type=password], | ||||||
|  |     .form-row input[type=email], | ||||||
|  |     .form-row input[type=url], | ||||||
|  |     .form-row input[type=tel], | ||||||
|  |     .form-row input[type=number], | ||||||
|  |     .form-row textarea, | ||||||
|  |     .form-row select, | ||||||
|  |     .form-row .vTextField { | ||||||
|  |         box-sizing: border-box; | ||||||
|  |         margin: 0; | ||||||
|  |         padding: 6px 8px; | ||||||
|  |         min-height: 36px; | ||||||
|  |         font-size: 14px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .form-row select { | ||||||
|  |         height: 36px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .form-row select[multiple] { | ||||||
|  |         height: auto; | ||||||
|  |         min-height: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fieldset .fieldBox { | ||||||
|  |         float: none; | ||||||
|  |         margin: 0 -10px; | ||||||
|  |         padding: 0 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fieldset .fieldBox + .fieldBox { | ||||||
|  |         margin-top: 10px; | ||||||
|  |         padding-top: 10px; | ||||||
|  |         border-top: 1px solid #eee; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     textarea { | ||||||
|  |         max-width: 518px; | ||||||
|  |         max-height: 120px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned label { | ||||||
|  |         padding-top: 6px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .add-another, | ||||||
|  |     .aligned .related-lookup, | ||||||
|  |     .aligned .datetimeshortcuts, | ||||||
|  |     .aligned .related-lookup + strong { | ||||||
|  |         align-self: center; | ||||||
|  |         margin-left: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     form .aligned ul.radiolist { | ||||||
|  |         margin-left: 2px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Related widget */ | ||||||
|  |  | ||||||
|  |     .related-widget-wrapper { | ||||||
|  |         float: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .related-widget-wrapper-link + .selector { | ||||||
|  |         max-width: calc(100% - 30px); | ||||||
|  |         margin-right: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     select + .related-widget-wrapper-link, | ||||||
|  |     .related-widget-wrapper-link + .related-widget-wrapper-link { | ||||||
|  |         margin-left: 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Selector */ | ||||||
|  |  | ||||||
|  |     .selector { | ||||||
|  |         display: flex; | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector .selector-filter { | ||||||
|  |         display: flex; | ||||||
|  |         align-items: center; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector .selector-filter label { | ||||||
|  |         margin: 0 8px 0 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector .selector-filter input { | ||||||
|  |         width: auto; | ||||||
|  |         min-height: 0; | ||||||
|  |         flex: 1 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector-available, .selector-chosen { | ||||||
|  |         width: auto; | ||||||
|  |         flex: 1 1; | ||||||
|  |         display: flex; | ||||||
|  |         flex-direction: column; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector select { | ||||||
|  |         width: 100%; | ||||||
|  |         flex: 1 0 auto; | ||||||
|  |         margin-bottom: 5px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector ul.selector-chooser { | ||||||
|  |         width: 26px; | ||||||
|  |         height: 52px; | ||||||
|  |         padding: 2px 0; | ||||||
|  |         margin: auto 15px; | ||||||
|  |         border-radius: 20px; | ||||||
|  |         transform: translateY(-10px); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector-add, .selector-remove { | ||||||
|  |         width: 20px; | ||||||
|  |         height: 20px; | ||||||
|  |         background-size: 20px auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector-add { | ||||||
|  |         background-position: 0 -120px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector-remove { | ||||||
|  |         background-position: 0 -80px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     a.selector-chooseall, a.selector-clearall { | ||||||
|  |         align-self: center; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked { | ||||||
|  |         flex-direction: column; | ||||||
|  |         max-width: 480px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked > * { | ||||||
|  |         flex: 0 1 auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked select { | ||||||
|  |         margin-bottom: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked .selector-available, .stacked .selector-chosen { | ||||||
|  |         width: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked ul.selector-chooser { | ||||||
|  |         width: 52px; | ||||||
|  |         height: 26px; | ||||||
|  |         padding: 0 2px; | ||||||
|  |         margin: 15px auto; | ||||||
|  |         transform: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked .selector-chooser li { | ||||||
|  |         padding: 3px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked .selector-add, .stacked .selector-remove { | ||||||
|  |         background-size: 20px auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked .selector-add { | ||||||
|  |         background-position: 0 -40px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked .active.selector-add { | ||||||
|  |         background-position: 0 -60px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked .selector-remove { | ||||||
|  |         background-position: 0 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .stacked .active.selector-remove { | ||||||
|  |         background-position: 0 -20px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .help-tooltip, .selector .help-icon { | ||||||
|  |         display: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     form .form-row p.datetime { | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .datetime input { | ||||||
|  |         width: 50%; | ||||||
|  |         max-width: 120px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .datetime span { | ||||||
|  |         font-size: 13px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .datetime .timezonewarning { | ||||||
|  |         display: block; | ||||||
|  |         font-size: 11px; | ||||||
|  |         color: #999; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .datetimeshortcuts { | ||||||
|  |         color: #ccc; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .form-row .datetime input.vDateField, .form-row .datetime input.vTimeField { | ||||||
|  |         width: 75%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group { | ||||||
|  |         overflow: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Messages */ | ||||||
|  |  | ||||||
|  |     ul.messagelist li { | ||||||
|  |         padding-left: 55px; | ||||||
|  |         background-position: 30px 12px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ul.messagelist li.error { | ||||||
|  |         background-position: 30px 12px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ul.messagelist li.warning { | ||||||
|  |         background-position: 30px 14px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Login */ | ||||||
|  |  | ||||||
|  |     .login #header { | ||||||
|  |         padding: 15px 20px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login #branding h1 { | ||||||
|  |         margin: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* GIS */ | ||||||
|  |  | ||||||
|  |     div.olMap { | ||||||
|  |         max-width: calc(100vw - 30px); | ||||||
|  |         max-height: 300px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .olMap + .clear_features { | ||||||
|  |         display: block; | ||||||
|  |         margin-top: 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Docs */ | ||||||
|  |  | ||||||
|  |     .module table.xfull { | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     pre.literal-block { | ||||||
|  |         overflow: auto; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Mobile */ | ||||||
|  |  | ||||||
|  | @media (max-width: 767px) { | ||||||
|  |     /* Layout */ | ||||||
|  |  | ||||||
|  |     #header, #content, #footer { | ||||||
|  |         padding: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #footer:empty { | ||||||
|  |         padding: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     div.breadcrumbs { | ||||||
|  |         padding: 10px 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Dashboard */ | ||||||
|  |  | ||||||
|  |     .colMS, .colSM { | ||||||
|  |         margin: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #content-related, .colSM #content-related { | ||||||
|  |         width: 100%; | ||||||
|  |         margin: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #content-related .module { | ||||||
|  |         margin-bottom: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #content-related .module h2 { | ||||||
|  |         padding: 10px 15px; | ||||||
|  |         font-size: 16px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Changelist */ | ||||||
|  |  | ||||||
|  |     #changelist { | ||||||
|  |         display: flex; | ||||||
|  |         flex-direction: column; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist #toolbar { | ||||||
|  |         order: 1; | ||||||
|  |         padding: 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .xfull { | ||||||
|  |         order: 2; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist-form { | ||||||
|  |         order: 3; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist-filter { | ||||||
|  |         order: 4; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions label { | ||||||
|  |         flex: 1 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions select { | ||||||
|  |         flex: 1 0; | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist .actions span { | ||||||
|  |         flex: 1 0 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .change-list .filtered .results, .change-list .filtered .paginator, | ||||||
|  |     .filtered #toolbar, .filtered .actions, .filtered div.xfull { | ||||||
|  |         margin-right: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #changelist-filter { | ||||||
|  |         position: static; | ||||||
|  |         width: auto; | ||||||
|  |         margin-top: 30px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .object-tools { | ||||||
|  |         float: none; | ||||||
|  |         margin: 0 0 15px; | ||||||
|  |         padding: 0; | ||||||
|  |         overflow: hidden; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .object-tools li { | ||||||
|  |         height: auto; | ||||||
|  |         margin-left: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .object-tools li + li { | ||||||
|  |         margin-left: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Forms */ | ||||||
|  |  | ||||||
|  |     .form-row { | ||||||
|  |         padding: 15px 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .form-row, | ||||||
|  |     .aligned .form-row > div { | ||||||
|  |         display: flex; | ||||||
|  |         flex-wrap: wrap; | ||||||
|  |         max-width: 100vw; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .form-row > div { | ||||||
|  |         width: calc(100vw - 30px); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     textarea { | ||||||
|  |         max-width: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .vURLField { | ||||||
|  |         width: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fieldset .fieldBox + .fieldBox { | ||||||
|  |         margin-top: 15px; | ||||||
|  |         padding-top: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fieldset.collapsed .form-row { | ||||||
|  |         display: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned label { | ||||||
|  |         width: 100%; | ||||||
|  |         padding: 0 0 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned label:after { | ||||||
|  |         max-height: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .form-row input, | ||||||
|  |     .aligned .form-row select, | ||||||
|  |     .aligned .form-row textarea { | ||||||
|  |         flex: 1 1 auto; | ||||||
|  |         max-width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .checkbox-row { | ||||||
|  |         align-items: center; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .checkbox-row input { | ||||||
|  |         flex: 0 1 auto; | ||||||
|  |         margin: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .vCheckboxLabel { | ||||||
|  |         flex: 1 0; | ||||||
|  |         padding: 1px 0 0 5px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned label + p, | ||||||
|  |     .aligned label + div.help, | ||||||
|  |     .aligned label + div.readonly { | ||||||
|  |         padding: 0; | ||||||
|  |         margin-left: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned p.file-upload { | ||||||
|  |         margin-left: 0; | ||||||
|  |         font-size: 13px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     span.clearable-file-input { | ||||||
|  |         margin-left: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     span.clearable-file-input label { | ||||||
|  |         font-size: 13px; | ||||||
|  |         padding-bottom: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .aligned .timezonewarning { | ||||||
|  |         flex: 1 0 100%; | ||||||
|  |         margin-top: 5px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     form .aligned .form-row div.help { | ||||||
|  |         width: 100%; | ||||||
|  |         margin: 5px 0 0; | ||||||
|  |         padding: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     form .aligned ul { | ||||||
|  |         margin-left: 0; | ||||||
|  |         padding-left: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     form .aligned ul.radiolist { | ||||||
|  |         margin-right: 15px; | ||||||
|  |         margin-bottom: -3px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     form .aligned ul.radiolist li + li { | ||||||
|  |         margin-top: 5px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Related widget */ | ||||||
|  |  | ||||||
|  |     .related-widget-wrapper { | ||||||
|  |         width: 100%; | ||||||
|  |         display: flex; | ||||||
|  |         align-items: flex-start; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .related-widget-wrapper .selector { | ||||||
|  |         order: 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .related-widget-wrapper > a { | ||||||
|  |         order: 2; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .related-widget-wrapper .radiolist ~ a { | ||||||
|  |         align-self: flex-end; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .related-widget-wrapper > select ~ a { | ||||||
|  |         align-self: center; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     select + .related-widget-wrapper-link, | ||||||
|  |     .related-widget-wrapper-link + .related-widget-wrapper-link { | ||||||
|  |         margin-left: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Selector */ | ||||||
|  |  | ||||||
|  |     .selector { | ||||||
|  |         flex-direction: column; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector > * { | ||||||
|  |         float: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector-available, .selector-chosen { | ||||||
|  |         margin-bottom: 0; | ||||||
|  |         flex: 1 1 auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector select { | ||||||
|  |         max-height: 96px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector ul.selector-chooser { | ||||||
|  |         display: block; | ||||||
|  |         float: none; | ||||||
|  |         width: 52px; | ||||||
|  |         height: 26px; | ||||||
|  |         padding: 0 2px; | ||||||
|  |         margin: 15px auto 20px; | ||||||
|  |         transform: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector ul.selector-chooser li { | ||||||
|  |         float: left; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector-remove { | ||||||
|  |         background-position: 0 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .selector-add  { | ||||||
|  |         background-position: 0 -40px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Inlines */ | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related { | ||||||
|  |         border: 2px solid #eee; | ||||||
|  |         border-radius: 4px; | ||||||
|  |         margin-top: 15px; | ||||||
|  |         overflow: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related > * { | ||||||
|  |         box-sizing: border-box; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related + .inline-related { | ||||||
|  |         margin-top: 30px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related .module { | ||||||
|  |         padding: 0 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related .module .form-row:last-child { | ||||||
|  |         border-bottom: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related h3 { | ||||||
|  |         padding: 10px; | ||||||
|  |         border-top-width: 0; | ||||||
|  |         border-bottom-width: 2px; | ||||||
|  |         display: flex; | ||||||
|  |         flex-wrap: wrap; | ||||||
|  |         align-items: center; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related h3 .inline_label { | ||||||
|  |         margin-right: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .inline-related h3 span.delete { | ||||||
|  |         float: none; | ||||||
|  |         flex: 1 1 100%; | ||||||
|  |         margin-top: 5px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .aligned .form-row > div:not([class]) { | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] .aligned label { | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group[data-inline-type="stacked"] div.add-row { | ||||||
|  |         margin-top: 15px; | ||||||
|  |         border: 1px solid #eee; | ||||||
|  |         border-radius: 4px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group div.add-row, | ||||||
|  |     .inline-group .tabular tr.add-row td { | ||||||
|  |         padding: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .inline-group div.add-row a, | ||||||
|  |     .inline-group .tabular tr.add-row td a { | ||||||
|  |         display: block; | ||||||
|  |         padding: 8px 10px 8px 26px; | ||||||
|  |         background-position: 8px 9px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Submit row */ | ||||||
|  |  | ||||||
|  |     .submit-row { | ||||||
|  |         padding: 10px 10px 0; | ||||||
|  |         margin: 0 0 15px; | ||||||
|  |         display: flex; | ||||||
|  |         flex-direction: column; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .submit-row > * { | ||||||
|  |         width: 100%; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .submit-row input, .submit-row input.default, .submit-row a, .submit-row a.closelink { | ||||||
|  |         float: none; | ||||||
|  |         margin: 0 0 10px; | ||||||
|  |         text-align: center; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .submit-row a.closelink { | ||||||
|  |         padding: 10px 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .submit-row p.deletelink-box { | ||||||
|  |         order: 4; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Messages */ | ||||||
|  |  | ||||||
|  |     ul.messagelist li { | ||||||
|  |         padding-left: 40px; | ||||||
|  |         background-position: 15px 12px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ul.messagelist li.error { | ||||||
|  |         background-position: 15px 12px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ul.messagelist li.warning { | ||||||
|  |         background-position: 15px 14px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Paginator */ | ||||||
|  |  | ||||||
|  |     .paginator .this-page, .paginator a:link, .paginator a:visited { | ||||||
|  |         padding: 4px 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Login */ | ||||||
|  |  | ||||||
|  |     body.login { | ||||||
|  |         padding: 0 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login #container { | ||||||
|  |         width: auto; | ||||||
|  |         max-width: 480px; | ||||||
|  |         margin: 50px auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login #header, | ||||||
|  |     .login #content { | ||||||
|  |         padding: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login #content-main { | ||||||
|  |         float: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login .form-row { | ||||||
|  |         padding: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login .form-row + .form-row { | ||||||
|  |         margin-top: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login .form-row label { | ||||||
|  |         display: block; | ||||||
|  |         margin: 0 0 5px; | ||||||
|  |         padding: 0; | ||||||
|  |         line-height: 1.2; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login .submit-row { | ||||||
|  |         padding: 15px 0 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login br, .login .submit-row label { | ||||||
|  |         display: none; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .login .submit-row input { | ||||||
|  |         margin: 0; | ||||||
|  |         text-transform: uppercase; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .errornote { | ||||||
|  |         margin: 0 0 20px; | ||||||
|  |         padding: 8px 12px; | ||||||
|  |         font-size: 13px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Calendar and clock */ | ||||||
|  |  | ||||||
|  |     .calendarbox, .clockbox { | ||||||
|  |         position: fixed !important; | ||||||
|  |         top: 50% !important; | ||||||
|  |         left: 50% !important; | ||||||
|  |         transform: translate(-50%, -50%); | ||||||
|  |         margin: 0; | ||||||
|  |         border: none; | ||||||
|  |         overflow: visible; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendarbox:before, .clockbox:before { | ||||||
|  |         content: ''; | ||||||
|  |         position: fixed; | ||||||
|  |         top: 50%; | ||||||
|  |         left: 50%; | ||||||
|  |         width: 100vw; | ||||||
|  |         height: 100vh; | ||||||
|  |         background: rgba(0, 0, 0, 0.75); | ||||||
|  |         transform: translate(-50%, -50%); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendarbox > *, .clockbox > * { | ||||||
|  |         position: relative; | ||||||
|  |         z-index: 1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendarbox > div:first-child { | ||||||
|  |         z-index: 2; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendarbox .calendar, .clockbox h2 { | ||||||
|  |         border-radius: 4px 4px 0 0; | ||||||
|  |         overflow: hidden; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendarbox .calendar-cancel, .clockbox .calendar-cancel { | ||||||
|  |         border-radius: 0 0 4px 4px; | ||||||
|  |         overflow: hidden; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendar-shortcuts { | ||||||
|  |         padding: 10px 0; | ||||||
|  |         font-size: 12px; | ||||||
|  |         line-height: 12px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendar-shortcuts a { | ||||||
|  |         margin: 0 4px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .timelist a { | ||||||
|  |         background: #fff; | ||||||
|  |         padding: 4px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendar-cancel { | ||||||
|  |         padding: 8px 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .clockbox h2 { | ||||||
|  |         padding: 8px 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendar caption { | ||||||
|  |         padding: 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     .calendarbox .calendarnav-previous, .calendarbox .calendarnav-next { | ||||||
|  |         z-index: 1; | ||||||
|  |         top: 10px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* History */ | ||||||
|  |  | ||||||
|  |     table#change-history tbody th, table#change-history tbody td { | ||||||
|  |         font-size: 13px; | ||||||
|  |         word-break: break-word; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     table#change-history tbody th { | ||||||
|  |         width: auto; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Docs */ | ||||||
|  |  | ||||||
|  |     table.model tbody th, table.model tbody td { | ||||||
|  |         font-size: 13px; | ||||||
|  |         word-break: break-word; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										84
									
								
								CalibreWebCompanion/static/admin/css/responsive_rtl.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,84 @@ | |||||||
|  | /* TABLETS */ | ||||||
|  |  | ||||||
|  | @media (max-width: 1024px) { | ||||||
|  |     [dir="rtl"] .colMS { | ||||||
|  |         margin-right: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] #user-tools { | ||||||
|  |         text-align: right; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] #changelist .actions label { | ||||||
|  |         padding-left: 10px; | ||||||
|  |         padding-right: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] #changelist .actions select { | ||||||
|  |         margin-left: 0; | ||||||
|  |         margin-right: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .change-list .filtered .results, | ||||||
|  |     [dir="rtl"] .change-list .filtered .paginator, | ||||||
|  |     [dir="rtl"] .filtered #toolbar, | ||||||
|  |     [dir="rtl"] .filtered div.xfull, | ||||||
|  |     [dir="rtl"] .filtered .actions { | ||||||
|  |         margin-right: 0; | ||||||
|  |         margin-left: 230px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .inline-group ul.tools a.add, | ||||||
|  |     [dir="rtl"] .inline-group div.add-row a, | ||||||
|  |     [dir="rtl"] .inline-group .tabular tr.add-row td a { | ||||||
|  |         padding: 8px 26px 8px 10px; | ||||||
|  |         background-position: calc(100% - 8px) 9px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .related-widget-wrapper-link + .selector { | ||||||
|  |         margin-right: 0; | ||||||
|  |         margin-left: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .selector .selector-filter label { | ||||||
|  |         margin-right: 0; | ||||||
|  |         margin-left: 8px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .object-tools li { | ||||||
|  |         float: right; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .object-tools li + li { | ||||||
|  |         margin-left: 0; | ||||||
|  |         margin-right: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .dashboard .module table td a { | ||||||
|  |         padding-left: 0; | ||||||
|  |         padding-right: 16px; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* MOBILE */ | ||||||
|  |  | ||||||
|  | @media (max-width: 767px) { | ||||||
|  |     [dir="rtl"] .change-list .filtered .results, | ||||||
|  |     [dir="rtl"] .change-list .filtered .paginator, | ||||||
|  |     [dir="rtl"] .filtered #toolbar, | ||||||
|  |     [dir="rtl"] .filtered div.xfull, | ||||||
|  |     [dir="rtl"] .filtered .actions { | ||||||
|  |         margin-left: 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .aligned .add-another, | ||||||
|  |     [dir="rtl"] .aligned .related-lookup, | ||||||
|  |     [dir="rtl"] .aligned .datetimeshortcuts { | ||||||
|  |         margin-left: 0; | ||||||
|  |         margin-right: 15px; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     [dir="rtl"] .aligned ul { | ||||||
|  |         margin-right: 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										269
									
								
								CalibreWebCompanion/static/admin/css/rtl.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,269 @@ | |||||||
|  | body { | ||||||
|  |     direction: rtl; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* LOGIN */ | ||||||
|  |  | ||||||
|  | .login .form-row { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login .form-row label { | ||||||
|  |     float: right; | ||||||
|  |     padding-left: 0.5em; | ||||||
|  |     padding-right: 0; | ||||||
|  |     text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .login .submit-row { | ||||||
|  |     clear: both; | ||||||
|  |     padding: 1em 9.4em 0 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* GLOBAL */ | ||||||
|  |  | ||||||
|  | th { | ||||||
|  |     text-align: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module h2, .module caption { | ||||||
|  |     text-align: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .module ul, .module ol { | ||||||
|  |     margin-left: 0; | ||||||
|  |     margin-right: 1.5em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .viewlink, .addlink, .changelink { | ||||||
|  |     padding-left: 0; | ||||||
|  |     padding-right: 16px; | ||||||
|  |     background-position: 100% 1px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .deletelink { | ||||||
|  |     padding-left: 0; | ||||||
|  |     padding-right: 16px; | ||||||
|  |     background-position: 100% 1px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .object-tools { | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th:first-child, | ||||||
|  | tfoot td:first-child { | ||||||
|  |     border-left: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* LAYOUT */ | ||||||
|  |  | ||||||
|  | #user-tools { | ||||||
|  |     right: auto; | ||||||
|  |     left: 0; | ||||||
|  |     text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | div.breadcrumbs { | ||||||
|  |     text-align: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-main { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #content-related { | ||||||
|  |     float: left; | ||||||
|  |     margin-left: -300px; | ||||||
|  |     margin-right: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .colMS { | ||||||
|  |     margin-left: 300px; | ||||||
|  |     margin-right: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* SORTABLE TABLES */ | ||||||
|  |  | ||||||
|  | table thead th.sorted .sortoptions { | ||||||
|  |    float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | thead th.sorted .text { | ||||||
|  |     padding-right: 0; | ||||||
|  |     padding-left: 42px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* dashboard styles */ | ||||||
|  |  | ||||||
|  | .dashboard .module table td a { | ||||||
|  |     padding-left: .6em; | ||||||
|  |     padding-right: 16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* changelists styles */ | ||||||
|  |  | ||||||
|  | .change-list .filtered table { | ||||||
|  |     border-left: none; | ||||||
|  |     border-right: 0px none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter { | ||||||
|  |     right: auto; | ||||||
|  |     left: 0; | ||||||
|  |     border-left: none; | ||||||
|  |     border-right: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .change-list .filtered .results, .change-list .filtered .paginator, .filtered #toolbar, .filtered div.xfull { | ||||||
|  |     margin-right: 0; | ||||||
|  |     margin-left: 280px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist-filter li.selected { | ||||||
|  |     border-left: none; | ||||||
|  |     padding-left: 10px; | ||||||
|  |     margin-left: 0; | ||||||
|  |     border-right: 5px solid #eaeaea; | ||||||
|  |     padding-right: 10px; | ||||||
|  |     margin-right: -15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .filtered .actions { | ||||||
|  |     margin-left: 280px; | ||||||
|  |     margin-right: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #changelist table tbody td:first-child, #changelist table tbody th:first-child { | ||||||
|  |     border-right: none; | ||||||
|  |     border-left: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* FORMS */ | ||||||
|  |  | ||||||
|  | .aligned label { | ||||||
|  |     padding: 0 0 3px 1em; | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row { | ||||||
|  |     text-align: left | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row p.deletelink-box { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row input.default { | ||||||
|  |     margin-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .vDateField, .vTimeField { | ||||||
|  |     margin-left: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned .form-row input { | ||||||
|  |     margin-left: 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned p.help, form .aligned div.help { | ||||||
|  |     clear: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .aligned ul { | ||||||
|  |     margin-right: 163px; | ||||||
|  |     margin-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form ul.inline li { | ||||||
|  |     float: right; | ||||||
|  |     padding-right: 0; | ||||||
|  |     padding-left: 7px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | input[type=submit].default, .submit-row input.default { | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fieldset .fieldBox { | ||||||
|  |     float: right; | ||||||
|  |     margin-left: 20px; | ||||||
|  |     margin-right: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .errorlist li { | ||||||
|  |     background-position: 100% 12px; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .errornote { | ||||||
|  |     background-position: 100% 12px; | ||||||
|  |     padding: 10px 12px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* WIDGETS */ | ||||||
|  |  | ||||||
|  | .calendarnav-previous { | ||||||
|  |     top: 0; | ||||||
|  |     left: auto; | ||||||
|  |     right: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarnav-next { | ||||||
|  |     top: 0; | ||||||
|  |     right: auto; | ||||||
|  |     left: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar caption, .calendarbox h2 { | ||||||
|  |     text-align: center; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .selector-filter { | ||||||
|  |     text-align: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-deletelink { | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | form .form-row p.datetime { | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .related-widget-wrapper { | ||||||
|  |     float: right; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* MISC */ | ||||||
|  |  | ||||||
|  | .inline-related h2, .inline-group h2 { | ||||||
|  |     text-align: right | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related h3 span.delete { | ||||||
|  |     padding-right: 20px; | ||||||
|  |     padding-left: inherit; | ||||||
|  |     left: 10px; | ||||||
|  |     right: inherit; | ||||||
|  |     float:left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-related h3 span.delete label { | ||||||
|  |     margin-left: inherit; | ||||||
|  |     margin-right: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* IE7 specific bug fixes */ | ||||||
|  |  | ||||||
|  | div.colM { | ||||||
|  |     position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .submit-row input { | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
							
								
								
									
										21
									
								
								CalibreWebCompanion/static/admin/css/vendor/select2/LICENSE-SELECT2.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | |||||||
|  | The MIT License (MIT) | ||||||
|  |  | ||||||
|  | Copyright (c) 2012-2017 Kevin Brown, Igor Vaynberg, and Select2 contributors | ||||||
|  |  | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  | of this software and associated documentation files (the "Software"), to deal | ||||||
|  | in the Software without restriction, including without limitation the rights | ||||||
|  | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  | copies of the Software, and to permit persons to whom the Software is | ||||||
|  | furnished to do so, subject to the following conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be included in | ||||||
|  | all copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||||
|  | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||||
|  | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||||
|  | THE SOFTWARE. | ||||||
							
								
								
									
										484
									
								
								CalibreWebCompanion/static/admin/css/vendor/select2/select2.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,484 @@ | |||||||
|  | .select2-container { | ||||||
|  |   box-sizing: border-box; | ||||||
|  |   display: inline-block; | ||||||
|  |   margin: 0; | ||||||
|  |   position: relative; | ||||||
|  |   vertical-align: middle; } | ||||||
|  |   .select2-container .select2-selection--single { | ||||||
|  |     box-sizing: border-box; | ||||||
|  |     cursor: pointer; | ||||||
|  |     display: block; | ||||||
|  |     height: 28px; | ||||||
|  |     user-select: none; | ||||||
|  |     -webkit-user-select: none; } | ||||||
|  |     .select2-container .select2-selection--single .select2-selection__rendered { | ||||||
|  |       display: block; | ||||||
|  |       padding-left: 8px; | ||||||
|  |       padding-right: 20px; | ||||||
|  |       overflow: hidden; | ||||||
|  |       text-overflow: ellipsis; | ||||||
|  |       white-space: nowrap; } | ||||||
|  |     .select2-container .select2-selection--single .select2-selection__clear { | ||||||
|  |       position: relative; } | ||||||
|  |   .select2-container[dir="rtl"] .select2-selection--single .select2-selection__rendered { | ||||||
|  |     padding-right: 8px; | ||||||
|  |     padding-left: 20px; } | ||||||
|  |   .select2-container .select2-selection--multiple { | ||||||
|  |     box-sizing: border-box; | ||||||
|  |     cursor: pointer; | ||||||
|  |     display: block; | ||||||
|  |     min-height: 32px; | ||||||
|  |     user-select: none; | ||||||
|  |     -webkit-user-select: none; } | ||||||
|  |     .select2-container .select2-selection--multiple .select2-selection__rendered { | ||||||
|  |       display: inline-block; | ||||||
|  |       overflow: hidden; | ||||||
|  |       padding-left: 8px; | ||||||
|  |       text-overflow: ellipsis; | ||||||
|  |       white-space: nowrap; } | ||||||
|  |   .select2-container .select2-search--inline { | ||||||
|  |     float: left; } | ||||||
|  |     .select2-container .select2-search--inline .select2-search__field { | ||||||
|  |       box-sizing: border-box; | ||||||
|  |       border: none; | ||||||
|  |       font-size: 100%; | ||||||
|  |       margin-top: 5px; | ||||||
|  |       padding: 0; } | ||||||
|  |       .select2-container .select2-search--inline .select2-search__field::-webkit-search-cancel-button { | ||||||
|  |         -webkit-appearance: none; } | ||||||
|  |  | ||||||
|  | .select2-dropdown { | ||||||
|  |   background-color: white; | ||||||
|  |   border: 1px solid #aaa; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   box-sizing: border-box; | ||||||
|  |   display: block; | ||||||
|  |   position: absolute; | ||||||
|  |   left: -100000px; | ||||||
|  |   width: 100%; | ||||||
|  |   z-index: 1051; } | ||||||
|  |  | ||||||
|  | .select2-results { | ||||||
|  |   display: block; } | ||||||
|  |  | ||||||
|  | .select2-results__options { | ||||||
|  |   list-style: none; | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 0; } | ||||||
|  |  | ||||||
|  | .select2-results__option { | ||||||
|  |   padding: 6px; | ||||||
|  |   user-select: none; | ||||||
|  |   -webkit-user-select: none; } | ||||||
|  |   .select2-results__option[aria-selected] { | ||||||
|  |     cursor: pointer; } | ||||||
|  |  | ||||||
|  | .select2-container--open .select2-dropdown { | ||||||
|  |   left: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--open .select2-dropdown--above { | ||||||
|  |   border-bottom: none; | ||||||
|  |   border-bottom-left-radius: 0; | ||||||
|  |   border-bottom-right-radius: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--open .select2-dropdown--below { | ||||||
|  |   border-top: none; | ||||||
|  |   border-top-left-radius: 0; | ||||||
|  |   border-top-right-radius: 0; } | ||||||
|  |  | ||||||
|  | .select2-search--dropdown { | ||||||
|  |   display: block; | ||||||
|  |   padding: 4px; } | ||||||
|  |   .select2-search--dropdown .select2-search__field { | ||||||
|  |     padding: 4px; | ||||||
|  |     width: 100%; | ||||||
|  |     box-sizing: border-box; } | ||||||
|  |     .select2-search--dropdown .select2-search__field::-webkit-search-cancel-button { | ||||||
|  |       -webkit-appearance: none; } | ||||||
|  |   .select2-search--dropdown.select2-search--hide { | ||||||
|  |     display: none; } | ||||||
|  |  | ||||||
|  | .select2-close-mask { | ||||||
|  |   border: 0; | ||||||
|  |   margin: 0; | ||||||
|  |   padding: 0; | ||||||
|  |   display: block; | ||||||
|  |   position: fixed; | ||||||
|  |   left: 0; | ||||||
|  |   top: 0; | ||||||
|  |   min-height: 100%; | ||||||
|  |   min-width: 100%; | ||||||
|  |   height: auto; | ||||||
|  |   width: auto; | ||||||
|  |   opacity: 0; | ||||||
|  |   z-index: 99; | ||||||
|  |   background-color: #fff; | ||||||
|  |   filter: alpha(opacity=0); } | ||||||
|  |  | ||||||
|  | .select2-hidden-accessible { | ||||||
|  |   border: 0 !important; | ||||||
|  |   clip: rect(0 0 0 0) !important; | ||||||
|  |   -webkit-clip-path: inset(50%) !important; | ||||||
|  |   clip-path: inset(50%) !important; | ||||||
|  |   height: 1px !important; | ||||||
|  |   overflow: hidden !important; | ||||||
|  |   padding: 0 !important; | ||||||
|  |   position: absolute !important; | ||||||
|  |   width: 1px !important; | ||||||
|  |   white-space: nowrap !important; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-selection--single { | ||||||
|  |   background-color: #fff; | ||||||
|  |   border: 1px solid #aaa; | ||||||
|  |   border-radius: 4px; } | ||||||
|  |   .select2-container--default .select2-selection--single .select2-selection__rendered { | ||||||
|  |     color: #444; | ||||||
|  |     line-height: 28px; } | ||||||
|  |   .select2-container--default .select2-selection--single .select2-selection__clear { | ||||||
|  |     cursor: pointer; | ||||||
|  |     float: right; | ||||||
|  |     font-weight: bold; } | ||||||
|  |   .select2-container--default .select2-selection--single .select2-selection__placeholder { | ||||||
|  |     color: #999; } | ||||||
|  |   .select2-container--default .select2-selection--single .select2-selection__arrow { | ||||||
|  |     height: 26px; | ||||||
|  |     position: absolute; | ||||||
|  |     top: 1px; | ||||||
|  |     right: 1px; | ||||||
|  |     width: 20px; } | ||||||
|  |     .select2-container--default .select2-selection--single .select2-selection__arrow b { | ||||||
|  |       border-color: #888 transparent transparent transparent; | ||||||
|  |       border-style: solid; | ||||||
|  |       border-width: 5px 4px 0 4px; | ||||||
|  |       height: 0; | ||||||
|  |       left: 50%; | ||||||
|  |       margin-left: -4px; | ||||||
|  |       margin-top: -2px; | ||||||
|  |       position: absolute; | ||||||
|  |       top: 50%; | ||||||
|  |       width: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--default[dir="rtl"] .select2-selection--single .select2-selection__clear { | ||||||
|  |   float: left; } | ||||||
|  |  | ||||||
|  | .select2-container--default[dir="rtl"] .select2-selection--single .select2-selection__arrow { | ||||||
|  |   left: 1px; | ||||||
|  |   right: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--default.select2-container--disabled .select2-selection--single { | ||||||
|  |   background-color: #eee; | ||||||
|  |   cursor: default; } | ||||||
|  |   .select2-container--default.select2-container--disabled .select2-selection--single .select2-selection__clear { | ||||||
|  |     display: none; } | ||||||
|  |  | ||||||
|  | .select2-container--default.select2-container--open .select2-selection--single .select2-selection__arrow b { | ||||||
|  |   border-color: transparent transparent #888 transparent; | ||||||
|  |   border-width: 0 4px 5px 4px; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-selection--multiple { | ||||||
|  |   background-color: white; | ||||||
|  |   border: 1px solid #aaa; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   cursor: text; } | ||||||
|  |   .select2-container--default .select2-selection--multiple .select2-selection__rendered { | ||||||
|  |     box-sizing: border-box; | ||||||
|  |     list-style: none; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0 5px; | ||||||
|  |     width: 100%; } | ||||||
|  |     .select2-container--default .select2-selection--multiple .select2-selection__rendered li { | ||||||
|  |       list-style: none; } | ||||||
|  |   .select2-container--default .select2-selection--multiple .select2-selection__placeholder { | ||||||
|  |     color: #999; | ||||||
|  |     margin-top: 5px; | ||||||
|  |     float: left; } | ||||||
|  |   .select2-container--default .select2-selection--multiple .select2-selection__clear { | ||||||
|  |     cursor: pointer; | ||||||
|  |     float: right; | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-top: 5px; | ||||||
|  |     margin-right: 10px; } | ||||||
|  |   .select2-container--default .select2-selection--multiple .select2-selection__choice { | ||||||
|  |     background-color: #e4e4e4; | ||||||
|  |     border: 1px solid #aaa; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     cursor: default; | ||||||
|  |     float: left; | ||||||
|  |     margin-right: 5px; | ||||||
|  |     margin-top: 5px; | ||||||
|  |     padding: 0 5px; } | ||||||
|  |   .select2-container--default .select2-selection--multiple .select2-selection__choice__remove { | ||||||
|  |     color: #999; | ||||||
|  |     cursor: pointer; | ||||||
|  |     display: inline-block; | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-right: 2px; } | ||||||
|  |     .select2-container--default .select2-selection--multiple .select2-selection__choice__remove:hover { | ||||||
|  |       color: #333; } | ||||||
|  |  | ||||||
|  | .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__choice, .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__placeholder, .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-search--inline { | ||||||
|  |   float: right; } | ||||||
|  |  | ||||||
|  | .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__choice { | ||||||
|  |   margin-left: 5px; | ||||||
|  |   margin-right: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--default[dir="rtl"] .select2-selection--multiple .select2-selection__choice__remove { | ||||||
|  |   margin-left: 2px; | ||||||
|  |   margin-right: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--default.select2-container--focus .select2-selection--multiple { | ||||||
|  |   border: solid black 1px; | ||||||
|  |   outline: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--default.select2-container--disabled .select2-selection--multiple { | ||||||
|  |   background-color: #eee; | ||||||
|  |   cursor: default; } | ||||||
|  |  | ||||||
|  | .select2-container--default.select2-container--disabled .select2-selection__choice__remove { | ||||||
|  |   display: none; } | ||||||
|  |  | ||||||
|  | .select2-container--default.select2-container--open.select2-container--above .select2-selection--single, .select2-container--default.select2-container--open.select2-container--above .select2-selection--multiple { | ||||||
|  |   border-top-left-radius: 0; | ||||||
|  |   border-top-right-radius: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--default.select2-container--open.select2-container--below .select2-selection--single, .select2-container--default.select2-container--open.select2-container--below .select2-selection--multiple { | ||||||
|  |   border-bottom-left-radius: 0; | ||||||
|  |   border-bottom-right-radius: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-search--dropdown .select2-search__field { | ||||||
|  |   border: 1px solid #aaa; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-search--inline .select2-search__field { | ||||||
|  |   background: transparent; | ||||||
|  |   border: none; | ||||||
|  |   outline: 0; | ||||||
|  |   box-shadow: none; | ||||||
|  |   -webkit-appearance: textfield; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-results > .select2-results__options { | ||||||
|  |   max-height: 200px; | ||||||
|  |   overflow-y: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-results__option[role=group] { | ||||||
|  |   padding: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-results__option[aria-disabled=true] { | ||||||
|  |   color: #999; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-results__option[aria-selected=true] { | ||||||
|  |   background-color: #ddd; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-results__option .select2-results__option { | ||||||
|  |   padding-left: 1em; } | ||||||
|  |   .select2-container--default .select2-results__option .select2-results__option .select2-results__group { | ||||||
|  |     padding-left: 0; } | ||||||
|  |   .select2-container--default .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |     margin-left: -1em; | ||||||
|  |     padding-left: 2em; } | ||||||
|  |     .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |       margin-left: -2em; | ||||||
|  |       padding-left: 3em; } | ||||||
|  |       .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |         margin-left: -3em; | ||||||
|  |         padding-left: 4em; } | ||||||
|  |         .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |           margin-left: -4em; | ||||||
|  |           padding-left: 5em; } | ||||||
|  |           .select2-container--default .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option .select2-results__option { | ||||||
|  |             margin-left: -5em; | ||||||
|  |             padding-left: 6em; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-results__option--highlighted[aria-selected] { | ||||||
|  |   background-color: #5897fb; | ||||||
|  |   color: white; } | ||||||
|  |  | ||||||
|  | .select2-container--default .select2-results__group { | ||||||
|  |   cursor: default; | ||||||
|  |   display: block; | ||||||
|  |   padding: 6px; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-selection--single { | ||||||
|  |   background-color: #f7f7f7; | ||||||
|  |   border: 1px solid #aaa; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   outline: 0; | ||||||
|  |   background-image: -webkit-linear-gradient(top, white 50%, #eeeeee 100%); | ||||||
|  |   background-image: -o-linear-gradient(top, white 50%, #eeeeee 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, white 50%, #eeeeee 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#FFFFFFFF', endColorstr='#FFEEEEEE', GradientType=0); } | ||||||
|  |   .select2-container--classic .select2-selection--single:focus { | ||||||
|  |     border: 1px solid #5897fb; } | ||||||
|  |   .select2-container--classic .select2-selection--single .select2-selection__rendered { | ||||||
|  |     color: #444; | ||||||
|  |     line-height: 28px; } | ||||||
|  |   .select2-container--classic .select2-selection--single .select2-selection__clear { | ||||||
|  |     cursor: pointer; | ||||||
|  |     float: right; | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-right: 10px; } | ||||||
|  |   .select2-container--classic .select2-selection--single .select2-selection__placeholder { | ||||||
|  |     color: #999; } | ||||||
|  |   .select2-container--classic .select2-selection--single .select2-selection__arrow { | ||||||
|  |     background-color: #ddd; | ||||||
|  |     border: none; | ||||||
|  |     border-left: 1px solid #aaa; | ||||||
|  |     border-top-right-radius: 4px; | ||||||
|  |     border-bottom-right-radius: 4px; | ||||||
|  |     height: 26px; | ||||||
|  |     position: absolute; | ||||||
|  |     top: 1px; | ||||||
|  |     right: 1px; | ||||||
|  |     width: 20px; | ||||||
|  |     background-image: -webkit-linear-gradient(top, #eeeeee 50%, #cccccc 100%); | ||||||
|  |     background-image: -o-linear-gradient(top, #eeeeee 50%, #cccccc 100%); | ||||||
|  |     background-image: linear-gradient(to bottom, #eeeeee 50%, #cccccc 100%); | ||||||
|  |     background-repeat: repeat-x; | ||||||
|  |     filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#FFEEEEEE', endColorstr='#FFCCCCCC', GradientType=0); } | ||||||
|  |     .select2-container--classic .select2-selection--single .select2-selection__arrow b { | ||||||
|  |       border-color: #888 transparent transparent transparent; | ||||||
|  |       border-style: solid; | ||||||
|  |       border-width: 5px 4px 0 4px; | ||||||
|  |       height: 0; | ||||||
|  |       left: 50%; | ||||||
|  |       margin-left: -4px; | ||||||
|  |       margin-top: -2px; | ||||||
|  |       position: absolute; | ||||||
|  |       top: 50%; | ||||||
|  |       width: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--classic[dir="rtl"] .select2-selection--single .select2-selection__clear { | ||||||
|  |   float: left; } | ||||||
|  |  | ||||||
|  | .select2-container--classic[dir="rtl"] .select2-selection--single .select2-selection__arrow { | ||||||
|  |   border: none; | ||||||
|  |   border-right: 1px solid #aaa; | ||||||
|  |   border-radius: 0; | ||||||
|  |   border-top-left-radius: 4px; | ||||||
|  |   border-bottom-left-radius: 4px; | ||||||
|  |   left: 1px; | ||||||
|  |   right: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--classic.select2-container--open .select2-selection--single { | ||||||
|  |   border: 1px solid #5897fb; } | ||||||
|  |   .select2-container--classic.select2-container--open .select2-selection--single .select2-selection__arrow { | ||||||
|  |     background: transparent; | ||||||
|  |     border: none; } | ||||||
|  |     .select2-container--classic.select2-container--open .select2-selection--single .select2-selection__arrow b { | ||||||
|  |       border-color: transparent transparent #888 transparent; | ||||||
|  |       border-width: 0 4px 5px 4px; } | ||||||
|  |  | ||||||
|  | .select2-container--classic.select2-container--open.select2-container--above .select2-selection--single { | ||||||
|  |   border-top: none; | ||||||
|  |   border-top-left-radius: 0; | ||||||
|  |   border-top-right-radius: 0; | ||||||
|  |   background-image: -webkit-linear-gradient(top, white 0%, #eeeeee 50%); | ||||||
|  |   background-image: -o-linear-gradient(top, white 0%, #eeeeee 50%); | ||||||
|  |   background-image: linear-gradient(to bottom, white 0%, #eeeeee 50%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#FFFFFFFF', endColorstr='#FFEEEEEE', GradientType=0); } | ||||||
|  |  | ||||||
|  | .select2-container--classic.select2-container--open.select2-container--below .select2-selection--single { | ||||||
|  |   border-bottom: none; | ||||||
|  |   border-bottom-left-radius: 0; | ||||||
|  |   border-bottom-right-radius: 0; | ||||||
|  |   background-image: -webkit-linear-gradient(top, #eeeeee 50%, white 100%); | ||||||
|  |   background-image: -o-linear-gradient(top, #eeeeee 50%, white 100%); | ||||||
|  |   background-image: linear-gradient(to bottom, #eeeeee 50%, white 100%); | ||||||
|  |   background-repeat: repeat-x; | ||||||
|  |   filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#FFEEEEEE', endColorstr='#FFFFFFFF', GradientType=0); } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-selection--multiple { | ||||||
|  |   background-color: white; | ||||||
|  |   border: 1px solid #aaa; | ||||||
|  |   border-radius: 4px; | ||||||
|  |   cursor: text; | ||||||
|  |   outline: 0; } | ||||||
|  |   .select2-container--classic .select2-selection--multiple:focus { | ||||||
|  |     border: 1px solid #5897fb; } | ||||||
|  |   .select2-container--classic .select2-selection--multiple .select2-selection__rendered { | ||||||
|  |     list-style: none; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0 5px; } | ||||||
|  |   .select2-container--classic .select2-selection--multiple .select2-selection__clear { | ||||||
|  |     display: none; } | ||||||
|  |   .select2-container--classic .select2-selection--multiple .select2-selection__choice { | ||||||
|  |     background-color: #e4e4e4; | ||||||
|  |     border: 1px solid #aaa; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     cursor: default; | ||||||
|  |     float: left; | ||||||
|  |     margin-right: 5px; | ||||||
|  |     margin-top: 5px; | ||||||
|  |     padding: 0 5px; } | ||||||
|  |   .select2-container--classic .select2-selection--multiple .select2-selection__choice__remove { | ||||||
|  |     color: #888; | ||||||
|  |     cursor: pointer; | ||||||
|  |     display: inline-block; | ||||||
|  |     font-weight: bold; | ||||||
|  |     margin-right: 2px; } | ||||||
|  |     .select2-container--classic .select2-selection--multiple .select2-selection__choice__remove:hover { | ||||||
|  |       color: #555; } | ||||||
|  |  | ||||||
|  | .select2-container--classic[dir="rtl"] .select2-selection--multiple .select2-selection__choice { | ||||||
|  |   float: right; | ||||||
|  |   margin-left: 5px; | ||||||
|  |   margin-right: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--classic[dir="rtl"] .select2-selection--multiple .select2-selection__choice__remove { | ||||||
|  |   margin-left: 2px; | ||||||
|  |   margin-right: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--classic.select2-container--open .select2-selection--multiple { | ||||||
|  |   border: 1px solid #5897fb; } | ||||||
|  |  | ||||||
|  | .select2-container--classic.select2-container--open.select2-container--above .select2-selection--multiple { | ||||||
|  |   border-top: none; | ||||||
|  |   border-top-left-radius: 0; | ||||||
|  |   border-top-right-radius: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--classic.select2-container--open.select2-container--below .select2-selection--multiple { | ||||||
|  |   border-bottom: none; | ||||||
|  |   border-bottom-left-radius: 0; | ||||||
|  |   border-bottom-right-radius: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-search--dropdown .select2-search__field { | ||||||
|  |   border: 1px solid #aaa; | ||||||
|  |   outline: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-search--inline .select2-search__field { | ||||||
|  |   outline: 0; | ||||||
|  |   box-shadow: none; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-dropdown { | ||||||
|  |   background-color: white; | ||||||
|  |   border: 1px solid transparent; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-dropdown--above { | ||||||
|  |   border-bottom: none; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-dropdown--below { | ||||||
|  |   border-top: none; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-results > .select2-results__options { | ||||||
|  |   max-height: 200px; | ||||||
|  |   overflow-y: auto; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-results__option[role=group] { | ||||||
|  |   padding: 0; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-results__option[aria-disabled=true] { | ||||||
|  |   color: grey; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-results__option--highlighted[aria-selected] { | ||||||
|  |   background-color: #3875d7; | ||||||
|  |   color: white; } | ||||||
|  |  | ||||||
|  | .select2-container--classic .select2-results__group { | ||||||
|  |   cursor: default; | ||||||
|  |   display: block; | ||||||
|  |   padding: 6px; } | ||||||
|  |  | ||||||
|  | .select2-container--classic.select2-container--open .select2-dropdown { | ||||||
|  |   border-color: #5897fb; } | ||||||
							
								
								
									
										1
									
								
								CalibreWebCompanion/static/admin/css/vendor/select2/select2.min.css
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										564
									
								
								CalibreWebCompanion/static/admin/css/widgets.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,564 @@ | |||||||
|  | /* SELECTOR (FILTER INTERFACE) */ | ||||||
|  |  | ||||||
|  | .selector { | ||||||
|  |     width: 800px; | ||||||
|  |     float: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector select { | ||||||
|  |     width: 380px; | ||||||
|  |     height: 17.2em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-available, .selector-chosen { | ||||||
|  |     float: left; | ||||||
|  |     width: 380px; | ||||||
|  |     text-align: center; | ||||||
|  |     margin-bottom: 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-chosen select { | ||||||
|  |     border-top: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-available h2, .selector-chosen h2 { | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-radius: 4px 4px 0 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-chosen h2 { | ||||||
|  |     background: #79aec8; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .selector-available h2 { | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .selector-filter { | ||||||
|  |     background: white; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     border-width: 0 1px; | ||||||
|  |     padding: 8px; | ||||||
|  |     color: #999; | ||||||
|  |     font-size: 10px; | ||||||
|  |     margin: 0; | ||||||
|  |     text-align: left; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .selector-filter label, | ||||||
|  | .inline-group .aligned .selector .selector-filter label { | ||||||
|  |     float: left; | ||||||
|  |     margin: 7px 0 0; | ||||||
|  |     width: 18px; | ||||||
|  |     height: 18px; | ||||||
|  |     padding: 0; | ||||||
|  |     overflow: hidden; | ||||||
|  |     line-height: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .selector-available input { | ||||||
|  |     width: 320px; | ||||||
|  |     margin-left: 8px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector ul.selector-chooser { | ||||||
|  |     float: left; | ||||||
|  |     width: 22px; | ||||||
|  |     background-color: #eee; | ||||||
|  |     border-radius: 10px; | ||||||
|  |     margin: 10em 5px 0 5px; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-chooser li { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 3px; | ||||||
|  |     list-style-type: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector select { | ||||||
|  |     padding: 0 10px; | ||||||
|  |     margin: 0 0 10px; | ||||||
|  |     border-radius: 0 0 4px 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-add, .selector-remove { | ||||||
|  |     width: 16px; | ||||||
|  |     height: 16px; | ||||||
|  |     display: block; | ||||||
|  |     text-indent: -3000px; | ||||||
|  |     overflow: hidden; | ||||||
|  |     cursor: default; | ||||||
|  |     opacity: 0.3; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .active.selector-add, .active.selector-remove { | ||||||
|  |     opacity: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .active.selector-add:hover, .active.selector-remove:hover { | ||||||
|  |     cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-add { | ||||||
|  |     background: url(../img/selector-icons.svg) 0 -96px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .active.selector-add:focus, .active.selector-add:hover { | ||||||
|  |     background-position: 0 -112px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector-remove { | ||||||
|  |     background: url(../img/selector-icons.svg) 0 -64px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .active.selector-remove:focus, .active.selector-remove:hover { | ||||||
|  |     background-position: 0 -80px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.selector-chooseall, a.selector-clearall { | ||||||
|  |     display: inline-block; | ||||||
|  |     height: 16px; | ||||||
|  |     text-align: left; | ||||||
|  |     margin: 1px auto 3px; | ||||||
|  |     overflow: hidden; | ||||||
|  |     font-weight: bold; | ||||||
|  |     line-height: 16px; | ||||||
|  |     color: #666; | ||||||
|  |     text-decoration: none; | ||||||
|  |     opacity: 0.3; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.active.selector-chooseall:focus, a.active.selector-clearall:focus, | ||||||
|  | a.active.selector-chooseall:hover, a.active.selector-clearall:hover { | ||||||
|  |     color: #447e9b; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.active.selector-chooseall, a.active.selector-clearall { | ||||||
|  |     opacity: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.active.selector-chooseall:hover, a.active.selector-clearall:hover { | ||||||
|  |     cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.selector-chooseall { | ||||||
|  |     padding: 0 18px 0 0; | ||||||
|  |     background: url(../img/selector-icons.svg) right -160px no-repeat; | ||||||
|  |     cursor: default; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.active.selector-chooseall:focus, a.active.selector-chooseall:hover { | ||||||
|  |     background-position: 100% -176px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.selector-clearall { | ||||||
|  |     padding: 0 0 0 18px; | ||||||
|  |     background: url(../img/selector-icons.svg) 0 -128px no-repeat; | ||||||
|  |     cursor: default; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | a.active.selector-clearall:focus, a.active.selector-clearall:hover { | ||||||
|  |     background-position: 0 -144px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* STACKED SELECTORS */ | ||||||
|  |  | ||||||
|  | .stacked { | ||||||
|  |     float: left; | ||||||
|  |     width: 490px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked select { | ||||||
|  |     width: 480px; | ||||||
|  |     height: 10.1em; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .selector-available, .stacked .selector-chosen { | ||||||
|  |     width: 480px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .selector-available { | ||||||
|  |     margin-bottom: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .selector-available input { | ||||||
|  |     width: 422px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked ul.selector-chooser { | ||||||
|  |     height: 22px; | ||||||
|  |     width: 50px; | ||||||
|  |     margin: 0 0 10px 40%; | ||||||
|  |     background-color: #eee; | ||||||
|  |     border-radius: 10px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .selector-chooser li { | ||||||
|  |     float: left; | ||||||
|  |     padding: 3px 3px 3px 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .selector-chooseall, .stacked .selector-clearall { | ||||||
|  |     display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .selector-add { | ||||||
|  |     background: url(../img/selector-icons.svg) 0 -32px no-repeat; | ||||||
|  |     cursor: default; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .active.selector-add { | ||||||
|  |     background-position: 0 -48px; | ||||||
|  |     cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .selector-remove { | ||||||
|  |     background: url(../img/selector-icons.svg) 0 0 no-repeat; | ||||||
|  |     cursor: default; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .stacked .active.selector-remove { | ||||||
|  |     background-position: 0 -16px; | ||||||
|  |     cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .help-icon { | ||||||
|  |     background: url(../img/icon-unknown.svg) 0 0 no-repeat; | ||||||
|  |     display: inline-block; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     margin: -2px 0 0 2px; | ||||||
|  |     width: 13px; | ||||||
|  |     height: 13px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .selector-chosen .help-icon { | ||||||
|  |     background: url(../img/icon-unknown-alt.svg) 0 0 no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .selector .search-label-icon { | ||||||
|  |     background: url(../img/search.svg) 0 0 no-repeat; | ||||||
|  |     display: inline-block; | ||||||
|  |     height: 18px; | ||||||
|  |     width: 18px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* DATE AND TIME */ | ||||||
|  |  | ||||||
|  | p.datetime { | ||||||
|  |     line-height: 20px; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     color: #666; | ||||||
|  |     font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datetime span { | ||||||
|  |     white-space: nowrap; | ||||||
|  |     font-weight: normal; | ||||||
|  |     font-size: 11px; | ||||||
|  |     color: #ccc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datetime input, .form-row .datetime input.vDateField, .form-row .datetime input.vTimeField { | ||||||
|  |     margin-left: 5px; | ||||||
|  |     margin-bottom: 4px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | table p.datetime { | ||||||
|  |     font-size: 11px; | ||||||
|  |     margin-left: 0; | ||||||
|  |     padding-left: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datetimeshortcuts .clock-icon, .datetimeshortcuts .date-icon { | ||||||
|  |     position: relative; | ||||||
|  |     display: inline-block; | ||||||
|  |     vertical-align: middle; | ||||||
|  |     height: 16px; | ||||||
|  |     width: 16px; | ||||||
|  |     overflow: hidden; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datetimeshortcuts .clock-icon { | ||||||
|  |     background: url(../img/icon-clock.svg) 0 0 no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datetimeshortcuts a:focus .clock-icon, | ||||||
|  | .datetimeshortcuts a:hover .clock-icon { | ||||||
|  |     background-position: 0 -16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datetimeshortcuts .date-icon { | ||||||
|  |     background: url(../img/icon-calendar.svg) 0 0 no-repeat; | ||||||
|  |     top: -1px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .datetimeshortcuts a:focus .date-icon, | ||||||
|  | .datetimeshortcuts a:hover .date-icon { | ||||||
|  |     background-position: 0 -16px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timezonewarning { | ||||||
|  |     font-size: 11px; | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* URL */ | ||||||
|  |  | ||||||
|  | p.url { | ||||||
|  |     line-height: 20px; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     color: #666; | ||||||
|  |     font-size: 11px; | ||||||
|  |     font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .url a { | ||||||
|  |     font-weight: normal; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* FILE UPLOADS */ | ||||||
|  |  | ||||||
|  | p.file-upload { | ||||||
|  |     line-height: 20px; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     color: #666; | ||||||
|  |     font-size: 11px; | ||||||
|  |     font-weight: bold; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .aligned p.file-upload { | ||||||
|  |     margin-left: 170px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .file-upload a { | ||||||
|  |     font-weight: normal; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .file-upload .deletelink { | ||||||
|  |     margin-left: 5px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | span.clearable-file-input label { | ||||||
|  |     color: #333; | ||||||
|  |     font-size: 11px; | ||||||
|  |     display: inline; | ||||||
|  |     float: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* CALENDARS & CLOCKS */ | ||||||
|  |  | ||||||
|  | .calendarbox, .clockbox { | ||||||
|  |     margin: 5px auto; | ||||||
|  |     font-size: 12px; | ||||||
|  |     width: 19em; | ||||||
|  |     text-align: center; | ||||||
|  |     background: white; | ||||||
|  |     border: 1px solid #ddd; | ||||||
|  |     border-radius: 4px; | ||||||
|  |     box-shadow: 0 2px 4px rgba(0, 0, 0, 0.15); | ||||||
|  |     overflow: hidden; | ||||||
|  |     position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .clockbox { | ||||||
|  |     width: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar table { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  |     border-collapse: collapse; | ||||||
|  |     background: white; | ||||||
|  |     width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar caption, .calendarbox h2 { | ||||||
|  |     margin: 0; | ||||||
|  |     text-align: center; | ||||||
|  |     border-top: none; | ||||||
|  |     background: #f5dd5d; | ||||||
|  |     font-weight: 700; | ||||||
|  |     font-size: 12px; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar th { | ||||||
|  |     padding: 8px 5px; | ||||||
|  |     background: #f8f8f8; | ||||||
|  |     border-bottom: 1px solid #ddd; | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 12px; | ||||||
|  |     text-align: center; | ||||||
|  |     color: #666; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar td { | ||||||
|  |     font-weight: 400; | ||||||
|  |     font-size: 12px; | ||||||
|  |     text-align: center; | ||||||
|  |     padding: 0; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  |     border-bottom: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar td.selected a { | ||||||
|  |     background: #79aec8; | ||||||
|  |     color: #fff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar td.nonday { | ||||||
|  |     background: #f8f8f8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar td.today a { | ||||||
|  |     font-weight: 700; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar td a, .timelist a { | ||||||
|  |     display: block; | ||||||
|  |     font-weight: 400; | ||||||
|  |     padding: 6px; | ||||||
|  |     text-decoration: none; | ||||||
|  |     color: #444; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar td a:focus, .timelist a:focus, | ||||||
|  | .calendar td a:hover, .timelist a:hover { | ||||||
|  |     background: #79aec8; | ||||||
|  |     color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar td a:active, .timelist a:active { | ||||||
|  |     background: #417690; | ||||||
|  |     color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarnav { | ||||||
|  |     font-size: 10px; | ||||||
|  |     text-align: center; | ||||||
|  |     color: #ccc; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 1px 3px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarnav a:link, #calendarnav a:visited, | ||||||
|  | #calendarnav a:focus, #calendarnav a:hover { | ||||||
|  |     color: #999; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar-shortcuts { | ||||||
|  |     background: white; | ||||||
|  |     font-size: 11px; | ||||||
|  |     line-height: 11px; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  |     padding: 8px 0; | ||||||
|  |     color: #ccc; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarbox .calendarnav-previous, .calendarbox .calendarnav-next { | ||||||
|  |     display: block; | ||||||
|  |     position: absolute; | ||||||
|  |     top: 8px; | ||||||
|  |     width: 15px; | ||||||
|  |     height: 15px; | ||||||
|  |     text-indent: -9999px; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarnav-previous { | ||||||
|  |     left: 10px; | ||||||
|  |     background: url(../img/calendar-icons.svg) 0 0 no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarbox .calendarnav-previous:focus, | ||||||
|  | .calendarbox .calendarnav-previous:hover { | ||||||
|  |     background-position: 0 -15px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarnav-next { | ||||||
|  |     right: 10px; | ||||||
|  |     background: url(../img/calendar-icons.svg) 0 -30px no-repeat; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendarbox .calendarnav-next:focus, | ||||||
|  | .calendarbox .calendarnav-next:hover { | ||||||
|  |     background-position: 0 -45px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar-cancel { | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 4px 0; | ||||||
|  |     font-size: 12px; | ||||||
|  |     background: #eee; | ||||||
|  |     border-top: 1px solid #ddd; | ||||||
|  |     color: #333; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar-cancel:focus, .calendar-cancel:hover { | ||||||
|  |     background: #ddd; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .calendar-cancel a { | ||||||
|  |     color: black; | ||||||
|  |     display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ul.timelist, .timelist li { | ||||||
|  |     list-style-type: none; | ||||||
|  |     margin: 0; | ||||||
|  |     padding: 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .timelist a { | ||||||
|  |     padding: 2px; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* EDIT INLINE */ | ||||||
|  |  | ||||||
|  | .inline-deletelink { | ||||||
|  |     float: right; | ||||||
|  |     text-indent: -9999px; | ||||||
|  |     background: url(../img/inline-delete.svg) 0 0 no-repeat; | ||||||
|  |     width: 16px; | ||||||
|  |     height: 16px; | ||||||
|  |     border: 0px none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .inline-deletelink:focus, .inline-deletelink:hover { | ||||||
|  |     cursor: pointer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* RELATED WIDGET WRAPPER */ | ||||||
|  | .related-widget-wrapper { | ||||||
|  |     float: left;       /* display properly in form rows with multiple fields */ | ||||||
|  |     overflow: hidden;  /* clear floated contents */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .related-widget-wrapper-link { | ||||||
|  |     opacity: 0.3; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .related-widget-wrapper-link:link { | ||||||
|  |     opacity: .8; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .related-widget-wrapper-link:link:focus, | ||||||
|  | .related-widget-wrapper-link:link:hover { | ||||||
|  |     opacity: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | select + .related-widget-wrapper-link, | ||||||
|  | .related-widget-wrapper-link + .related-widget-wrapper-link { | ||||||
|  |     margin-left: 7px; | ||||||
|  | } | ||||||
							
								
								
									
										202
									
								
								CalibreWebCompanion/static/admin/fonts/LICENSE.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,202 @@ | |||||||
|  |  | ||||||
|  |                                  Apache License | ||||||
|  |                            Version 2.0, January 2004 | ||||||
|  |                         http://www.apache.org/licenses/ | ||||||
|  |  | ||||||
|  |    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||||
|  |  | ||||||
|  |    1. Definitions. | ||||||
|  |  | ||||||
|  |       "License" shall mean the terms and conditions for use, reproduction, | ||||||
|  |       and distribution as defined by Sections 1 through 9 of this document. | ||||||
|  |  | ||||||
|  |       "Licensor" shall mean the copyright owner or entity authorized by | ||||||
|  |       the copyright owner that is granting the License. | ||||||
|  |  | ||||||
|  |       "Legal Entity" shall mean the union of the acting entity and all | ||||||
|  |       other entities that control, are controlled by, or are under common | ||||||
|  |       control with that entity. For the purposes of this definition, | ||||||
|  |       "control" means (i) the power, direct or indirect, to cause the | ||||||
|  |       direction or management of such entity, whether by contract or | ||||||
|  |       otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||||
|  |       outstanding shares, or (iii) beneficial ownership of such entity. | ||||||
|  |  | ||||||
|  |       "You" (or "Your") shall mean an individual or Legal Entity | ||||||
|  |       exercising permissions granted by this License. | ||||||
|  |  | ||||||
|  |       "Source" form shall mean the preferred form for making modifications, | ||||||
|  |       including but not limited to software source code, documentation | ||||||
|  |       source, and configuration files. | ||||||
|  |  | ||||||
|  |       "Object" form shall mean any form resulting from mechanical | ||||||
|  |       transformation or translation of a Source form, including but | ||||||
|  |       not limited to compiled object code, generated documentation, | ||||||
|  |       and conversions to other media types. | ||||||
|  |  | ||||||
|  |       "Work" shall mean the work of authorship, whether in Source or | ||||||
|  |       Object form, made available under the License, as indicated by a | ||||||
|  |       copyright notice that is included in or attached to the work | ||||||
|  |       (an example is provided in the Appendix below). | ||||||
|  |  | ||||||
|  |       "Derivative Works" shall mean any work, whether in Source or Object | ||||||
|  |       form, that is based on (or derived from) the Work and for which the | ||||||
|  |       editorial revisions, annotations, elaborations, or other modifications | ||||||
|  |       represent, as a whole, an original work of authorship. For the purposes | ||||||
|  |       of this License, Derivative Works shall not include works that remain | ||||||
|  |       separable from, or merely link (or bind by name) to the interfaces of, | ||||||
|  |       the Work and Derivative Works thereof. | ||||||
|  |  | ||||||
|  |       "Contribution" shall mean any work of authorship, including | ||||||
|  |       the original version of the Work and any modifications or additions | ||||||
|  |       to that Work or Derivative Works thereof, that is intentionally | ||||||
|  |       submitted to Licensor for inclusion in the Work by the copyright owner | ||||||
|  |       or by an individual or Legal Entity authorized to submit on behalf of | ||||||
|  |       the copyright owner. For the purposes of this definition, "submitted" | ||||||
|  |       means any form of electronic, verbal, or written communication sent | ||||||
|  |       to the Licensor or its representatives, including but not limited to | ||||||
|  |       communication on electronic mailing lists, source code control systems, | ||||||
|  |       and issue tracking systems that are managed by, or on behalf of, the | ||||||
|  |       Licensor for the purpose of discussing and improving the Work, but | ||||||
|  |       excluding communication that is conspicuously marked or otherwise | ||||||
|  |       designated in writing by the copyright owner as "Not a Contribution." | ||||||
|  |  | ||||||
|  |       "Contributor" shall mean Licensor and any individual or Legal Entity | ||||||
|  |       on behalf of whom a Contribution has been received by Licensor and | ||||||
|  |       subsequently incorporated within the Work. | ||||||
|  |  | ||||||
|  |    2. Grant of Copyright License. Subject to the terms and conditions of | ||||||
|  |       this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |       copyright license to reproduce, prepare Derivative Works of, | ||||||
|  |       publicly display, publicly perform, sublicense, and distribute the | ||||||
|  |       Work and such Derivative Works in Source or Object form. | ||||||
|  |  | ||||||
|  |    3. Grant of Patent License. Subject to the terms and conditions of | ||||||
|  |       this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |       (except as stated in this section) patent license to make, have made, | ||||||
|  |       use, offer to sell, sell, import, and otherwise transfer the Work, | ||||||
|  |       where such license applies only to those patent claims licensable | ||||||
|  |       by such Contributor that are necessarily infringed by their | ||||||
|  |       Contribution(s) alone or by combination of their Contribution(s) | ||||||
|  |       with the Work to which such Contribution(s) was submitted. If You | ||||||
|  |       institute patent litigation against any entity (including a | ||||||
|  |       cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||||
|  |       or a Contribution incorporated within the Work constitutes direct | ||||||
|  |       or contributory patent infringement, then any patent licenses | ||||||
|  |       granted to You under this License for that Work shall terminate | ||||||
|  |       as of the date such litigation is filed. | ||||||
|  |  | ||||||
|  |    4. Redistribution. You may reproduce and distribute copies of the | ||||||
|  |       Work or Derivative Works thereof in any medium, with or without | ||||||
|  |       modifications, and in Source or Object form, provided that You | ||||||
|  |       meet the following conditions: | ||||||
|  |  | ||||||
|  |       (a) You must give any other recipients of the Work or | ||||||
|  |           Derivative Works a copy of this License; and | ||||||
|  |  | ||||||
|  |       (b) You must cause any modified files to carry prominent notices | ||||||
|  |           stating that You changed the files; and | ||||||
|  |  | ||||||
|  |       (c) You must retain, in the Source form of any Derivative Works | ||||||
|  |           that You distribute, all copyright, patent, trademark, and | ||||||
|  |           attribution notices from the Source form of the Work, | ||||||
|  |           excluding those notices that do not pertain to any part of | ||||||
|  |           the Derivative Works; and | ||||||
|  |  | ||||||
|  |       (d) If the Work includes a "NOTICE" text file as part of its | ||||||
|  |           distribution, then any Derivative Works that You distribute must | ||||||
|  |           include a readable copy of the attribution notices contained | ||||||
|  |           within such NOTICE file, excluding those notices that do not | ||||||
|  |           pertain to any part of the Derivative Works, in at least one | ||||||
|  |           of the following places: within a NOTICE text file distributed | ||||||
|  |           as part of the Derivative Works; within the Source form or | ||||||
|  |           documentation, if provided along with the Derivative Works; or, | ||||||
|  |           within a display generated by the Derivative Works, if and | ||||||
|  |           wherever such third-party notices normally appear. The contents | ||||||
|  |           of the NOTICE file are for informational purposes only and | ||||||
|  |           do not modify the License. You may add Your own attribution | ||||||
|  |           notices within Derivative Works that You distribute, alongside | ||||||
|  |           or as an addendum to the NOTICE text from the Work, provided | ||||||
|  |           that such additional attribution notices cannot be construed | ||||||
|  |           as modifying the License. | ||||||
|  |  | ||||||
|  |       You may add Your own copyright statement to Your modifications and | ||||||
|  |       may provide additional or different license terms and conditions | ||||||
|  |       for use, reproduction, or distribution of Your modifications, or | ||||||
|  |       for any such Derivative Works as a whole, provided Your use, | ||||||
|  |       reproduction, and distribution of the Work otherwise complies with | ||||||
|  |       the conditions stated in this License. | ||||||
|  |  | ||||||
|  |    5. Submission of Contributions. Unless You explicitly state otherwise, | ||||||
|  |       any Contribution intentionally submitted for inclusion in the Work | ||||||
|  |       by You to the Licensor shall be under the terms and conditions of | ||||||
|  |       this License, without any additional terms or conditions. | ||||||
|  |       Notwithstanding the above, nothing herein shall supersede or modify | ||||||
|  |       the terms of any separate license agreement you may have executed | ||||||
|  |       with Licensor regarding such Contributions. | ||||||
|  |  | ||||||
|  |    6. Trademarks. This License does not grant permission to use the trade | ||||||
|  |       names, trademarks, service marks, or product names of the Licensor, | ||||||
|  |       except as required for reasonable and customary use in describing the | ||||||
|  |       origin of the Work and reproducing the content of the NOTICE file. | ||||||
|  |  | ||||||
|  |    7. Disclaimer of Warranty. Unless required by applicable law or | ||||||
|  |       agreed to in writing, Licensor provides the Work (and each | ||||||
|  |       Contributor provides its Contributions) on an "AS IS" BASIS, | ||||||
|  |       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||||
|  |       implied, including, without limitation, any warranties or conditions | ||||||
|  |       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||||
|  |       PARTICULAR PURPOSE. You are solely responsible for determining the | ||||||
|  |       appropriateness of using or redistributing the Work and assume any | ||||||
|  |       risks associated with Your exercise of permissions under this License. | ||||||
|  |  | ||||||
|  |    8. Limitation of Liability. In no event and under no legal theory, | ||||||
|  |       whether in tort (including negligence), contract, or otherwise, | ||||||
|  |       unless required by applicable law (such as deliberate and grossly | ||||||
|  |       negligent acts) or agreed to in writing, shall any Contributor be | ||||||
|  |       liable to You for damages, including any direct, indirect, special, | ||||||
|  |       incidental, or consequential damages of any character arising as a | ||||||
|  |       result of this License or out of the use or inability to use the | ||||||
|  |       Work (including but not limited to damages for loss of goodwill, | ||||||
|  |       work stoppage, computer failure or malfunction, or any and all | ||||||
|  |       other commercial damages or losses), even if such Contributor | ||||||
|  |       has been advised of the possibility of such damages. | ||||||
|  |  | ||||||
|  |    9. Accepting Warranty or Additional Liability. While redistributing | ||||||
|  |       the Work or Derivative Works thereof, You may choose to offer, | ||||||
|  |       and charge a fee for, acceptance of support, warranty, indemnity, | ||||||
|  |       or other liability obligations and/or rights consistent with this | ||||||
|  |       License. However, in accepting such obligations, You may act only | ||||||
|  |       on Your own behalf and on Your sole responsibility, not on behalf | ||||||
|  |       of any other Contributor, and only if You agree to indemnify, | ||||||
|  |       defend, and hold each Contributor harmless for any liability | ||||||
|  |       incurred by, or claims asserted against, such Contributor by reason | ||||||
|  |       of your accepting any such warranty or additional liability. | ||||||
|  |  | ||||||
|  |    END OF TERMS AND CONDITIONS | ||||||
|  |  | ||||||
|  |    APPENDIX: How to apply the Apache License to your work. | ||||||
|  |  | ||||||
|  |       To apply the Apache License to your work, attach the following | ||||||
|  |       boilerplate notice, with the fields enclosed by brackets "[]" | ||||||
|  |       replaced with your own identifying information. (Don't include | ||||||
|  |       the brackets!)  The text should be enclosed in the appropriate | ||||||
|  |       comment syntax for the file format. We also recommend that a | ||||||
|  |       file or class name and description of purpose be included on the | ||||||
|  |       same "printed page" as the copyright notice for easier | ||||||
|  |       identification within third-party archives. | ||||||
|  |  | ||||||
|  |    Copyright [yyyy] [name of copyright owner] | ||||||
|  |  | ||||||
|  |    Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  |    you may not use this file except in compliance with the License. | ||||||
|  |    You may obtain a copy of the License at | ||||||
|  |  | ||||||
|  |        http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  |  | ||||||
|  |    Unless required by applicable law or agreed to in writing, software | ||||||
|  |    distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  |    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  |    See the License for the specific language governing permissions and | ||||||
|  |    limitations under the License. | ||||||
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/fonts/README.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | Roboto webfont source: https://www.google.com/fonts/specimen/Roboto | ||||||
|  | WOFF files extracted using https://github.com/majodev/google-webfonts-helper | ||||||
|  | Weights used in this project: Light (300), Regular (400), Bold (700) | ||||||
							
								
								
									
										
											BIN
										
									
								
								CalibreWebCompanion/static/admin/fonts/Roboto-Bold-webfont.woff
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										
											BIN
										
									
								
								CalibreWebCompanion/static/admin/fonts/Roboto-Light-webfont.woff
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										20
									
								
								CalibreWebCompanion/static/admin/img/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | |||||||
|  | The MIT License (MIT) | ||||||
|  |  | ||||||
|  | Copyright (c) 2014 Code Charm Ltd | ||||||
|  |  | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining a copy of | ||||||
|  | this software and associated documentation files (the "Software"), to deal in | ||||||
|  | the Software without restriction, including without limitation the rights to | ||||||
|  | use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of | ||||||
|  | the Software, and to permit persons to whom the Software is furnished to do so, | ||||||
|  | subject to the following conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be included in all | ||||||
|  | copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS | ||||||
|  | FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR | ||||||
|  | COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER | ||||||
|  | IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||||||
|  | CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | ||||||
							
								
								
									
										7
									
								
								CalibreWebCompanion/static/admin/img/README.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | |||||||
|  | All icons are taken from Font Awesome (http://fontawesome.io/) project. | ||||||
|  | The Font Awesome font is licensed under the SIL OFL 1.1: | ||||||
|  | - https://scripts.sil.org/OFL | ||||||
|  |  | ||||||
|  | SVG icons source: https://github.com/encharm/Font-Awesome-SVG-PNG | ||||||
|  | Font-Awesome-SVG-PNG is licensed under the MIT license (see file license | ||||||
|  | in current folder). | ||||||
							
								
								
									
										14
									
								
								CalibreWebCompanion/static/admin/img/calendar-icons.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | <svg width="15" height="60" viewBox="0 0 1792 7168" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | ||||||
|  |   <defs> | ||||||
|  |     <g id="previous"> | ||||||
|  |       <path d="M1037 1395l102-102q19-19 19-45t-19-45l-307-307 307-307q19-19 19-45t-19-45l-102-102q-19-19-45-19t-45 19l-454 454q-19 19-19 45t19 45l454 454q19 19 45 19t45-19zm627-499q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="next"> | ||||||
|  |       <path d="M845 1395l454-454q19-19 19-45t-19-45l-454-454q-19-19-45-19t-45 19l-102 102q-19 19-19 45t19 45l307 307-307 307q-19 19-19 45t19 45l102 102q19 19 45 19t45-19zm819-499q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |   </defs> | ||||||
|  |   <use xlink:href="#previous" x="0" y="0" fill="#333333" /> | ||||||
|  |   <use xlink:href="#previous" x="0" y="1792" fill="#000000" /> | ||||||
|  |   <use xlink:href="#next" x="0" y="3584" fill="#333333" /> | ||||||
|  |   <use xlink:href="#next" x="0" y="5376" fill="#000000" /> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.1 KiB | 
| @@ -0,0 +1 @@ | |||||||
|  | <svg width="24" height="22" viewBox="0 0 847 779" xmlns="http://www.w3.org/2000/svg"><g><path fill="#EBECE6" d="M120 1h607c66 0 120 54 120 120v536c0 66-54 120-120 120h-607c-66 0-120-54-120-120v-536c0-66 54-120 120-120z"/><path fill="#9E9E93" d="M120 1h607c66 0 120 54 120 120v536c0 66-54 120-120 120h-607c-66 0-120-54-120-120v-536c0-66 54-120 120-120zm607 25h-607c-26 0-50 11-67 28-17 18-28 41-28 67v536c0 27 11 50 28 68 17 17 41 27 67 27h607c26 0 49-10 67-27 17-18 28-41 28-68v-536c0-26-11-49-28-67-18-17-41-28-67-28z"/><path stroke="#A9A8A4" stroke-width="20" d="M706 295l-68 281"/><path stroke="#E47474" stroke-width="20" d="M316 648l390-353M141 435l175 213"/><path stroke="#C9C9C9" stroke-width="20" d="M319 151l-178 284M706 295l-387-144"/><g fill="#040405"><path d="M319 111c22 0 40 18 40 40s-18 40-40 40-40-18-40-40 18-40 40-40zM141 395c22 0 40 18 40 40s-18 40-40 40c-23 0-41-18-41-40s18-40 41-40zM316 608c22 0 40 18 40 40 0 23-18 41-40 41s-40-18-40-41c0-22 18-40 40-40zM706 254c22 0 40 18 40 41 0 22-18 40-40 40s-40-18-40-40c0-23 18-41 40-41zM638 536c22 0 40 18 40 40s-18 40-40 40-40-18-40-40 18-40 40-40z"/></g></g></svg> | ||||||
| After Width: | Height: | Size: 1.1 KiB | 
| @@ -0,0 +1 @@ | |||||||
|  | <svg width="24" height="22" viewBox="0 0 847 779" xmlns="http://www.w3.org/2000/svg"><g><path fill="#F1C02A" d="M120 1h607c66 0 120 54 120 120v536c0 66-54 120-120 120h-607c-66 0-120-54-120-120v-536c0-66 54-120 120-120z"/><path fill="#9E9E93" d="M120 1h607c66 0 120 54 120 120v536c0 66-54 120-120 120h-607c-66 0-120-54-120-120v-536c0-66 54-120 120-120zm607 25h-607c-26 0-50 11-67 28-17 18-28 41-28 67v536c0 27 11 50 28 68 17 17 41 27 67 27h607c26 0 49-10 67-27 17-18 28-41 28-68v-536c0-26-11-49-28-67-18-17-41-28-67-28z"/><path stroke="#A9A8A4" stroke-width="20" d="M706 295l-68 281"/><path stroke="#E47474" stroke-width="20" d="M316 648l390-353M141 435l175 213"/><path stroke="#C9A741" stroke-width="20" d="M319 151l-178 284M706 295l-387-144"/><g fill="#040405"><path d="M319 111c22 0 40 18 40 40s-18 40-40 40-40-18-40-40 18-40 40-40zM141 395c22 0 40 18 40 40s-18 40-40 40c-23 0-41-18-41-40s18-40 41-40zM316 608c22 0 40 18 40 40 0 23-18 41-40 41s-40-18-40-41c0-22 18-40 40-40zM706 254c22 0 40 18 40 41 0 22-18 40-40 40s-40-18-40-40c0-23 18-41 40-41zM638 536c22 0 40 18 40 40s-18 40-40 40-40-18-40-40 18-40 40-40z"/></g></g></svg> | ||||||
| After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-addlink.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#70bf2b" d="M1600 796v192q0 40-28 68t-68 28h-416v416q0 40-28 68t-68 28h-192q-40 0-68-28t-28-68v-416h-416q-40 0-68-28t-28-68v-192q0-40 28-68t68-28h416v-416q0-40 28-68t68-28h192q40 0 68 28t28 68v416h416q40 0 68 28t28 68z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 331 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-alert.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="14" height="14" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#efb80b" d="M1024 1375v-190q0-14-9.5-23.5t-22.5-9.5h-192q-13 0-22.5 9.5t-9.5 23.5v190q0 14 9.5 23.5t22.5 9.5h192q13 0 22.5-9.5t9.5-23.5zm-2-374l18-459q0-12-10-19-13-11-24-11h-220q-11 0-24 11-10 7-10 21l17 457q0 10 10 16.5t24 6.5h185q14 0 23.5-6.5t10.5-16.5zm-14-934l768 1408q35 63-2 126-17 29-46.5 46t-63.5 17h-1536q-34 0-63.5-17t-46.5-46q-37-63-2-126l768-1408q17-31 47-49t65-18 65 18 47 49z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 504 B | 
							
								
								
									
										9
									
								
								CalibreWebCompanion/static/admin/img/icon-calendar.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | <svg width="16" height="32" viewBox="0 0 1792 3584" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | ||||||
|  |   <defs> | ||||||
|  |     <g id="icon"> | ||||||
|  |       <path d="M192 1664h288v-288h-288v288zm352 0h320v-288h-320v288zm-352-352h288v-320h-288v320zm352 0h320v-320h-320v320zm-352-384h288v-288h-288v288zm736 736h320v-288h-320v288zm-384-736h320v-288h-320v288zm768 736h288v-288h-288v288zm-384-352h320v-320h-320v320zm-352-864v-288q0-13-9.5-22.5t-22.5-9.5h-64q-13 0-22.5 9.5t-9.5 22.5v288q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5-9.5t9.5-22.5zm736 864h288v-320h-288v320zm-384-384h320v-288h-320v288zm384 0h288v-288h-288v288zm32-480v-288q0-13-9.5-22.5t-22.5-9.5h-64q-13 0-22.5 9.5t-9.5 22.5v288q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5-9.5t9.5-22.5zm384-64v1280q0 52-38 90t-90 38h-1408q-52 0-90-38t-38-90v-1280q0-52 38-90t90-38h128v-96q0-66 47-113t113-47h64q66 0 113 47t47 113v96h384v-96q0-66 47-113t113-47h64q66 0 113 47t47 113v96h128q52 0 90 38t38 90z"/> | ||||||
|  |     </g> | ||||||
|  |   </defs> | ||||||
|  |   <use xlink:href="#icon" x="0" y="0" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#icon" x="0" y="1792" fill="#003366" /> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-changelink.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#efb80b" d="M491 1536l91-91-235-235-91 91v107h128v128h107zm523-928q0-22-22-22-10 0-17 7l-542 542q-7 7-7 17 0 22 22 22 10 0 17-7l542-542q7-7 7-17zm-54-192l416 416-832 832h-416v-416zm683 96q0 53-37 90l-166 166-416-416 166-165q36-38 90-38 53 0 91 38l235 234q37 39 37 91z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 380 B | 
							
								
								
									
										9
									
								
								CalibreWebCompanion/static/admin/img/icon-clock.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | <svg width="16" height="32" viewBox="0 0 1792 3584" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | ||||||
|  |   <defs> | ||||||
|  |     <g id="icon"> | ||||||
|  |       <path d="M1024 544v448q0 14-9 23t-23 9h-320q-14 0-23-9t-9-23v-64q0-14 9-23t23-9h224v-352q0-14 9-23t23-9h64q14 0 23 9t9 23zm416 352q0-148-73-273t-198-198-273-73-273 73-198 198-73 273 73 273 198 198 273 73 273-73 198-198 73-273zm224 0q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |   </defs> | ||||||
|  |   <use xlink:href="#icon" x="0" y="0" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#icon" x="0" y="1792" fill="#003366" /> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 677 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-deletelink.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="14" height="14" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#dd4646" d="M1490 1322q0 40-28 68l-136 136q-28 28-68 28t-68-28l-294-294-294 294q-28 28-68 28t-68-28l-136-136q-28-28-28-68t28-68l294-294-294-294q-28-28-28-68t28-68l136-136q28-28 68-28t68 28l294 294 294-294q28-28 68-28t68 28l136 136q28 28 28 68t-28 68l-294 294 294 294q28 28 28 68z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 392 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-no.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#dd4646" d="M1277 1122q0-26-19-45l-181-181 181-181q19-19 19-45 0-27-19-46l-90-90q-19-19-46-19-26 0-45 19l-181 181-181-181q-19-19-45-19-27 0-46 19l-90 90q-19 19-19 46 0 26 19 45l181 181-181 181q-19 19-19 45 0 27 19 46l90 90q19 19 46 19 26 0 45-19l181-181 181 181q19 19 45 19 27 0 46-19l90-90q19-19 19-46zm387-226q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 560 B | 
| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#ffffff" d="M1024 1376v-192q0-14-9-23t-23-9h-192q-14 0-23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23-9t9-23zm256-672q0-88-55.5-163t-138.5-116-170-41q-243 0-371 213-15 24 8 42l132 100q7 6 19 6 16 0 25-12 53-68 86-92 34-24 86-24 48 0 85.5 26t37.5 59q0 38-20 61t-68 45q-63 28-115.5 86.5t-52.5 125.5v36q0 14 9 23t23 9h192q14 0 23-9t9-23q0-19 21.5-49.5t54.5-49.5q32-18 49-28.5t46-35 44.5-48 28-60.5 12.5-81zm384 192q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 655 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-unknown.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#666666" d="M1024 1376v-192q0-14-9-23t-23-9h-192q-14 0-23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23-9t9-23zm256-672q0-88-55.5-163t-138.5-116-170-41q-243 0-371 213-15 24 8 42l132 100q7 6 19 6 16 0 25-12 53-68 86-92 34-24 86-24 48 0 85.5 26t37.5 59q0 38-20 61t-68 45q-63 28-115.5 86.5t-52.5 125.5v36q0 14 9 23t23 9h192q14 0 23-9t9-23q0-19 21.5-49.5t54.5-49.5q32-18 49-28.5t46-35 44.5-48 28-60.5 12.5-81zm384 192q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 655 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-viewlink.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#2b70bf" d="M1664 960q-152-236-381-353 61 104 61 225 0 185-131.5 316.5t-316.5 131.5-316.5-131.5-131.5-316.5q0-121 61-225-229 117-381 353 133 205 333.5 326.5t434.5 121.5 434.5-121.5 333.5-326.5zm-720-384q0-20-14-34t-34-14q-125 0-214.5 89.5t-89.5 214.5q0 20 14 34t34 14 34-14 14-34q0-86 61-147t147-61q20 0 34-14t14-34zm848 384q0 34-20 69-140 230-376.5 368.5t-499.5 138.5-499.5-139-376.5-368q-20-35-20-69t20-69q140-229 376.5-368t499.5-139 499.5 139 376.5 368q20 35 20 69z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 581 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/icon-yes.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#70bf2b" d="M1412 734q0-28-18-46l-91-90q-19-19-45-19t-45 19l-408 407-226-226q-19-19-45-19t-45 19l-91 90q-18 18-18 46 0 27 18 45l362 362q19 19 45 19 27 0 46-19l543-543q18-18 18-45zm252 162q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 436 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/inline-delete.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="16" height="16" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#999999" d="M1277 1122q0-26-19-45l-181-181 181-181q19-19 19-45 0-27-19-46l-90-90q-19-19-46-19-26 0-45 19l-181 181-181-181q-19-19-45-19-27 0-46 19l-90 90q-19 19-19 46 0 26 19 45l181 181-181 181q-19 19-19 45 0 27 19 46l90 90q19 19 46 19 26 0 45-19l181-181 181 181q19 19 45 19 27 0 46-19l90-90q19-19 19-46zm387-226q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 560 B | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/search.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="15" height="15" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#555555" d="M1216 832q0-185-131.5-316.5t-316.5-131.5-316.5 131.5-131.5 316.5 131.5 316.5 316.5 131.5 316.5-131.5 131.5-316.5zm512 832q0 52-38 90t-90 38q-54 0-90-38l-343-342q-179 124-399 124-143 0-273.5-55.5t-225-150-150-225-55.5-273.5 55.5-273.5 150-225 225-150 273.5-55.5 273.5 55.5 225 150 150 225 55.5 273.5q0 220-124 399l343 343q37 37 37 90z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 458 B | 
							
								
								
									
										34
									
								
								CalibreWebCompanion/static/admin/img/selector-icons.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | |||||||
|  | <svg width="16" height="192" viewBox="0 0 1792 21504" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | ||||||
|  |   <defs> | ||||||
|  |     <g id="up"> | ||||||
|  |       <path d="M1412 895q0-27-18-45l-362-362-91-91q-18-18-45-18t-45 18l-91 91-362 362q-18 18-18 45t18 45l91 91q18 18 45 18t45-18l189-189v502q0 26 19 45t45 19h128q26 0 45-19t19-45v-502l189 189q19 19 45 19t45-19l91-91q18-18 18-45zm252 1q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="down"> | ||||||
|  |       <path d="M1412 897q0-27-18-45l-91-91q-18-18-45-18t-45 18l-189 189v-502q0-26-19-45t-45-19h-128q-26 0-45 19t-19 45v502l-189-189q-19-19-45-19t-45 19l-91 91q-18 18-18 45t18 45l362 362 91 91q18 18 45 18t45-18l91-91 362-362q18-18 18-45zm252-1q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="left"> | ||||||
|  |       <path d="M1408 960v-128q0-26-19-45t-45-19h-502l189-189q19-19 19-45t-19-45l-91-91q-18-18-45-18t-45 18l-362 362-91 91q-18 18-18 45t18 45l91 91 362 362q18 18 45 18t45-18l91-91q18-18 18-45t-18-45l-189-189h502q26 0 45-19t19-45zm256-64q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="right"> | ||||||
|  |       <path d="M1413 896q0-27-18-45l-91-91-362-362q-18-18-45-18t-45 18l-91 91q-18 18-18 45t18 45l189 189h-502q-26 0-45 19t-19 45v128q0 26 19 45t45 19h502l-189 189q-19 19-19 45t19 45l91 91q18 18 45 18t45-18l362-362 91-91q18-18 18-45zm251 0q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="clearall"> | ||||||
|  |       <path transform="translate(336, 336) scale(0.75)" d="M1037 1395l102-102q19-19 19-45t-19-45l-307-307 307-307q19-19 19-45t-19-45l-102-102q-19-19-45-19t-45 19l-454 454q-19 19-19 45t19 45l454 454q19 19 45 19t45-19zm627-499q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="chooseall"> | ||||||
|  |       <path transform="translate(336, 336) scale(0.75)" d="M845 1395l454-454q19-19 19-45t-19-45l-454-454q-19-19-45-19t-45 19l-102 102q-19 19-19 45t19 45l307 307-307 307q-19 19-19 45t19 45l102 102q19 19 45 19t45-19zm819-499q0 209-103 385.5t-279.5 279.5-385.5 103-385.5-103-279.5-279.5-103-385.5 103-385.5 279.5-279.5 385.5-103 385.5 103 279.5 279.5 103 385.5z"/> | ||||||
|  |     </g> | ||||||
|  |   </defs> | ||||||
|  |   <use xlink:href="#up" x="0" y="0" fill="#666666" /> | ||||||
|  |   <use xlink:href="#up" x="0" y="1792" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#down" x="0" y="3584" fill="#666666" /> | ||||||
|  |   <use xlink:href="#down" x="0" y="5376" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#left" x="0" y="7168" fill="#666666" /> | ||||||
|  |   <use xlink:href="#left" x="0" y="8960" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#right" x="0" y="10752" fill="#666666" /> | ||||||
|  |   <use xlink:href="#right" x="0" y="12544" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#clearall" x="0" y="14336" fill="#666666" /> | ||||||
|  |   <use xlink:href="#clearall" x="0" y="16128" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#chooseall" x="0" y="17920" fill="#666666" /> | ||||||
|  |   <use xlink:href="#chooseall" x="0" y="19712" fill="#447e9b" /> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 3.2 KiB | 
							
								
								
									
										19
									
								
								CalibreWebCompanion/static/admin/img/sorting-icons.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,19 @@ | |||||||
|  | <svg width="14" height="84" viewBox="0 0 1792 10752" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | ||||||
|  |   <defs> | ||||||
|  |     <g id="sort"> | ||||||
|  |       <path d="M1408 1088q0 26-19 45l-448 448q-19 19-45 19t-45-19l-448-448q-19-19-19-45t19-45 45-19h896q26 0 45 19t19 45zm0-384q0 26-19 45t-45 19h-896q-26 0-45-19t-19-45 19-45l448-448q19-19 45-19t45 19l448 448q19 19 19 45z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="ascending"> | ||||||
|  |       <path d="M1408 1216q0 26-19 45t-45 19h-896q-26 0-45-19t-19-45 19-45l448-448q19-19 45-19t45 19l448 448q19 19 19 45z"/> | ||||||
|  |     </g> | ||||||
|  |     <g id="descending"> | ||||||
|  |       <path d="M1408 704q0 26-19 45l-448 448q-19 19-45 19t-45-19l-448-448q-19-19-19-45t19-45 45-19h896q26 0 45 19t19 45z"/> | ||||||
|  |     </g> | ||||||
|  |   </defs> | ||||||
|  |   <use xlink:href="#sort" x="0" y="0" fill="#999999" /> | ||||||
|  |   <use xlink:href="#sort" x="0" y="1792" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#ascending" x="0" y="3584" fill="#999999" /> | ||||||
|  |   <use xlink:href="#ascending" x="0" y="5376" fill="#447e9b" /> | ||||||
|  |   <use xlink:href="#descending" x="0" y="7168" fill="#999999" /> | ||||||
|  |   <use xlink:href="#descending" x="0" y="8960" fill="#447e9b" /> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/img/tooltag-add.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#ffffff" d="M1600 736v192q0 40-28 68t-68 28h-416v416q0 40-28 68t-68 28h-192q-40 0-68-28t-28-68v-416h-416q-40 0-68-28t-28-68v-192q0-40 28-68t68-28h416v-416q0-40 28-68t68-28h192q40 0 68 28t28 68v416h416q40 0 68 28t28 68z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 331 B | 
| @@ -0,0 +1,3 @@ | |||||||
|  | <svg width="13" height="13" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"> | ||||||
|  |   <path fill="#ffffff" d="M1363 877l-742 742q-19 19-45 19t-45-19l-166-166q-19-19-19-45t19-45l531-531-531-531q-19-19-19-45t19-45l166-166q19-19 45-19t45 19l742 742q19 19 19 45t-19 45z"/> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 280 B | 
							
								
								
									
										144
									
								
								CalibreWebCompanion/static/admin/js/SelectBox.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,144 @@ | |||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     var SelectBox = { | ||||||
|  |         cache: {}, | ||||||
|  |         init: function(id) { | ||||||
|  |             var box = document.getElementById(id); | ||||||
|  |             var node; | ||||||
|  |             SelectBox.cache[id] = []; | ||||||
|  |             var cache = SelectBox.cache[id]; | ||||||
|  |             var boxOptions = box.options; | ||||||
|  |             var boxOptionsLength = boxOptions.length; | ||||||
|  |             for (var i = 0, j = boxOptionsLength; i < j; i++) { | ||||||
|  |                 node = boxOptions[i]; | ||||||
|  |                 cache.push({value: node.value, text: node.text, displayed: 1}); | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         redisplay: function(id) { | ||||||
|  |             // Repopulate HTML select box from cache | ||||||
|  |             var box = document.getElementById(id); | ||||||
|  |             var node; | ||||||
|  |             $(box).empty(); // clear all options | ||||||
|  |             var new_options = box.outerHTML.slice(0, -9); // grab just the opening tag | ||||||
|  |             var cache = SelectBox.cache[id]; | ||||||
|  |             for (var i = 0, j = cache.length; i < j; i++) { | ||||||
|  |                 node = cache[i]; | ||||||
|  |                 if (node.displayed) { | ||||||
|  |                     var new_option = new Option(node.text, node.value, false, false); | ||||||
|  |                     // Shows a tooltip when hovering over the option | ||||||
|  |                     new_option.setAttribute("title", node.text); | ||||||
|  |                     new_options += new_option.outerHTML; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             new_options += '</select>'; | ||||||
|  |             box.outerHTML = new_options; | ||||||
|  |         }, | ||||||
|  |         filter: function(id, text) { | ||||||
|  |             // Redisplay the HTML select box, displaying only the choices containing ALL | ||||||
|  |             // the words in text. (It's an AND search.) | ||||||
|  |             var tokens = text.toLowerCase().split(/\s+/); | ||||||
|  |             var node, token; | ||||||
|  |             var cache = SelectBox.cache[id]; | ||||||
|  |             for (var i = 0, j = cache.length; i < j; i++) { | ||||||
|  |                 node = cache[i]; | ||||||
|  |                 node.displayed = 1; | ||||||
|  |                 var node_text = node.text.toLowerCase(); | ||||||
|  |                 var numTokens = tokens.length; | ||||||
|  |                 for (var k = 0; k < numTokens; k++) { | ||||||
|  |                     token = tokens[k]; | ||||||
|  |                     if (node_text.indexOf(token) === -1) { | ||||||
|  |                         node.displayed = 0; | ||||||
|  |                         break; // Once the first token isn't found we're done | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             SelectBox.redisplay(id); | ||||||
|  |         }, | ||||||
|  |         delete_from_cache: function(id, value) { | ||||||
|  |             var node, delete_index = null; | ||||||
|  |             var cache = SelectBox.cache[id]; | ||||||
|  |             for (var i = 0, j = cache.length; i < j; i++) { | ||||||
|  |                 node = cache[i]; | ||||||
|  |                 if (node.value === value) { | ||||||
|  |                     delete_index = i; | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             cache.splice(delete_index, 1); | ||||||
|  |         }, | ||||||
|  |         add_to_cache: function(id, option) { | ||||||
|  |             SelectBox.cache[id].push({value: option.value, text: option.text, displayed: 1}); | ||||||
|  |         }, | ||||||
|  |         cache_contains: function(id, value) { | ||||||
|  |             // Check if an item is contained in the cache | ||||||
|  |             var node; | ||||||
|  |             var cache = SelectBox.cache[id]; | ||||||
|  |             for (var i = 0, j = cache.length; i < j; i++) { | ||||||
|  |                 node = cache[i]; | ||||||
|  |                 if (node.value === value) { | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return false; | ||||||
|  |         }, | ||||||
|  |         move: function(from, to) { | ||||||
|  |             var from_box = document.getElementById(from); | ||||||
|  |             var option; | ||||||
|  |             var boxOptions = from_box.options; | ||||||
|  |             var boxOptionsLength = boxOptions.length; | ||||||
|  |             for (var i = 0, j = boxOptionsLength; i < j; i++) { | ||||||
|  |                 option = boxOptions[i]; | ||||||
|  |                 var option_value = option.value; | ||||||
|  |                 if (option.selected && SelectBox.cache_contains(from, option_value)) { | ||||||
|  |                     SelectBox.add_to_cache(to, {value: option_value, text: option.text, displayed: 1}); | ||||||
|  |                     SelectBox.delete_from_cache(from, option_value); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             SelectBox.redisplay(from); | ||||||
|  |             SelectBox.redisplay(to); | ||||||
|  |         }, | ||||||
|  |         move_all: function(from, to) { | ||||||
|  |             var from_box = document.getElementById(from); | ||||||
|  |             var option; | ||||||
|  |             var boxOptions = from_box.options; | ||||||
|  |             var boxOptionsLength = boxOptions.length; | ||||||
|  |             for (var i = 0, j = boxOptionsLength; i < j; i++) { | ||||||
|  |                 option = boxOptions[i]; | ||||||
|  |                 var option_value = option.value; | ||||||
|  |                 if (SelectBox.cache_contains(from, option_value)) { | ||||||
|  |                     SelectBox.add_to_cache(to, {value: option_value, text: option.text, displayed: 1}); | ||||||
|  |                     SelectBox.delete_from_cache(from, option_value); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             SelectBox.redisplay(from); | ||||||
|  |             SelectBox.redisplay(to); | ||||||
|  |         }, | ||||||
|  |         sort: function(id) { | ||||||
|  |             SelectBox.cache[id].sort(function(a, b) { | ||||||
|  |                 a = a.text.toLowerCase(); | ||||||
|  |                 b = b.text.toLowerCase(); | ||||||
|  |                 try { | ||||||
|  |                     if (a > b) { | ||||||
|  |                         return 1; | ||||||
|  |                     } | ||||||
|  |                     if (a < b) { | ||||||
|  |                         return -1; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 catch (e) { | ||||||
|  |                     // silently fail on IE 'unknown' exception | ||||||
|  |                 } | ||||||
|  |                 return 0; | ||||||
|  |             } ); | ||||||
|  |         }, | ||||||
|  |         select_all: function(id) { | ||||||
|  |             var box = document.getElementById(id); | ||||||
|  |             var boxOptions = box.options; | ||||||
|  |             var boxOptionsLength = boxOptions.length; | ||||||
|  |             for (var i = 0; i < boxOptionsLength; i++) { | ||||||
|  |                 boxOptions[i].selected = 'selected'; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |     window.SelectBox = SelectBox; | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										246
									
								
								CalibreWebCompanion/static/admin/js/SelectFilter2.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,246 @@ | |||||||
|  | /*global SelectBox, gettext, interpolate, quickElement, SelectFilter*/ | ||||||
|  | /* | ||||||
|  | SelectFilter2 - Turns a multiple-select box into a filter interface. | ||||||
|  |  | ||||||
|  | Requires jQuery, core.js, and SelectBox.js. | ||||||
|  | */ | ||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     function findForm(node) { | ||||||
|  |         // returns the node of the form containing the given node | ||||||
|  |         if (node.tagName.toLowerCase() !== 'form') { | ||||||
|  |             return findForm(node.parentNode); | ||||||
|  |         } | ||||||
|  |         return node; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     window.SelectFilter = { | ||||||
|  |         init: function(field_id, field_name, is_stacked) { | ||||||
|  |             if (field_id.match(/__prefix__/)) { | ||||||
|  |                 // Don't initialize on empty forms. | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             var from_box = document.getElementById(field_id); | ||||||
|  |             from_box.id += '_from'; // change its ID | ||||||
|  |             from_box.className = 'filtered'; | ||||||
|  |  | ||||||
|  |             var ps = from_box.parentNode.getElementsByTagName('p'); | ||||||
|  |             for (var i = 0; i < ps.length; i++) { | ||||||
|  |                 if (ps[i].className.indexOf("info") !== -1) { | ||||||
|  |                     // Remove <p class="info">, because it just gets in the way. | ||||||
|  |                     from_box.parentNode.removeChild(ps[i]); | ||||||
|  |                 } else if (ps[i].className.indexOf("help") !== -1) { | ||||||
|  |                     // Move help text up to the top so it isn't below the select | ||||||
|  |                     // boxes or wrapped off on the side to the right of the add | ||||||
|  |                     // button: | ||||||
|  |                     from_box.parentNode.insertBefore(ps[i], from_box.parentNode.firstChild); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // <div class="selector"> or <div class="selector stacked"> | ||||||
|  |             var selector_div = quickElement('div', from_box.parentNode); | ||||||
|  |             selector_div.className = is_stacked ? 'selector stacked' : 'selector'; | ||||||
|  |  | ||||||
|  |             // <div class="selector-available"> | ||||||
|  |             var selector_available = quickElement('div', selector_div); | ||||||
|  |             selector_available.className = 'selector-available'; | ||||||
|  |             var title_available = quickElement('h2', selector_available, interpolate(gettext('Available %s') + ' ', [field_name])); | ||||||
|  |             quickElement( | ||||||
|  |                 'span', title_available, '', | ||||||
|  |                 'class', 'help help-tooltip help-icon', | ||||||
|  |                 'title', interpolate( | ||||||
|  |                     gettext( | ||||||
|  |                         'This is the list of available %s. You may choose some by ' + | ||||||
|  |                         'selecting them in the box below and then clicking the ' + | ||||||
|  |                         '"Choose" arrow between the two boxes.' | ||||||
|  |                     ), | ||||||
|  |                     [field_name] | ||||||
|  |                 ) | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             var filter_p = quickElement('p', selector_available, '', 'id', field_id + '_filter'); | ||||||
|  |             filter_p.className = 'selector-filter'; | ||||||
|  |  | ||||||
|  |             var search_filter_label = quickElement('label', filter_p, '', 'for', field_id + '_input'); | ||||||
|  |  | ||||||
|  |             quickElement( | ||||||
|  |                 'span', search_filter_label, '', | ||||||
|  |                 'class', 'help-tooltip search-label-icon', | ||||||
|  |                 'title', interpolate(gettext("Type into this box to filter down the list of available %s."), [field_name]) | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             filter_p.appendChild(document.createTextNode(' ')); | ||||||
|  |  | ||||||
|  |             var filter_input = quickElement('input', filter_p, '', 'type', 'text', 'placeholder', gettext("Filter")); | ||||||
|  |             filter_input.id = field_id + '_input'; | ||||||
|  |  | ||||||
|  |             selector_available.appendChild(from_box); | ||||||
|  |             var choose_all = quickElement('a', selector_available, gettext('Choose all'), 'title', interpolate(gettext('Click to choose all %s at once.'), [field_name]), 'href', '#', 'id', field_id + '_add_all_link'); | ||||||
|  |             choose_all.className = 'selector-chooseall'; | ||||||
|  |  | ||||||
|  |             // <ul class="selector-chooser"> | ||||||
|  |             var selector_chooser = quickElement('ul', selector_div); | ||||||
|  |             selector_chooser.className = 'selector-chooser'; | ||||||
|  |             var add_link = quickElement('a', quickElement('li', selector_chooser), gettext('Choose'), 'title', gettext('Choose'), 'href', '#', 'id', field_id + '_add_link'); | ||||||
|  |             add_link.className = 'selector-add'; | ||||||
|  |             var remove_link = quickElement('a', quickElement('li', selector_chooser), gettext('Remove'), 'title', gettext('Remove'), 'href', '#', 'id', field_id + '_remove_link'); | ||||||
|  |             remove_link.className = 'selector-remove'; | ||||||
|  |  | ||||||
|  |             // <div class="selector-chosen"> | ||||||
|  |             var selector_chosen = quickElement('div', selector_div); | ||||||
|  |             selector_chosen.className = 'selector-chosen'; | ||||||
|  |             var title_chosen = quickElement('h2', selector_chosen, interpolate(gettext('Chosen %s') + ' ', [field_name])); | ||||||
|  |             quickElement( | ||||||
|  |                 'span', title_chosen, '', | ||||||
|  |                 'class', 'help help-tooltip help-icon', | ||||||
|  |                 'title', interpolate( | ||||||
|  |                     gettext( | ||||||
|  |                         'This is the list of chosen %s. You may remove some by ' + | ||||||
|  |                         'selecting them in the box below and then clicking the ' + | ||||||
|  |                         '"Remove" arrow between the two boxes.' | ||||||
|  |                     ), | ||||||
|  |                     [field_name] | ||||||
|  |                 ) | ||||||
|  |             ); | ||||||
|  |  | ||||||
|  |             var to_box = quickElement('select', selector_chosen, '', 'id', field_id + '_to', 'multiple', 'multiple', 'size', from_box.size, 'name', from_box.getAttribute('name')); | ||||||
|  |             to_box.className = 'filtered'; | ||||||
|  |             var clear_all = quickElement('a', selector_chosen, gettext('Remove all'), 'title', interpolate(gettext('Click to remove all chosen %s at once.'), [field_name]), 'href', '#', 'id', field_id + '_remove_all_link'); | ||||||
|  |             clear_all.className = 'selector-clearall'; | ||||||
|  |  | ||||||
|  |             from_box.setAttribute('name', from_box.getAttribute('name') + '_old'); | ||||||
|  |  | ||||||
|  |             // Set up the JavaScript event handlers for the select box filter interface | ||||||
|  |             var move_selection = function(e, elem, move_func, from, to) { | ||||||
|  |                 if (elem.className.indexOf('active') !== -1) { | ||||||
|  |                     move_func(from, to); | ||||||
|  |                     SelectFilter.refresh_icons(field_id); | ||||||
|  |                 } | ||||||
|  |                 e.preventDefault(); | ||||||
|  |             }; | ||||||
|  |             choose_all.addEventListener('click', function(e) { | ||||||
|  |                 move_selection(e, this, SelectBox.move_all, field_id + '_from', field_id + '_to'); | ||||||
|  |             }); | ||||||
|  |             add_link.addEventListener('click', function(e) { | ||||||
|  |                 move_selection(e, this, SelectBox.move, field_id + '_from', field_id + '_to'); | ||||||
|  |             }); | ||||||
|  |             remove_link.addEventListener('click', function(e) { | ||||||
|  |                 move_selection(e, this, SelectBox.move, field_id + '_to', field_id + '_from'); | ||||||
|  |             }); | ||||||
|  |             clear_all.addEventListener('click', function(e) { | ||||||
|  |                 move_selection(e, this, SelectBox.move_all, field_id + '_to', field_id + '_from'); | ||||||
|  |             }); | ||||||
|  |             filter_input.addEventListener('keypress', function(e) { | ||||||
|  |                 SelectFilter.filter_key_press(e, field_id); | ||||||
|  |             }); | ||||||
|  |             filter_input.addEventListener('keyup', function(e) { | ||||||
|  |                 SelectFilter.filter_key_up(e, field_id); | ||||||
|  |             }); | ||||||
|  |             filter_input.addEventListener('keydown', function(e) { | ||||||
|  |                 SelectFilter.filter_key_down(e, field_id); | ||||||
|  |             }); | ||||||
|  |             selector_div.addEventListener('change', function(e) { | ||||||
|  |                 if (e.target.tagName === 'SELECT') { | ||||||
|  |                     SelectFilter.refresh_icons(field_id); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |             selector_div.addEventListener('dblclick', function(e) { | ||||||
|  |                 if (e.target.tagName === 'OPTION') { | ||||||
|  |                     if (e.target.closest('select').id === field_id + '_to') { | ||||||
|  |                         SelectBox.move(field_id + '_to', field_id + '_from'); | ||||||
|  |                     } else { | ||||||
|  |                         SelectBox.move(field_id + '_from', field_id + '_to'); | ||||||
|  |                     } | ||||||
|  |                     SelectFilter.refresh_icons(field_id); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |             findForm(from_box).addEventListener('submit', function() { | ||||||
|  |                 SelectBox.select_all(field_id + '_to'); | ||||||
|  |             }); | ||||||
|  |             SelectBox.init(field_id + '_from'); | ||||||
|  |             SelectBox.init(field_id + '_to'); | ||||||
|  |             // Move selected from_box options to to_box | ||||||
|  |             SelectBox.move(field_id + '_from', field_id + '_to'); | ||||||
|  |  | ||||||
|  |             if (!is_stacked) { | ||||||
|  |                 // In horizontal mode, give the same height to the two boxes. | ||||||
|  |                 var j_from_box = $('#' + field_id + '_from'); | ||||||
|  |                 var j_to_box = $('#' + field_id + '_to'); | ||||||
|  |                 j_to_box.height($(filter_p).outerHeight() + j_from_box.outerHeight()); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Initial icon refresh | ||||||
|  |             SelectFilter.refresh_icons(field_id); | ||||||
|  |         }, | ||||||
|  |         any_selected: function(field) { | ||||||
|  |             var any_selected = false; | ||||||
|  |             try { | ||||||
|  |                 // Temporarily add the required attribute and check validity. | ||||||
|  |                 // This is much faster in WebKit browsers than the fallback. | ||||||
|  |                 field.attr('required', 'required'); | ||||||
|  |                 any_selected = field.is(':valid'); | ||||||
|  |             } catch (e) { | ||||||
|  |                 // Browsers that don't support :valid (IE < 10) | ||||||
|  |                 any_selected = field.find('option:selected').length > 0; | ||||||
|  |             } | ||||||
|  |             field.removeAttr('required'); | ||||||
|  |             return any_selected; | ||||||
|  |         }, | ||||||
|  |         refresh_icons: function(field_id) { | ||||||
|  |             var from = $('#' + field_id + '_from'); | ||||||
|  |             var to = $('#' + field_id + '_to'); | ||||||
|  |             // Active if at least one item is selected | ||||||
|  |             $('#' + field_id + '_add_link').toggleClass('active', SelectFilter.any_selected(from)); | ||||||
|  |             $('#' + field_id + '_remove_link').toggleClass('active', SelectFilter.any_selected(to)); | ||||||
|  |             // Active if the corresponding box isn't empty | ||||||
|  |             $('#' + field_id + '_add_all_link').toggleClass('active', from.find('option').length > 0); | ||||||
|  |             $('#' + field_id + '_remove_all_link').toggleClass('active', to.find('option').length > 0); | ||||||
|  |         }, | ||||||
|  |         filter_key_press: function(event, field_id) { | ||||||
|  |             var from = document.getElementById(field_id + '_from'); | ||||||
|  |             // don't submit form if user pressed Enter | ||||||
|  |             if ((event.which && event.which === 13) || (event.keyCode && event.keyCode === 13)) { | ||||||
|  |                 from.selectedIndex = 0; | ||||||
|  |                 SelectBox.move(field_id + '_from', field_id + '_to'); | ||||||
|  |                 from.selectedIndex = 0; | ||||||
|  |                 event.preventDefault(); | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         filter_key_up: function(event, field_id) { | ||||||
|  |             var from = document.getElementById(field_id + '_from'); | ||||||
|  |             var temp = from.selectedIndex; | ||||||
|  |             SelectBox.filter(field_id + '_from', document.getElementById(field_id + '_input').value); | ||||||
|  |             from.selectedIndex = temp; | ||||||
|  |             return true; | ||||||
|  |         }, | ||||||
|  |         filter_key_down: function(event, field_id) { | ||||||
|  |             var from = document.getElementById(field_id + '_from'); | ||||||
|  |             // right arrow -- move across | ||||||
|  |             if ((event.which && event.which === 39) || (event.keyCode && event.keyCode === 39)) { | ||||||
|  |                 var old_index = from.selectedIndex; | ||||||
|  |                 SelectBox.move(field_id + '_from', field_id + '_to'); | ||||||
|  |                 from.selectedIndex = (old_index === from.length) ? from.length - 1 : old_index; | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |             // down arrow -- wrap around | ||||||
|  |             if ((event.which && event.which === 40) || (event.keyCode && event.keyCode === 40)) { | ||||||
|  |                 from.selectedIndex = (from.length === from.selectedIndex + 1) ? 0 : from.selectedIndex + 1; | ||||||
|  |             } | ||||||
|  |             // up arrow -- wrap around | ||||||
|  |             if ((event.which && event.which === 38) || (event.keyCode && event.keyCode === 38)) { | ||||||
|  |                 from.selectedIndex = (from.selectedIndex === 0) ? from.length - 1 : from.selectedIndex - 1; | ||||||
|  |             } | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     window.addEventListener('load', function(e) { | ||||||
|  |         $('select.selectfilter, select.selectfilterstacked').each(function() { | ||||||
|  |             var $el = $(this), | ||||||
|  |                 data = $el.data(); | ||||||
|  |             SelectFilter.init($el.attr('id'), data.fieldName, parseInt(data.isStacked, 10)); | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										153
									
								
								CalibreWebCompanion/static/admin/js/actions.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,153 @@ | |||||||
|  | /*global gettext, interpolate, ngettext*/ | ||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     var lastChecked; | ||||||
|  |  | ||||||
|  |     $.fn.actions = function(opts) { | ||||||
|  |         var options = $.extend({}, $.fn.actions.defaults, opts); | ||||||
|  |         var actionCheckboxes = $(this); | ||||||
|  |         var list_editable_changed = false; | ||||||
|  |         var showQuestion = function() { | ||||||
|  |                 $(options.acrossClears).hide(); | ||||||
|  |                 $(options.acrossQuestions).show(); | ||||||
|  |                 $(options.allContainer).hide(); | ||||||
|  |             }, | ||||||
|  |             showClear = function() { | ||||||
|  |                 $(options.acrossClears).show(); | ||||||
|  |                 $(options.acrossQuestions).hide(); | ||||||
|  |                 $(options.actionContainer).toggleClass(options.selectedClass); | ||||||
|  |                 $(options.allContainer).show(); | ||||||
|  |                 $(options.counterContainer).hide(); | ||||||
|  |             }, | ||||||
|  |             reset = function() { | ||||||
|  |                 $(options.acrossClears).hide(); | ||||||
|  |                 $(options.acrossQuestions).hide(); | ||||||
|  |                 $(options.allContainer).hide(); | ||||||
|  |                 $(options.counterContainer).show(); | ||||||
|  |             }, | ||||||
|  |             clearAcross = function() { | ||||||
|  |                 reset(); | ||||||
|  |                 $(options.acrossInput).val(0); | ||||||
|  |                 $(options.actionContainer).removeClass(options.selectedClass); | ||||||
|  |             }, | ||||||
|  |             checker = function(checked) { | ||||||
|  |                 if (checked) { | ||||||
|  |                     showQuestion(); | ||||||
|  |                 } else { | ||||||
|  |                     reset(); | ||||||
|  |                 } | ||||||
|  |                 $(actionCheckboxes).prop("checked", checked) | ||||||
|  |                     .parent().parent().toggleClass(options.selectedClass, checked); | ||||||
|  |             }, | ||||||
|  |             updateCounter = function() { | ||||||
|  |                 var sel = $(actionCheckboxes).filter(":checked").length; | ||||||
|  |                 // data-actions-icnt is defined in the generated HTML | ||||||
|  |                 // and contains the total amount of objects in the queryset | ||||||
|  |                 var actions_icnt = $('.action-counter').data('actionsIcnt'); | ||||||
|  |                 $(options.counterContainer).html(interpolate( | ||||||
|  |                     ngettext('%(sel)s of %(cnt)s selected', '%(sel)s of %(cnt)s selected', sel), { | ||||||
|  |                         sel: sel, | ||||||
|  |                         cnt: actions_icnt | ||||||
|  |                     }, true)); | ||||||
|  |                 $(options.allToggle).prop("checked", function() { | ||||||
|  |                     var value; | ||||||
|  |                     if (sel === actionCheckboxes.length) { | ||||||
|  |                         value = true; | ||||||
|  |                         showQuestion(); | ||||||
|  |                     } else { | ||||||
|  |                         value = false; | ||||||
|  |                         clearAcross(); | ||||||
|  |                     } | ||||||
|  |                     return value; | ||||||
|  |                 }); | ||||||
|  |             }; | ||||||
|  |         // Show counter by default | ||||||
|  |         $(options.counterContainer).show(); | ||||||
|  |         // Check state of checkboxes and reinit state if needed | ||||||
|  |         $(this).filter(":checked").each(function(i) { | ||||||
|  |             $(this).parent().parent().toggleClass(options.selectedClass); | ||||||
|  |             updateCounter(); | ||||||
|  |             if ($(options.acrossInput).val() === 1) { | ||||||
|  |                 showClear(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         $(options.allToggle).show().on('click', function() { | ||||||
|  |             checker($(this).prop("checked")); | ||||||
|  |             updateCounter(); | ||||||
|  |         }); | ||||||
|  |         $("a", options.acrossQuestions).on('click', function(event) { | ||||||
|  |             event.preventDefault(); | ||||||
|  |             $(options.acrossInput).val(1); | ||||||
|  |             showClear(); | ||||||
|  |         }); | ||||||
|  |         $("a", options.acrossClears).on('click', function(event) { | ||||||
|  |             event.preventDefault(); | ||||||
|  |             $(options.allToggle).prop("checked", false); | ||||||
|  |             clearAcross(); | ||||||
|  |             checker(0); | ||||||
|  |             updateCounter(); | ||||||
|  |         }); | ||||||
|  |         lastChecked = null; | ||||||
|  |         $(actionCheckboxes).on('click', function(event) { | ||||||
|  |             if (!event) { event = window.event; } | ||||||
|  |             var target = event.target ? event.target : event.srcElement; | ||||||
|  |             if (lastChecked && $.data(lastChecked) !== $.data(target) && event.shiftKey === true) { | ||||||
|  |                 var inrange = false; | ||||||
|  |                 $(lastChecked).prop("checked", target.checked) | ||||||
|  |                     .parent().parent().toggleClass(options.selectedClass, target.checked); | ||||||
|  |                 $(actionCheckboxes).each(function() { | ||||||
|  |                     if ($.data(this) === $.data(lastChecked) || $.data(this) === $.data(target)) { | ||||||
|  |                         inrange = (inrange) ? false : true; | ||||||
|  |                     } | ||||||
|  |                     if (inrange) { | ||||||
|  |                         $(this).prop("checked", target.checked) | ||||||
|  |                             .parent().parent().toggleClass(options.selectedClass, target.checked); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |             $(target).parent().parent().toggleClass(options.selectedClass, target.checked); | ||||||
|  |             lastChecked = target; | ||||||
|  |             updateCounter(); | ||||||
|  |         }); | ||||||
|  |         $('form#changelist-form table#result_list tr').on('change', 'td:gt(0) :input', function() { | ||||||
|  |             list_editable_changed = true; | ||||||
|  |         }); | ||||||
|  |         $('form#changelist-form button[name="index"]').on('click', function(event) { | ||||||
|  |             if (list_editable_changed) { | ||||||
|  |                 return confirm(gettext("You have unsaved changes on individual editable fields. If you run an action, your unsaved changes will be lost.")); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         $('form#changelist-form input[name="_save"]').on('click', function(event) { | ||||||
|  |             var action_changed = false; | ||||||
|  |             $('select option:selected', options.actionContainer).each(function() { | ||||||
|  |                 if ($(this).val()) { | ||||||
|  |                     action_changed = true; | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |             if (action_changed) { | ||||||
|  |                 if (list_editable_changed) { | ||||||
|  |                     return confirm(gettext("You have selected an action, but you haven't saved your changes to individual fields yet. Please click OK to save. You'll need to re-run the action.")); | ||||||
|  |                 } else { | ||||||
|  |                     return confirm(gettext("You have selected an action, and you haven't made any changes on individual fields. You're probably looking for the Go button rather than the Save button.")); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     }; | ||||||
|  |     /* Setup plugin defaults */ | ||||||
|  |     $.fn.actions.defaults = { | ||||||
|  |         actionContainer: "div.actions", | ||||||
|  |         counterContainer: "span.action-counter", | ||||||
|  |         allContainer: "div.actions span.all", | ||||||
|  |         acrossInput: "div.actions input.select-across", | ||||||
|  |         acrossQuestions: "div.actions span.question", | ||||||
|  |         acrossClears: "div.actions span.clear", | ||||||
|  |         allToggle: "#action-toggle", | ||||||
|  |         selectedClass: "selected" | ||||||
|  |     }; | ||||||
|  |     $(document).ready(function() { | ||||||
|  |         var $actionsEls = $('tr input.action-select'); | ||||||
|  |         if ($actionsEls.length > 0) { | ||||||
|  |             $actionsEls.actions(); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										7
									
								
								CalibreWebCompanion/static/admin/js/actions.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | |||||||
|  | (function(a){var f;a.fn.actions=function(e){var b=a.extend({},a.fn.actions.defaults,e),g=a(this),k=!1,l=function(){a(b.acrossClears).hide();a(b.acrossQuestions).show();a(b.allContainer).hide()},m=function(){a(b.acrossClears).show();a(b.acrossQuestions).hide();a(b.actionContainer).toggleClass(b.selectedClass);a(b.allContainer).show();a(b.counterContainer).hide()},n=function(){a(b.acrossClears).hide();a(b.acrossQuestions).hide();a(b.allContainer).hide();a(b.counterContainer).show()},p=function(){n(); | ||||||
|  | a(b.acrossInput).val(0);a(b.actionContainer).removeClass(b.selectedClass)},q=function(c){c?l():n();a(g).prop("checked",c).parent().parent().toggleClass(b.selectedClass,c)},h=function(){var c=a(g).filter(":checked").length,d=a(".action-counter").data("actionsIcnt");a(b.counterContainer).html(interpolate(ngettext("%(sel)s of %(cnt)s selected","%(sel)s of %(cnt)s selected",c),{sel:c,cnt:d},!0));a(b.allToggle).prop("checked",function(){if(c===g.length){var a=!0;l()}else a=!1,p();return a})};a(b.counterContainer).show(); | ||||||
|  | a(this).filter(":checked").each(function(c){a(this).parent().parent().toggleClass(b.selectedClass);h();1===a(b.acrossInput).val()&&m()});a(b.allToggle).show().on("click",function(){q(a(this).prop("checked"));h()});a("a",b.acrossQuestions).on("click",function(c){c.preventDefault();a(b.acrossInput).val(1);m()});a("a",b.acrossClears).on("click",function(c){c.preventDefault();a(b.allToggle).prop("checked",!1);p();q(0);h()});f=null;a(g).on("click",function(c){c||(c=window.event);var d=c.target?c.target: | ||||||
|  | c.srcElement;if(f&&a.data(f)!==a.data(d)&&!0===c.shiftKey){var e=!1;a(f).prop("checked",d.checked).parent().parent().toggleClass(b.selectedClass,d.checked);a(g).each(function(){if(a.data(this)===a.data(f)||a.data(this)===a.data(d))e=e?!1:!0;e&&a(this).prop("checked",d.checked).parent().parent().toggleClass(b.selectedClass,d.checked)})}a(d).parent().parent().toggleClass(b.selectedClass,d.checked);f=d;h()});a("form#changelist-form table#result_list tr").on("change","td:gt(0) :input",function(){k=!0}); | ||||||
|  | a('form#changelist-form button[name="index"]').on("click",function(a){if(k)return confirm(gettext("You have unsaved changes on individual editable fields. If you run an action, your unsaved changes will be lost."))});a('form#changelist-form input[name="_save"]').on("click",function(c){var d=!1;a("select option:selected",b.actionContainer).each(function(){a(this).val()&&(d=!0)});if(d)return k?confirm(gettext("You have selected an action, but you haven't saved your changes to individual fields yet. Please click OK to save. You'll need to re-run the action.")): | ||||||
|  | confirm(gettext("You have selected an action, and you haven't made any changes on individual fields. You're probably looking for the Go button rather than the Save button."))})};a.fn.actions.defaults={actionContainer:"div.actions",counterContainer:"span.action-counter",allContainer:"div.actions span.all",acrossInput:"div.actions input.select-across",acrossQuestions:"div.actions span.question",acrossClears:"div.actions span.clear",allToggle:"#action-toggle",selectedClass:"selected"};a(document).ready(function(){var e= | ||||||
|  | a("tr input.action-select");0<e.length&&e.actions()})})(django.jQuery); | ||||||
							
								
								
									
										423
									
								
								CalibreWebCompanion/static/admin/js/admin/DateTimeShortcuts.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,423 @@ | |||||||
|  | /*global Calendar, findPosX, findPosY, get_format, gettext, gettext_noop, interpolate, ngettext, quickElement*/ | ||||||
|  | // Inserts shortcut buttons after all of the following: | ||||||
|  | //     <input type="text" class="vDateField"> | ||||||
|  | //     <input type="text" class="vTimeField"> | ||||||
|  | (function() { | ||||||
|  |     'use strict'; | ||||||
|  |     var DateTimeShortcuts = { | ||||||
|  |         calendars: [], | ||||||
|  |         calendarInputs: [], | ||||||
|  |         clockInputs: [], | ||||||
|  |         clockHours: { | ||||||
|  |             default_: [ | ||||||
|  |                 [gettext_noop('Now'), -1], | ||||||
|  |                 [gettext_noop('Midnight'), 0], | ||||||
|  |                 [gettext_noop('6 a.m.'), 6], | ||||||
|  |                 [gettext_noop('Noon'), 12], | ||||||
|  |                 [gettext_noop('6 p.m.'), 18] | ||||||
|  |             ] | ||||||
|  |         }, | ||||||
|  |         dismissClockFunc: [], | ||||||
|  |         dismissCalendarFunc: [], | ||||||
|  |         calendarDivName1: 'calendarbox', // name of calendar <div> that gets toggled | ||||||
|  |         calendarDivName2: 'calendarin', // name of <div> that contains calendar | ||||||
|  |         calendarLinkName: 'calendarlink', // name of the link that is used to toggle | ||||||
|  |         clockDivName: 'clockbox', // name of clock <div> that gets toggled | ||||||
|  |         clockLinkName: 'clocklink', // name of the link that is used to toggle | ||||||
|  |         shortCutsClass: 'datetimeshortcuts', // class of the clock and cal shortcuts | ||||||
|  |         timezoneWarningClass: 'timezonewarning', // class of the warning for timezone mismatch | ||||||
|  |         timezoneOffset: 0, | ||||||
|  |         init: function() { | ||||||
|  |             var body = document.getElementsByTagName('body')[0]; | ||||||
|  |             var serverOffset = body.getAttribute('data-admin-utc-offset'); | ||||||
|  |             if (serverOffset) { | ||||||
|  |                 var localOffset = new Date().getTimezoneOffset() * -60; | ||||||
|  |                 DateTimeShortcuts.timezoneOffset = localOffset - serverOffset; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             var inputs = document.getElementsByTagName('input'); | ||||||
|  |             for (var i = 0; i < inputs.length; i++) { | ||||||
|  |                 var inp = inputs[i]; | ||||||
|  |                 if (inp.getAttribute('type') === 'text' && inp.className.match(/vTimeField/)) { | ||||||
|  |                     DateTimeShortcuts.addClock(inp); | ||||||
|  |                     DateTimeShortcuts.addTimezoneWarning(inp); | ||||||
|  |                 } | ||||||
|  |                 else if (inp.getAttribute('type') === 'text' && inp.className.match(/vDateField/)) { | ||||||
|  |                     DateTimeShortcuts.addCalendar(inp); | ||||||
|  |                     DateTimeShortcuts.addTimezoneWarning(inp); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         // Return the current time while accounting for the server timezone. | ||||||
|  |         now: function() { | ||||||
|  |             var body = document.getElementsByTagName('body')[0]; | ||||||
|  |             var serverOffset = body.getAttribute('data-admin-utc-offset'); | ||||||
|  |             if (serverOffset) { | ||||||
|  |                 var localNow = new Date(); | ||||||
|  |                 var localOffset = localNow.getTimezoneOffset() * -60; | ||||||
|  |                 localNow.setTime(localNow.getTime() + 1000 * (serverOffset - localOffset)); | ||||||
|  |                 return localNow; | ||||||
|  |             } else { | ||||||
|  |                 return new Date(); | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         // Add a warning when the time zone in the browser and backend do not match. | ||||||
|  |         addTimezoneWarning: function(inp) { | ||||||
|  |             var warningClass = DateTimeShortcuts.timezoneWarningClass; | ||||||
|  |             var timezoneOffset = DateTimeShortcuts.timezoneOffset / 3600; | ||||||
|  |  | ||||||
|  |             // Only warn if there is a time zone mismatch. | ||||||
|  |             if (!timezoneOffset) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Check if warning is already there. | ||||||
|  |             if (inp.parentNode.querySelectorAll('.' + warningClass).length) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             var message; | ||||||
|  |             if (timezoneOffset > 0) { | ||||||
|  |                 message = ngettext( | ||||||
|  |                     'Note: You are %s hour ahead of server time.', | ||||||
|  |                     'Note: You are %s hours ahead of server time.', | ||||||
|  |                     timezoneOffset | ||||||
|  |                 ); | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 timezoneOffset *= -1; | ||||||
|  |                 message = ngettext( | ||||||
|  |                     'Note: You are %s hour behind server time.', | ||||||
|  |                     'Note: You are %s hours behind server time.', | ||||||
|  |                     timezoneOffset | ||||||
|  |                 ); | ||||||
|  |             } | ||||||
|  |             message = interpolate(message, [timezoneOffset]); | ||||||
|  |  | ||||||
|  |             var warning = document.createElement('span'); | ||||||
|  |             warning.className = warningClass; | ||||||
|  |             warning.textContent = message; | ||||||
|  |             inp.parentNode.appendChild(document.createElement('br')); | ||||||
|  |             inp.parentNode.appendChild(warning); | ||||||
|  |         }, | ||||||
|  |         // Add clock widget to a given field | ||||||
|  |         addClock: function(inp) { | ||||||
|  |             var num = DateTimeShortcuts.clockInputs.length; | ||||||
|  |             DateTimeShortcuts.clockInputs[num] = inp; | ||||||
|  |             DateTimeShortcuts.dismissClockFunc[num] = function() { DateTimeShortcuts.dismissClock(num); return true; }; | ||||||
|  |  | ||||||
|  |             // Shortcut links (clock icon and "Now" link) | ||||||
|  |             var shortcuts_span = document.createElement('span'); | ||||||
|  |             shortcuts_span.className = DateTimeShortcuts.shortCutsClass; | ||||||
|  |             inp.parentNode.insertBefore(shortcuts_span, inp.nextSibling); | ||||||
|  |             var now_link = document.createElement('a'); | ||||||
|  |             now_link.setAttribute('href', "#"); | ||||||
|  |             now_link.textContent = gettext('Now'); | ||||||
|  |             now_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.handleClockQuicklink(num, -1); | ||||||
|  |             }); | ||||||
|  |             var clock_link = document.createElement('a'); | ||||||
|  |             clock_link.setAttribute('href', '#'); | ||||||
|  |             clock_link.id = DateTimeShortcuts.clockLinkName + num; | ||||||
|  |             clock_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 // avoid triggering the document click handler to dismiss the clock | ||||||
|  |                 e.stopPropagation(); | ||||||
|  |                 DateTimeShortcuts.openClock(num); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             quickElement( | ||||||
|  |                 'span', clock_link, '', | ||||||
|  |                 'class', 'clock-icon', | ||||||
|  |                 'title', gettext('Choose a Time') | ||||||
|  |             ); | ||||||
|  |             shortcuts_span.appendChild(document.createTextNode('\u00A0')); | ||||||
|  |             shortcuts_span.appendChild(now_link); | ||||||
|  |             shortcuts_span.appendChild(document.createTextNode('\u00A0|\u00A0')); | ||||||
|  |             shortcuts_span.appendChild(clock_link); | ||||||
|  |  | ||||||
|  |             // Create clock link div | ||||||
|  |             // | ||||||
|  |             // Markup looks like: | ||||||
|  |             // <div id="clockbox1" class="clockbox module"> | ||||||
|  |             //     <h2>Choose a time</h2> | ||||||
|  |             //     <ul class="timelist"> | ||||||
|  |             //         <li><a href="#">Now</a></li> | ||||||
|  |             //         <li><a href="#">Midnight</a></li> | ||||||
|  |             //         <li><a href="#">6 a.m.</a></li> | ||||||
|  |             //         <li><a href="#">Noon</a></li> | ||||||
|  |             //         <li><a href="#">6 p.m.</a></li> | ||||||
|  |             //     </ul> | ||||||
|  |             //     <p class="calendar-cancel"><a href="#">Cancel</a></p> | ||||||
|  |             // </div> | ||||||
|  |  | ||||||
|  |             var clock_box = document.createElement('div'); | ||||||
|  |             clock_box.style.display = 'none'; | ||||||
|  |             clock_box.style.position = 'absolute'; | ||||||
|  |             clock_box.className = 'clockbox module'; | ||||||
|  |             clock_box.setAttribute('id', DateTimeShortcuts.clockDivName + num); | ||||||
|  |             document.body.appendChild(clock_box); | ||||||
|  |             clock_box.addEventListener('click', function(e) { e.stopPropagation(); }); | ||||||
|  |  | ||||||
|  |             quickElement('h2', clock_box, gettext('Choose a time')); | ||||||
|  |             var time_list = quickElement('ul', clock_box); | ||||||
|  |             time_list.className = 'timelist'; | ||||||
|  |             // The list of choices can be overridden in JavaScript like this: | ||||||
|  |             // DateTimeShortcuts.clockHours.name = [['3 a.m.', 3]]; | ||||||
|  |             // where name is the name attribute of the <input>. | ||||||
|  |             var name = typeof DateTimeShortcuts.clockHours[inp.name] === 'undefined' ? 'default_' : inp.name; | ||||||
|  |             DateTimeShortcuts.clockHours[name].forEach(function(element) { | ||||||
|  |                 var time_link = quickElement('a', quickElement('li', time_list), gettext(element[0]), 'href', '#'); | ||||||
|  |                 time_link.addEventListener('click', function(e) { | ||||||
|  |                     e.preventDefault(); | ||||||
|  |                     DateTimeShortcuts.handleClockQuicklink(num, element[1]); | ||||||
|  |                 }); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             var cancel_p = quickElement('p', clock_box); | ||||||
|  |             cancel_p.className = 'calendar-cancel'; | ||||||
|  |             var cancel_link = quickElement('a', cancel_p, gettext('Cancel'), 'href', '#'); | ||||||
|  |             cancel_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.dismissClock(num); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             document.addEventListener('keyup', function(event) { | ||||||
|  |                 if (event.which === 27) { | ||||||
|  |                     // ESC key closes popup | ||||||
|  |                     DateTimeShortcuts.dismissClock(num); | ||||||
|  |                     event.preventDefault(); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         }, | ||||||
|  |         openClock: function(num) { | ||||||
|  |             var clock_box = document.getElementById(DateTimeShortcuts.clockDivName + num); | ||||||
|  |             var clock_link = document.getElementById(DateTimeShortcuts.clockLinkName + num); | ||||||
|  |  | ||||||
|  |             // Recalculate the clockbox position | ||||||
|  |             // is it left-to-right or right-to-left layout ? | ||||||
|  |             if (window.getComputedStyle(document.body).direction !== 'rtl') { | ||||||
|  |                 clock_box.style.left = findPosX(clock_link) + 17 + 'px'; | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 // since style's width is in em, it'd be tough to calculate | ||||||
|  |                 // px value of it. let's use an estimated px for now | ||||||
|  |                 // TODO: IE returns wrong value for findPosX when in rtl mode | ||||||
|  |                 //       (it returns as it was left aligned), needs to be fixed. | ||||||
|  |                 clock_box.style.left = findPosX(clock_link) - 110 + 'px'; | ||||||
|  |             } | ||||||
|  |             clock_box.style.top = Math.max(0, findPosY(clock_link) - 30) + 'px'; | ||||||
|  |  | ||||||
|  |             // Show the clock box | ||||||
|  |             clock_box.style.display = 'block'; | ||||||
|  |             document.addEventListener('click', DateTimeShortcuts.dismissClockFunc[num]); | ||||||
|  |         }, | ||||||
|  |         dismissClock: function(num) { | ||||||
|  |             document.getElementById(DateTimeShortcuts.clockDivName + num).style.display = 'none'; | ||||||
|  |             document.removeEventListener('click', DateTimeShortcuts.dismissClockFunc[num]); | ||||||
|  |         }, | ||||||
|  |         handleClockQuicklink: function(num, val) { | ||||||
|  |             var d; | ||||||
|  |             if (val === -1) { | ||||||
|  |                 d = DateTimeShortcuts.now(); | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 d = new Date(1970, 1, 1, val, 0, 0, 0); | ||||||
|  |             } | ||||||
|  |             DateTimeShortcuts.clockInputs[num].value = d.strftime(get_format('TIME_INPUT_FORMATS')[0]); | ||||||
|  |             DateTimeShortcuts.clockInputs[num].focus(); | ||||||
|  |             DateTimeShortcuts.dismissClock(num); | ||||||
|  |         }, | ||||||
|  |         // Add calendar widget to a given field. | ||||||
|  |         addCalendar: function(inp) { | ||||||
|  |             var num = DateTimeShortcuts.calendars.length; | ||||||
|  |  | ||||||
|  |             DateTimeShortcuts.calendarInputs[num] = inp; | ||||||
|  |             DateTimeShortcuts.dismissCalendarFunc[num] = function() { DateTimeShortcuts.dismissCalendar(num); return true; }; | ||||||
|  |  | ||||||
|  |             // Shortcut links (calendar icon and "Today" link) | ||||||
|  |             var shortcuts_span = document.createElement('span'); | ||||||
|  |             shortcuts_span.className = DateTimeShortcuts.shortCutsClass; | ||||||
|  |             inp.parentNode.insertBefore(shortcuts_span, inp.nextSibling); | ||||||
|  |             var today_link = document.createElement('a'); | ||||||
|  |             today_link.setAttribute('href', '#'); | ||||||
|  |             today_link.appendChild(document.createTextNode(gettext('Today'))); | ||||||
|  |             today_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.handleCalendarQuickLink(num, 0); | ||||||
|  |             }); | ||||||
|  |             var cal_link = document.createElement('a'); | ||||||
|  |             cal_link.setAttribute('href', '#'); | ||||||
|  |             cal_link.id = DateTimeShortcuts.calendarLinkName + num; | ||||||
|  |             cal_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 // avoid triggering the document click handler to dismiss the calendar | ||||||
|  |                 e.stopPropagation(); | ||||||
|  |                 DateTimeShortcuts.openCalendar(num); | ||||||
|  |             }); | ||||||
|  |             quickElement( | ||||||
|  |                 'span', cal_link, '', | ||||||
|  |                 'class', 'date-icon', | ||||||
|  |                 'title', gettext('Choose a Date') | ||||||
|  |             ); | ||||||
|  |             shortcuts_span.appendChild(document.createTextNode('\u00A0')); | ||||||
|  |             shortcuts_span.appendChild(today_link); | ||||||
|  |             shortcuts_span.appendChild(document.createTextNode('\u00A0|\u00A0')); | ||||||
|  |             shortcuts_span.appendChild(cal_link); | ||||||
|  |  | ||||||
|  |             // Create calendarbox div. | ||||||
|  |             // | ||||||
|  |             // Markup looks like: | ||||||
|  |             // | ||||||
|  |             // <div id="calendarbox3" class="calendarbox module"> | ||||||
|  |             //     <h2> | ||||||
|  |             //           <a href="#" class="link-previous">‹</a> | ||||||
|  |             //           <a href="#" class="link-next">›</a> February 2003 | ||||||
|  |             //     </h2> | ||||||
|  |             //     <div class="calendar" id="calendarin3"> | ||||||
|  |             //         <!-- (cal) --> | ||||||
|  |             //     </div> | ||||||
|  |             //     <div class="calendar-shortcuts"> | ||||||
|  |             //          <a href="#">Yesterday</a> | <a href="#">Today</a> | <a href="#">Tomorrow</a> | ||||||
|  |             //     </div> | ||||||
|  |             //     <p class="calendar-cancel"><a href="#">Cancel</a></p> | ||||||
|  |             // </div> | ||||||
|  |             var cal_box = document.createElement('div'); | ||||||
|  |             cal_box.style.display = 'none'; | ||||||
|  |             cal_box.style.position = 'absolute'; | ||||||
|  |             cal_box.className = 'calendarbox module'; | ||||||
|  |             cal_box.setAttribute('id', DateTimeShortcuts.calendarDivName1 + num); | ||||||
|  |             document.body.appendChild(cal_box); | ||||||
|  |             cal_box.addEventListener('click', function(e) { e.stopPropagation(); }); | ||||||
|  |  | ||||||
|  |             // next-prev links | ||||||
|  |             var cal_nav = quickElement('div', cal_box); | ||||||
|  |             var cal_nav_prev = quickElement('a', cal_nav, '<', 'href', '#'); | ||||||
|  |             cal_nav_prev.className = 'calendarnav-previous'; | ||||||
|  |             cal_nav_prev.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.drawPrev(num); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             var cal_nav_next = quickElement('a', cal_nav, '>', 'href', '#'); | ||||||
|  |             cal_nav_next.className = 'calendarnav-next'; | ||||||
|  |             cal_nav_next.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.drawNext(num); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             // main box | ||||||
|  |             var cal_main = quickElement('div', cal_box, '', 'id', DateTimeShortcuts.calendarDivName2 + num); | ||||||
|  |             cal_main.className = 'calendar'; | ||||||
|  |             DateTimeShortcuts.calendars[num] = new Calendar(DateTimeShortcuts.calendarDivName2 + num, DateTimeShortcuts.handleCalendarCallback(num)); | ||||||
|  |             DateTimeShortcuts.calendars[num].drawCurrent(); | ||||||
|  |  | ||||||
|  |             // calendar shortcuts | ||||||
|  |             var shortcuts = quickElement('div', cal_box); | ||||||
|  |             shortcuts.className = 'calendar-shortcuts'; | ||||||
|  |             var day_link = quickElement('a', shortcuts, gettext('Yesterday'), 'href', '#'); | ||||||
|  |             day_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.handleCalendarQuickLink(num, -1); | ||||||
|  |             }); | ||||||
|  |             shortcuts.appendChild(document.createTextNode('\u00A0|\u00A0')); | ||||||
|  |             day_link = quickElement('a', shortcuts, gettext('Today'), 'href', '#'); | ||||||
|  |             day_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.handleCalendarQuickLink(num, 0); | ||||||
|  |             }); | ||||||
|  |             shortcuts.appendChild(document.createTextNode('\u00A0|\u00A0')); | ||||||
|  |             day_link = quickElement('a', shortcuts, gettext('Tomorrow'), 'href', '#'); | ||||||
|  |             day_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.handleCalendarQuickLink(num, +1); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             // cancel bar | ||||||
|  |             var cancel_p = quickElement('p', cal_box); | ||||||
|  |             cancel_p.className = 'calendar-cancel'; | ||||||
|  |             var cancel_link = quickElement('a', cancel_p, gettext('Cancel'), 'href', '#'); | ||||||
|  |             cancel_link.addEventListener('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 DateTimeShortcuts.dismissCalendar(num); | ||||||
|  |             }); | ||||||
|  |             document.addEventListener('keyup', function(event) { | ||||||
|  |                 if (event.which === 27) { | ||||||
|  |                     // ESC key closes popup | ||||||
|  |                     DateTimeShortcuts.dismissCalendar(num); | ||||||
|  |                     event.preventDefault(); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         }, | ||||||
|  |         openCalendar: function(num) { | ||||||
|  |             var cal_box = document.getElementById(DateTimeShortcuts.calendarDivName1 + num); | ||||||
|  |             var cal_link = document.getElementById(DateTimeShortcuts.calendarLinkName + num); | ||||||
|  |             var inp = DateTimeShortcuts.calendarInputs[num]; | ||||||
|  |  | ||||||
|  |             // Determine if the current value in the input has a valid date. | ||||||
|  |             // If so, draw the calendar with that date's year and month. | ||||||
|  |             if (inp.value) { | ||||||
|  |                 var format = get_format('DATE_INPUT_FORMATS')[0]; | ||||||
|  |                 var selected = inp.value.strptime(format); | ||||||
|  |                 var year = selected.getUTCFullYear(); | ||||||
|  |                 var month = selected.getUTCMonth() + 1; | ||||||
|  |                 var re = /\d{4}/; | ||||||
|  |                 if (re.test(year.toString()) && month >= 1 && month <= 12) { | ||||||
|  |                     DateTimeShortcuts.calendars[num].drawDate(month, year, selected); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Recalculate the clockbox position | ||||||
|  |             // is it left-to-right or right-to-left layout ? | ||||||
|  |             if (window.getComputedStyle(document.body).direction !== 'rtl') { | ||||||
|  |                 cal_box.style.left = findPosX(cal_link) + 17 + 'px'; | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 // since style's width is in em, it'd be tough to calculate | ||||||
|  |                 // px value of it. let's use an estimated px for now | ||||||
|  |                 // TODO: IE returns wrong value for findPosX when in rtl mode | ||||||
|  |                 //       (it returns as it was left aligned), needs to be fixed. | ||||||
|  |                 cal_box.style.left = findPosX(cal_link) - 180 + 'px'; | ||||||
|  |             } | ||||||
|  |             cal_box.style.top = Math.max(0, findPosY(cal_link) - 75) + 'px'; | ||||||
|  |  | ||||||
|  |             cal_box.style.display = 'block'; | ||||||
|  |             document.addEventListener('click', DateTimeShortcuts.dismissCalendarFunc[num]); | ||||||
|  |         }, | ||||||
|  |         dismissCalendar: function(num) { | ||||||
|  |             document.getElementById(DateTimeShortcuts.calendarDivName1 + num).style.display = 'none'; | ||||||
|  |             document.removeEventListener('click', DateTimeShortcuts.dismissCalendarFunc[num]); | ||||||
|  |         }, | ||||||
|  |         drawPrev: function(num) { | ||||||
|  |             DateTimeShortcuts.calendars[num].drawPreviousMonth(); | ||||||
|  |         }, | ||||||
|  |         drawNext: function(num) { | ||||||
|  |             DateTimeShortcuts.calendars[num].drawNextMonth(); | ||||||
|  |         }, | ||||||
|  |         handleCalendarCallback: function(num) { | ||||||
|  |             var format = get_format('DATE_INPUT_FORMATS')[0]; | ||||||
|  |             // the format needs to be escaped a little | ||||||
|  |             format = format.replace('\\', '\\\\') | ||||||
|  |                 .replace('\r', '\\r') | ||||||
|  |                 .replace('\n', '\\n') | ||||||
|  |                 .replace('\t', '\\t') | ||||||
|  |                 .replace("'", "\\'"); | ||||||
|  |             return function(y, m, d) { | ||||||
|  |                 DateTimeShortcuts.calendarInputs[num].value = new Date(y, m - 1, d).strftime(format); | ||||||
|  |                 DateTimeShortcuts.calendarInputs[num].focus(); | ||||||
|  |                 document.getElementById(DateTimeShortcuts.calendarDivName1 + num).style.display = 'none'; | ||||||
|  |             }; | ||||||
|  |         }, | ||||||
|  |         handleCalendarQuickLink: function(num, offset) { | ||||||
|  |             var d = DateTimeShortcuts.now(); | ||||||
|  |             d.setDate(d.getDate() + offset); | ||||||
|  |             DateTimeShortcuts.calendarInputs[num].value = d.strftime(get_format('DATE_INPUT_FORMATS')[0]); | ||||||
|  |             DateTimeShortcuts.calendarInputs[num].focus(); | ||||||
|  |             DateTimeShortcuts.dismissCalendar(num); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     window.addEventListener('load', DateTimeShortcuts.init); | ||||||
|  |     window.DateTimeShortcuts = DateTimeShortcuts; | ||||||
|  | })(); | ||||||
| @@ -0,0 +1,181 @@ | |||||||
|  | /*global SelectBox, interpolate*/ | ||||||
|  | // Handles related-objects functionality: lookup link for raw_id_fields | ||||||
|  | // and Add Another links. | ||||||
|  |  | ||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |  | ||||||
|  |     // IE doesn't accept periods or dashes in the window name, but the element IDs | ||||||
|  |     // we use to generate popup window names may contain them, therefore we map them | ||||||
|  |     // to allowed characters in a reversible way so that we can locate the correct | ||||||
|  |     // element when the popup window is dismissed. | ||||||
|  |     function id_to_windowname(text) { | ||||||
|  |         text = text.replace(/\./g, '__dot__'); | ||||||
|  |         text = text.replace(/\-/g, '__dash__'); | ||||||
|  |         return text; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function windowname_to_id(text) { | ||||||
|  |         text = text.replace(/__dot__/g, '.'); | ||||||
|  |         text = text.replace(/__dash__/g, '-'); | ||||||
|  |         return text; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function showAdminPopup(triggeringLink, name_regexp, add_popup) { | ||||||
|  |         var name = triggeringLink.id.replace(name_regexp, ''); | ||||||
|  |         name = id_to_windowname(name); | ||||||
|  |         var href = triggeringLink.href; | ||||||
|  |         if (add_popup) { | ||||||
|  |             if (href.indexOf('?') === -1) { | ||||||
|  |                 href += '?_popup=1'; | ||||||
|  |             } else { | ||||||
|  |                 href += '&_popup=1'; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         var win = window.open(href, name, 'height=500,width=800,resizable=yes,scrollbars=yes'); | ||||||
|  |         win.focus(); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function showRelatedObjectLookupPopup(triggeringLink) { | ||||||
|  |         return showAdminPopup(triggeringLink, /^lookup_/, true); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function dismissRelatedLookupPopup(win, chosenId) { | ||||||
|  |         var name = windowname_to_id(win.name); | ||||||
|  |         var elem = document.getElementById(name); | ||||||
|  |         if (elem.className.indexOf('vManyToManyRawIdAdminField') !== -1 && elem.value) { | ||||||
|  |             elem.value += ',' + chosenId; | ||||||
|  |         } else { | ||||||
|  |             document.getElementById(name).value = chosenId; | ||||||
|  |         } | ||||||
|  |         win.close(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function showRelatedObjectPopup(triggeringLink) { | ||||||
|  |         return showAdminPopup(triggeringLink, /^(change|add|delete)_/, false); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function updateRelatedObjectLinks(triggeringLink) { | ||||||
|  |         var $this = $(triggeringLink); | ||||||
|  |         var siblings = $this.nextAll('.view-related, .change-related, .delete-related'); | ||||||
|  |         if (!siblings.length) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         var value = $this.val(); | ||||||
|  |         if (value) { | ||||||
|  |             siblings.each(function() { | ||||||
|  |                 var elm = $(this); | ||||||
|  |                 elm.attr('href', elm.attr('data-href-template').replace('__fk__', value)); | ||||||
|  |             }); | ||||||
|  |         } else { | ||||||
|  |             siblings.removeAttr('href'); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function dismissAddRelatedObjectPopup(win, newId, newRepr) { | ||||||
|  |         var name = windowname_to_id(win.name); | ||||||
|  |         var elem = document.getElementById(name); | ||||||
|  |         if (elem) { | ||||||
|  |             var elemName = elem.nodeName.toUpperCase(); | ||||||
|  |             if (elemName === 'SELECT') { | ||||||
|  |                 elem.options[elem.options.length] = new Option(newRepr, newId, true, true); | ||||||
|  |             } else if (elemName === 'INPUT') { | ||||||
|  |                 if (elem.className.indexOf('vManyToManyRawIdAdminField') !== -1 && elem.value) { | ||||||
|  |                     elem.value += ',' + newId; | ||||||
|  |                 } else { | ||||||
|  |                     elem.value = newId; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             // Trigger a change event to update related links if required. | ||||||
|  |             $(elem).trigger('change'); | ||||||
|  |         } else { | ||||||
|  |             var toId = name + "_to"; | ||||||
|  |             var o = new Option(newRepr, newId); | ||||||
|  |             SelectBox.add_to_cache(toId, o); | ||||||
|  |             SelectBox.redisplay(toId); | ||||||
|  |         } | ||||||
|  |         win.close(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function dismissChangeRelatedObjectPopup(win, objId, newRepr, newId) { | ||||||
|  |         var id = windowname_to_id(win.name).replace(/^edit_/, ''); | ||||||
|  |         var selectsSelector = interpolate('#%s, #%s_from, #%s_to', [id, id, id]); | ||||||
|  |         var selects = $(selectsSelector); | ||||||
|  |         selects.find('option').each(function() { | ||||||
|  |             if (this.value === objId) { | ||||||
|  |                 this.textContent = newRepr; | ||||||
|  |                 this.value = newId; | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         selects.next().find('.select2-selection__rendered').each(function() { | ||||||
|  |             // The element can have a clear button as a child. | ||||||
|  |             // Use the lastChild to modify only the displayed value. | ||||||
|  |             this.lastChild.textContent = newRepr; | ||||||
|  |             this.title = newRepr; | ||||||
|  |         }); | ||||||
|  |         win.close(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     function dismissDeleteRelatedObjectPopup(win, objId) { | ||||||
|  |         var id = windowname_to_id(win.name).replace(/^delete_/, ''); | ||||||
|  |         var selectsSelector = interpolate('#%s, #%s_from, #%s_to', [id, id, id]); | ||||||
|  |         var selects = $(selectsSelector); | ||||||
|  |         selects.find('option').each(function() { | ||||||
|  |             if (this.value === objId) { | ||||||
|  |                 $(this).remove(); | ||||||
|  |             } | ||||||
|  |         }).trigger('change'); | ||||||
|  |         win.close(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Global for testing purposes | ||||||
|  |     window.id_to_windowname = id_to_windowname; | ||||||
|  |     window.windowname_to_id = windowname_to_id; | ||||||
|  |  | ||||||
|  |     window.showRelatedObjectLookupPopup = showRelatedObjectLookupPopup; | ||||||
|  |     window.dismissRelatedLookupPopup = dismissRelatedLookupPopup; | ||||||
|  |     window.showRelatedObjectPopup = showRelatedObjectPopup; | ||||||
|  |     window.updateRelatedObjectLinks = updateRelatedObjectLinks; | ||||||
|  |     window.dismissAddRelatedObjectPopup = dismissAddRelatedObjectPopup; | ||||||
|  |     window.dismissChangeRelatedObjectPopup = dismissChangeRelatedObjectPopup; | ||||||
|  |     window.dismissDeleteRelatedObjectPopup = dismissDeleteRelatedObjectPopup; | ||||||
|  |  | ||||||
|  |     // Kept for backward compatibility | ||||||
|  |     window.showAddAnotherPopup = showRelatedObjectPopup; | ||||||
|  |     window.dismissAddAnotherPopup = dismissAddRelatedObjectPopup; | ||||||
|  |  | ||||||
|  |     $(document).ready(function() { | ||||||
|  |         $("a[data-popup-opener]").on('click', function(event) { | ||||||
|  |             event.preventDefault(); | ||||||
|  |             opener.dismissRelatedLookupPopup(window, $(this).data("popup-opener")); | ||||||
|  |         }); | ||||||
|  |         $('body').on('click', '.related-widget-wrapper-link', function(e) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             if (this.href) { | ||||||
|  |                 var event = $.Event('django:show-related', {href: this.href}); | ||||||
|  |                 $(this).trigger(event); | ||||||
|  |                 if (!event.isDefaultPrevented()) { | ||||||
|  |                     showRelatedObjectPopup(this); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         $('body').on('change', '.related-widget-wrapper select', function(e) { | ||||||
|  |             var event = $.Event('django:update-related'); | ||||||
|  |             $(this).trigger(event); | ||||||
|  |             if (!event.isDefaultPrevented()) { | ||||||
|  |                 updateRelatedObjectLinks(this); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         $('.related-widget-wrapper select').trigger('change'); | ||||||
|  |         $('body').on('click', '.related-lookup', function(e) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             var event = $.Event('django:lookup-related'); | ||||||
|  |             $(this).trigger(event); | ||||||
|  |             if (!event.isDefaultPrevented()) { | ||||||
|  |                 showRelatedObjectLookupPopup(this); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										37
									
								
								CalibreWebCompanion/static/admin/js/autocomplete.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,37 @@ | |||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     var init = function($element, options) { | ||||||
|  |         var settings = $.extend({ | ||||||
|  |             ajax: { | ||||||
|  |                 data: function(params) { | ||||||
|  |                     return { | ||||||
|  |                         term: params.term, | ||||||
|  |                         page: params.page | ||||||
|  |                     }; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, options); | ||||||
|  |         $element.select2(settings); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $.fn.djangoAdminSelect2 = function(options) { | ||||||
|  |         var settings = $.extend({}, options); | ||||||
|  |         $.each(this, function(i, element) { | ||||||
|  |             var $element = $(element); | ||||||
|  |             init($element, settings); | ||||||
|  |         }); | ||||||
|  |         return this; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $(function() { | ||||||
|  |         // Initialize all autocomplete widgets except the one in the template | ||||||
|  |         // form used when a new formset is added. | ||||||
|  |         $('.admin-autocomplete').not('[name*=__prefix__]').djangoAdminSelect2(); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     $(document).on('formset:added', (function() { | ||||||
|  |         return function(event, $newFormset) { | ||||||
|  |             return $newFormset.find('.admin-autocomplete').djangoAdminSelect2(); | ||||||
|  |         }; | ||||||
|  |     })(this)); | ||||||
|  | }(django.jQuery)); | ||||||
							
								
								
									
										208
									
								
								CalibreWebCompanion/static/admin/js/calendar.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,208 @@ | |||||||
|  | /*global gettext, pgettext, get_format, quickElement, removeChildren*/ | ||||||
|  | /* | ||||||
|  | calendar.js - Calendar functions by Adrian Holovaty | ||||||
|  | depends on core.js for utility functions like removeChildren or quickElement | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | (function() { | ||||||
|  |     'use strict'; | ||||||
|  |     // CalendarNamespace -- Provides a collection of HTML calendar-related helper functions | ||||||
|  |     var CalendarNamespace = { | ||||||
|  |         monthsOfYear: [ | ||||||
|  |             gettext('January'), | ||||||
|  |             gettext('February'), | ||||||
|  |             gettext('March'), | ||||||
|  |             gettext('April'), | ||||||
|  |             gettext('May'), | ||||||
|  |             gettext('June'), | ||||||
|  |             gettext('July'), | ||||||
|  |             gettext('August'), | ||||||
|  |             gettext('September'), | ||||||
|  |             gettext('October'), | ||||||
|  |             gettext('November'), | ||||||
|  |             gettext('December') | ||||||
|  |         ], | ||||||
|  |         daysOfWeek: [ | ||||||
|  |             pgettext('one letter Sunday', 'S'), | ||||||
|  |             pgettext('one letter Monday', 'M'), | ||||||
|  |             pgettext('one letter Tuesday', 'T'), | ||||||
|  |             pgettext('one letter Wednesday', 'W'), | ||||||
|  |             pgettext('one letter Thursday', 'T'), | ||||||
|  |             pgettext('one letter Friday', 'F'), | ||||||
|  |             pgettext('one letter Saturday', 'S') | ||||||
|  |         ], | ||||||
|  |         firstDayOfWeek: parseInt(get_format('FIRST_DAY_OF_WEEK')), | ||||||
|  |         isLeapYear: function(year) { | ||||||
|  |             return (((year % 4) === 0) && ((year % 100) !== 0 ) || ((year % 400) === 0)); | ||||||
|  |         }, | ||||||
|  |         getDaysInMonth: function(month, year) { | ||||||
|  |             var days; | ||||||
|  |             if (month === 1 || month === 3 || month === 5 || month === 7 || month === 8 || month === 10 || month === 12) { | ||||||
|  |                 days = 31; | ||||||
|  |             } | ||||||
|  |             else if (month === 4 || month === 6 || month === 9 || month === 11) { | ||||||
|  |                 days = 30; | ||||||
|  |             } | ||||||
|  |             else if (month === 2 && CalendarNamespace.isLeapYear(year)) { | ||||||
|  |                 days = 29; | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 days = 28; | ||||||
|  |             } | ||||||
|  |             return days; | ||||||
|  |         }, | ||||||
|  |         draw: function(month, year, div_id, callback, selected) { // month = 1-12, year = 1-9999 | ||||||
|  |             var today = new Date(); | ||||||
|  |             var todayDay = today.getDate(); | ||||||
|  |             var todayMonth = today.getMonth() + 1; | ||||||
|  |             var todayYear = today.getFullYear(); | ||||||
|  |             var todayClass = ''; | ||||||
|  |  | ||||||
|  |             // Use UTC functions here because the date field does not contain time | ||||||
|  |             // and using the UTC function variants prevent the local time offset | ||||||
|  |             // from altering the date, specifically the day field.  For example: | ||||||
|  |             // | ||||||
|  |             // ``` | ||||||
|  |             // var x = new Date('2013-10-02'); | ||||||
|  |             // var day = x.getDate(); | ||||||
|  |             // ``` | ||||||
|  |             // | ||||||
|  |             // The day variable above will be 1 instead of 2 in, say, US Pacific time | ||||||
|  |             // zone. | ||||||
|  |             var isSelectedMonth = false; | ||||||
|  |             if (typeof selected !== 'undefined') { | ||||||
|  |                 isSelectedMonth = (selected.getUTCFullYear() === year && (selected.getUTCMonth() + 1) === month); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             month = parseInt(month); | ||||||
|  |             year = parseInt(year); | ||||||
|  |             var calDiv = document.getElementById(div_id); | ||||||
|  |             removeChildren(calDiv); | ||||||
|  |             var calTable = document.createElement('table'); | ||||||
|  |             quickElement('caption', calTable, CalendarNamespace.monthsOfYear[month - 1] + ' ' + year); | ||||||
|  |             var tableBody = quickElement('tbody', calTable); | ||||||
|  |  | ||||||
|  |             // Draw days-of-week header | ||||||
|  |             var tableRow = quickElement('tr', tableBody); | ||||||
|  |             for (var i = 0; i < 7; i++) { | ||||||
|  |                 quickElement('th', tableRow, CalendarNamespace.daysOfWeek[(i + CalendarNamespace.firstDayOfWeek) % 7]); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             var startingPos = new Date(year, month - 1, 1 - CalendarNamespace.firstDayOfWeek).getDay(); | ||||||
|  |             var days = CalendarNamespace.getDaysInMonth(month, year); | ||||||
|  |  | ||||||
|  |             var nonDayCell; | ||||||
|  |  | ||||||
|  |             // Draw blanks before first of month | ||||||
|  |             tableRow = quickElement('tr', tableBody); | ||||||
|  |             for (i = 0; i < startingPos; i++) { | ||||||
|  |                 nonDayCell = quickElement('td', tableRow, ' '); | ||||||
|  |                 nonDayCell.className = "nonday"; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             function calendarMonth(y, m) { | ||||||
|  |                 function onClick(e) { | ||||||
|  |                     e.preventDefault(); | ||||||
|  |                     callback(y, m, this.textContent); | ||||||
|  |                 } | ||||||
|  |                 return onClick; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Draw days of month | ||||||
|  |             var currentDay = 1; | ||||||
|  |             for (i = startingPos; currentDay <= days; i++) { | ||||||
|  |                 if (i % 7 === 0 && currentDay !== 1) { | ||||||
|  |                     tableRow = quickElement('tr', tableBody); | ||||||
|  |                 } | ||||||
|  |                 if ((currentDay === todayDay) && (month === todayMonth) && (year === todayYear)) { | ||||||
|  |                     todayClass = 'today'; | ||||||
|  |                 } else { | ||||||
|  |                     todayClass = ''; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 // use UTC function; see above for explanation. | ||||||
|  |                 if (isSelectedMonth && currentDay === selected.getUTCDate()) { | ||||||
|  |                     if (todayClass !== '') { | ||||||
|  |                         todayClass += " "; | ||||||
|  |                     } | ||||||
|  |                     todayClass += "selected"; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 var cell = quickElement('td', tableRow, '', 'class', todayClass); | ||||||
|  |                 var link = quickElement('a', cell, currentDay, 'href', '#'); | ||||||
|  |                 link.addEventListener('click', calendarMonth(year, month)); | ||||||
|  |                 currentDay++; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Draw blanks after end of month (optional, but makes for valid code) | ||||||
|  |             while (tableRow.childNodes.length < 7) { | ||||||
|  |                 nonDayCell = quickElement('td', tableRow, ' '); | ||||||
|  |                 nonDayCell.className = "nonday"; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             calDiv.appendChild(calTable); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // Calendar -- A calendar instance | ||||||
|  |     function Calendar(div_id, callback, selected) { | ||||||
|  |         // div_id (string) is the ID of the element in which the calendar will | ||||||
|  |         //     be displayed | ||||||
|  |         // callback (string) is the name of a JavaScript function that will be | ||||||
|  |         //     called with the parameters (year, month, day) when a day in the | ||||||
|  |         //     calendar is clicked | ||||||
|  |         this.div_id = div_id; | ||||||
|  |         this.callback = callback; | ||||||
|  |         this.today = new Date(); | ||||||
|  |         this.currentMonth = this.today.getMonth() + 1; | ||||||
|  |         this.currentYear = this.today.getFullYear(); | ||||||
|  |         if (typeof selected !== 'undefined') { | ||||||
|  |             this.selected = selected; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     Calendar.prototype = { | ||||||
|  |         drawCurrent: function() { | ||||||
|  |             CalendarNamespace.draw(this.currentMonth, this.currentYear, this.div_id, this.callback, this.selected); | ||||||
|  |         }, | ||||||
|  |         drawDate: function(month, year, selected) { | ||||||
|  |             this.currentMonth = month; | ||||||
|  |             this.currentYear = year; | ||||||
|  |  | ||||||
|  |             if(selected) { | ||||||
|  |                 this.selected = selected; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             this.drawCurrent(); | ||||||
|  |         }, | ||||||
|  |         drawPreviousMonth: function() { | ||||||
|  |             if (this.currentMonth === 1) { | ||||||
|  |                 this.currentMonth = 12; | ||||||
|  |                 this.currentYear--; | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 this.currentMonth--; | ||||||
|  |             } | ||||||
|  |             this.drawCurrent(); | ||||||
|  |         }, | ||||||
|  |         drawNextMonth: function() { | ||||||
|  |             if (this.currentMonth === 12) { | ||||||
|  |                 this.currentMonth = 1; | ||||||
|  |                 this.currentYear++; | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 this.currentMonth++; | ||||||
|  |             } | ||||||
|  |             this.drawCurrent(); | ||||||
|  |         }, | ||||||
|  |         drawPreviousYear: function() { | ||||||
|  |             this.currentYear--; | ||||||
|  |             this.drawCurrent(); | ||||||
|  |         }, | ||||||
|  |         drawNextYear: function() { | ||||||
|  |             this.currentYear++; | ||||||
|  |             this.drawCurrent(); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |     window.Calendar = Calendar; | ||||||
|  |     window.CalendarNamespace = CalendarNamespace; | ||||||
|  | })(); | ||||||
							
								
								
									
										13
									
								
								CalibreWebCompanion/static/admin/js/cancel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     $(function() { | ||||||
|  |         $('.cancel-link').on('click', function(e) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             if (window.location.search.indexOf('&_popup=1') === -1) { | ||||||
|  |                 window.history.back(); // Go back if not a popup. | ||||||
|  |             } else { | ||||||
|  |                 window.close(); // Otherwise, close the popup. | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										20
									
								
								CalibreWebCompanion/static/admin/js/change_form.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | |||||||
|  | /*global showAddAnotherPopup, showRelatedObjectLookupPopup showRelatedObjectPopup updateRelatedObjectLinks*/ | ||||||
|  |  | ||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     $(document).ready(function() { | ||||||
|  |         var modelName = $('#django-admin-form-add-constants').data('modelName'); | ||||||
|  |         $('body').on('click', '.add-another', function(e) { | ||||||
|  |             e.preventDefault(); | ||||||
|  |             var event = $.Event('django:add-another-related'); | ||||||
|  |             $(this).trigger(event); | ||||||
|  |             if (!event.isDefaultPrevented()) { | ||||||
|  |                 showAddAnotherPopup(this); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         if (modelName) { | ||||||
|  |             $('form#' + modelName + '_form :input:visible:enabled:first').focus(); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										55
									
								
								CalibreWebCompanion/static/admin/js/collapse.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,55 @@ | |||||||
|  | /*global gettext*/ | ||||||
|  | (function() { | ||||||
|  |     'use strict'; | ||||||
|  |     var closestElem = function(elem, tagName) { | ||||||
|  |         if (elem.nodeName === tagName.toUpperCase()) { | ||||||
|  |             return elem; | ||||||
|  |         } | ||||||
|  |         if (elem.parentNode.nodeName === 'BODY') { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return elem.parentNode && closestElem(elem.parentNode, tagName); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     window.addEventListener('load', function() { | ||||||
|  |         // Add anchor tag for Show/Hide link | ||||||
|  |         var fieldsets = document.querySelectorAll('fieldset.collapse'); | ||||||
|  |         for (var i = 0; i < fieldsets.length; i++) { | ||||||
|  |             var elem = fieldsets[i]; | ||||||
|  |             // Don't hide if fields in this fieldset have errors | ||||||
|  |             if (elem.querySelectorAll('div.errors').length === 0) { | ||||||
|  |                 elem.classList.add('collapsed'); | ||||||
|  |                 var h2 = elem.querySelector('h2'); | ||||||
|  |                 var link = document.createElement('a'); | ||||||
|  |                 link.setAttribute('id', 'fieldsetcollapser' + i); | ||||||
|  |                 link.setAttribute('class', 'collapse-toggle'); | ||||||
|  |                 link.setAttribute('href', '#'); | ||||||
|  |                 link.textContent = gettext('Show'); | ||||||
|  |                 h2.appendChild(document.createTextNode(' (')); | ||||||
|  |                 h2.appendChild(link); | ||||||
|  |                 h2.appendChild(document.createTextNode(')')); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         // Add toggle to hide/show anchor tag | ||||||
|  |         var toggleFunc = function(ev) { | ||||||
|  |             if (ev.target.matches('.collapse-toggle')) { | ||||||
|  |                 ev.preventDefault(); | ||||||
|  |                 ev.stopPropagation(); | ||||||
|  |                 var fieldset = closestElem(ev.target, 'fieldset'); | ||||||
|  |                 if (fieldset.classList.contains('collapsed')) { | ||||||
|  |                     // Show | ||||||
|  |                     ev.target.textContent = gettext('Hide'); | ||||||
|  |                     fieldset.classList.remove('collapsed'); | ||||||
|  |                 } else { | ||||||
|  |                     // Hide | ||||||
|  |                     ev.target.textContent = gettext('Show'); | ||||||
|  |                     fieldset.classList.add('collapsed'); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         var inlineDivs = document.querySelectorAll('fieldset.module'); | ||||||
|  |         for (i = 0; i < inlineDivs.length; i++) { | ||||||
|  |             inlineDivs[i].addEventListener('click', toggleFunc); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | })(); | ||||||
							
								
								
									
										3
									
								
								CalibreWebCompanion/static/admin/js/collapse.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | (function(){var e=function(b,a){return b.nodeName===a.toUpperCase()?b:"BODY"===b.parentNode.nodeName?null:b.parentNode&&e(b.parentNode,a)};window.addEventListener("load",function(){for(var b=document.querySelectorAll("fieldset.collapse"),a=0;a<b.length;a++){var c=b[a];if(0===c.querySelectorAll("div.errors").length){c.classList.add("collapsed");c=c.querySelector("h2");var d=document.createElement("a");d.setAttribute("id","fieldsetcollapser"+a);d.setAttribute("class","collapse-toggle");d.setAttribute("href", | ||||||
|  | "#");d.textContent=gettext("Show");c.appendChild(document.createTextNode(" ("));c.appendChild(d);c.appendChild(document.createTextNode(")"))}}b=function(a){if(a.target.matches(".collapse-toggle")){a.preventDefault();a.stopPropagation();var b=e(a.target,"fieldset");b.classList.contains("collapsed")?(a.target.textContent=gettext("Hide"),b.classList.remove("collapsed")):(a.target.textContent=gettext("Show"),b.classList.add("collapsed"))}};c=document.querySelectorAll("fieldset.module");for(a=0;a<c.length;a++)c[a].addEventListener("click", | ||||||
|  | b)})})(); | ||||||
							
								
								
									
										173
									
								
								CalibreWebCompanion/static/admin/js/core.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,173 @@ | |||||||
|  | // Core javascript helper functions | ||||||
|  |  | ||||||
|  | // basic browser identification & version | ||||||
|  | var isOpera = (navigator.userAgent.indexOf("Opera") >= 0) && parseFloat(navigator.appVersion); | ||||||
|  | var isIE = ((document.all) && (!isOpera)) && parseFloat(navigator.appVersion.split("MSIE ")[1].split(";")[0]); | ||||||
|  |  | ||||||
|  | // quickElement(tagType, parentReference [, textInChildNode, attribute, attributeValue ...]); | ||||||
|  | function quickElement() { | ||||||
|  |     'use strict'; | ||||||
|  |     var obj = document.createElement(arguments[0]); | ||||||
|  |     if (arguments[2]) { | ||||||
|  |         var textNode = document.createTextNode(arguments[2]); | ||||||
|  |         obj.appendChild(textNode); | ||||||
|  |     } | ||||||
|  |     var len = arguments.length; | ||||||
|  |     for (var i = 3; i < len; i += 2) { | ||||||
|  |         obj.setAttribute(arguments[i], arguments[i + 1]); | ||||||
|  |     } | ||||||
|  |     arguments[1].appendChild(obj); | ||||||
|  |     return obj; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // "a" is reference to an object | ||||||
|  | function removeChildren(a) { | ||||||
|  |     'use strict'; | ||||||
|  |     while (a.hasChildNodes()) { | ||||||
|  |         a.removeChild(a.lastChild); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ---------------------------------------------------------------------------- | ||||||
|  | // Find-position functions by PPK | ||||||
|  | // See https://www.quirksmode.org/js/findpos.html | ||||||
|  | // ---------------------------------------------------------------------------- | ||||||
|  | function findPosX(obj) { | ||||||
|  |     'use strict'; | ||||||
|  |     var curleft = 0; | ||||||
|  |     if (obj.offsetParent) { | ||||||
|  |         while (obj.offsetParent) { | ||||||
|  |             curleft += obj.offsetLeft - ((isOpera) ? 0 : obj.scrollLeft); | ||||||
|  |             obj = obj.offsetParent; | ||||||
|  |         } | ||||||
|  |         // IE offsetParent does not include the top-level | ||||||
|  |         if (isIE && obj.parentElement) { | ||||||
|  |             curleft += obj.offsetLeft - obj.scrollLeft; | ||||||
|  |         } | ||||||
|  |     } else if (obj.x) { | ||||||
|  |         curleft += obj.x; | ||||||
|  |     } | ||||||
|  |     return curleft; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function findPosY(obj) { | ||||||
|  |     'use strict'; | ||||||
|  |     var curtop = 0; | ||||||
|  |     if (obj.offsetParent) { | ||||||
|  |         while (obj.offsetParent) { | ||||||
|  |             curtop += obj.offsetTop - ((isOpera) ? 0 : obj.scrollTop); | ||||||
|  |             obj = obj.offsetParent; | ||||||
|  |         } | ||||||
|  |         // IE offsetParent does not include the top-level | ||||||
|  |         if (isIE && obj.parentElement) { | ||||||
|  |             curtop += obj.offsetTop - obj.scrollTop; | ||||||
|  |         } | ||||||
|  |     } else if (obj.y) { | ||||||
|  |         curtop += obj.y; | ||||||
|  |     } | ||||||
|  |     return curtop; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | //----------------------------------------------------------------------------- | ||||||
|  | // Date object extensions | ||||||
|  | // ---------------------------------------------------------------------------- | ||||||
|  | (function() { | ||||||
|  |     'use strict'; | ||||||
|  |     Date.prototype.getTwelveHours = function() { | ||||||
|  |         return this.getHours() % 12 || 12; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.getTwoDigitMonth = function() { | ||||||
|  |         return (this.getMonth() < 9) ? '0' + (this.getMonth() + 1) : (this.getMonth() + 1); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.getTwoDigitDate = function() { | ||||||
|  |         return (this.getDate() < 10) ? '0' + this.getDate() : this.getDate(); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.getTwoDigitTwelveHour = function() { | ||||||
|  |         return (this.getTwelveHours() < 10) ? '0' + this.getTwelveHours() : this.getTwelveHours(); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.getTwoDigitHour = function() { | ||||||
|  |         return (this.getHours() < 10) ? '0' + this.getHours() : this.getHours(); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.getTwoDigitMinute = function() { | ||||||
|  |         return (this.getMinutes() < 10) ? '0' + this.getMinutes() : this.getMinutes(); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.getTwoDigitSecond = function() { | ||||||
|  |         return (this.getSeconds() < 10) ? '0' + this.getSeconds() : this.getSeconds(); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.getFullMonthName = function() { | ||||||
|  |         return typeof window.CalendarNamespace === "undefined" | ||||||
|  |             ? this.getTwoDigitMonth() | ||||||
|  |             : window.CalendarNamespace.monthsOfYear[this.getMonth()]; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     Date.prototype.strftime = function(format) { | ||||||
|  |         var fields = { | ||||||
|  |             B: this.getFullMonthName(), | ||||||
|  |             c: this.toString(), | ||||||
|  |             d: this.getTwoDigitDate(), | ||||||
|  |             H: this.getTwoDigitHour(), | ||||||
|  |             I: this.getTwoDigitTwelveHour(), | ||||||
|  |             m: this.getTwoDigitMonth(), | ||||||
|  |             M: this.getTwoDigitMinute(), | ||||||
|  |             p: (this.getHours() >= 12) ? 'PM' : 'AM', | ||||||
|  |             S: this.getTwoDigitSecond(), | ||||||
|  |             w: '0' + this.getDay(), | ||||||
|  |             x: this.toLocaleDateString(), | ||||||
|  |             X: this.toLocaleTimeString(), | ||||||
|  |             y: ('' + this.getFullYear()).substr(2, 4), | ||||||
|  |             Y: '' + this.getFullYear(), | ||||||
|  |             '%': '%' | ||||||
|  |         }; | ||||||
|  |         var result = '', i = 0; | ||||||
|  |         while (i < format.length) { | ||||||
|  |             if (format.charAt(i) === '%') { | ||||||
|  |                 result = result + fields[format.charAt(i + 1)]; | ||||||
|  |                 ++i; | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 result = result + format.charAt(i); | ||||||
|  |             } | ||||||
|  |             ++i; | ||||||
|  |         } | ||||||
|  |         return result; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // ---------------------------------------------------------------------------- | ||||||
|  |     // String object extensions | ||||||
|  |     // ---------------------------------------------------------------------------- | ||||||
|  |     String.prototype.strptime = function(format) { | ||||||
|  |         var split_format = format.split(/[.\-/]/); | ||||||
|  |         var date = this.split(/[.\-/]/); | ||||||
|  |         var i = 0; | ||||||
|  |         var day, month, year; | ||||||
|  |         while (i < split_format.length) { | ||||||
|  |             switch (split_format[i]) { | ||||||
|  |             case "%d": | ||||||
|  |                 day = date[i]; | ||||||
|  |                 break; | ||||||
|  |             case "%m": | ||||||
|  |                 month = date[i] - 1; | ||||||
|  |                 break; | ||||||
|  |             case "%Y": | ||||||
|  |                 year = date[i]; | ||||||
|  |                 break; | ||||||
|  |             case "%y": | ||||||
|  |                 year = date[i]; | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             ++i; | ||||||
|  |         } | ||||||
|  |         // Create Date object from UTC since the parsed value is supposed to be | ||||||
|  |         // in UTC, not local time. Also, the calendar uses UTC functions for | ||||||
|  |         // date extraction. | ||||||
|  |         return new Date(Date.UTC(year, month, day)); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  | })(); | ||||||
							
								
								
									
										298
									
								
								CalibreWebCompanion/static/admin/js/inlines.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,298 @@ | |||||||
|  | /*global DateTimeShortcuts, SelectFilter*/ | ||||||
|  | /** | ||||||
|  |  * Django admin inlines | ||||||
|  |  * | ||||||
|  |  * Based on jQuery Formset 1.1 | ||||||
|  |  * @author Stanislaus Madueke (stan DOT madueke AT gmail DOT com) | ||||||
|  |  * @requires jQuery 1.2.6 or later | ||||||
|  |  * | ||||||
|  |  * Copyright (c) 2009, Stanislaus Madueke | ||||||
|  |  * All rights reserved. | ||||||
|  |  * | ||||||
|  |  * Spiced up with Code from Zain Memon's GSoC project 2009 | ||||||
|  |  * and modified for Django by Jannis Leidel, Travis Swicegood and Julien Phalip. | ||||||
|  |  * | ||||||
|  |  * Licensed under the New BSD License | ||||||
|  |  * See: https://opensource.org/licenses/bsd-license.php | ||||||
|  |  */ | ||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     $.fn.formset = function(opts) { | ||||||
|  |         var options = $.extend({}, $.fn.formset.defaults, opts); | ||||||
|  |         var $this = $(this); | ||||||
|  |         var $parent = $this.parent(); | ||||||
|  |         var updateElementIndex = function(el, prefix, ndx) { | ||||||
|  |             var id_regex = new RegExp("(" + prefix + "-(\\d+|__prefix__))"); | ||||||
|  |             var replacement = prefix + "-" + ndx; | ||||||
|  |             if ($(el).prop("for")) { | ||||||
|  |                 $(el).prop("for", $(el).prop("for").replace(id_regex, replacement)); | ||||||
|  |             } | ||||||
|  |             if (el.id) { | ||||||
|  |                 el.id = el.id.replace(id_regex, replacement); | ||||||
|  |             } | ||||||
|  |             if (el.name) { | ||||||
|  |                 el.name = el.name.replace(id_regex, replacement); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |         var totalForms = $("#id_" + options.prefix + "-TOTAL_FORMS").prop("autocomplete", "off"); | ||||||
|  |         var nextIndex = parseInt(totalForms.val(), 10); | ||||||
|  |         var maxForms = $("#id_" + options.prefix + "-MAX_NUM_FORMS").prop("autocomplete", "off"); | ||||||
|  |         // only show the add button if we are allowed to add more items, | ||||||
|  |         // note that max_num = None translates to a blank string. | ||||||
|  |         var showAddButton = maxForms.val() === '' || (maxForms.val() - totalForms.val()) > 0; | ||||||
|  |         $this.each(function(i) { | ||||||
|  |             $(this).not("." + options.emptyCssClass).addClass(options.formCssClass); | ||||||
|  |         }); | ||||||
|  |         if ($this.length && showAddButton) { | ||||||
|  |             var addButton = options.addButton; | ||||||
|  |             if (addButton === null) { | ||||||
|  |                 if ($this.prop("tagName") === "TR") { | ||||||
|  |                     // If forms are laid out as table rows, insert the | ||||||
|  |                     // "add" button in a new table row: | ||||||
|  |                     var numCols = this.eq(-1).children().length; | ||||||
|  |                     $parent.append('<tr class="' + options.addCssClass + '"><td colspan="' + numCols + '"><a href="#">' + options.addText + "</a></tr>"); | ||||||
|  |                     addButton = $parent.find("tr:last a"); | ||||||
|  |                 } else { | ||||||
|  |                     // Otherwise, insert it immediately after the last form: | ||||||
|  |                     $this.filter(":last").after('<div class="' + options.addCssClass + '"><a href="#">' + options.addText + "</a></div>"); | ||||||
|  |                     addButton = $this.filter(":last").next().find("a"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             addButton.on('click', function(e) { | ||||||
|  |                 e.preventDefault(); | ||||||
|  |                 var template = $("#" + options.prefix + "-empty"); | ||||||
|  |                 var row = template.clone(true); | ||||||
|  |                 row.removeClass(options.emptyCssClass) | ||||||
|  |                     .addClass(options.formCssClass) | ||||||
|  |                     .attr("id", options.prefix + "-" + nextIndex); | ||||||
|  |                 if (row.is("tr")) { | ||||||
|  |                     // If the forms are laid out in table rows, insert | ||||||
|  |                     // the remove button into the last table cell: | ||||||
|  |                     row.children(":last").append('<div><a class="' + options.deleteCssClass + '" href="#">' + options.deleteText + "</a></div>"); | ||||||
|  |                 } else if (row.is("ul") || row.is("ol")) { | ||||||
|  |                     // If they're laid out as an ordered/unordered list, | ||||||
|  |                     // insert an <li> after the last list item: | ||||||
|  |                     row.append('<li><a class="' + options.deleteCssClass + '" href="#">' + options.deleteText + "</a></li>"); | ||||||
|  |                 } else { | ||||||
|  |                     // Otherwise, just insert the remove button as the | ||||||
|  |                     // last child element of the form's container: | ||||||
|  |                     row.children(":first").append('<span><a class="' + options.deleteCssClass + '" href="#">' + options.deleteText + "</a></span>"); | ||||||
|  |                 } | ||||||
|  |                 row.find("*").each(function() { | ||||||
|  |                     updateElementIndex(this, options.prefix, totalForms.val()); | ||||||
|  |                 }); | ||||||
|  |                 // Insert the new form when it has been fully edited | ||||||
|  |                 row.insertBefore($(template)); | ||||||
|  |                 // Update number of total forms | ||||||
|  |                 $(totalForms).val(parseInt(totalForms.val(), 10) + 1); | ||||||
|  |                 nextIndex += 1; | ||||||
|  |                 // Hide add button in case we've hit the max, except we want to add infinitely | ||||||
|  |                 if ((maxForms.val() !== '') && (maxForms.val() - totalForms.val()) <= 0) { | ||||||
|  |                     addButton.parent().hide(); | ||||||
|  |                 } | ||||||
|  |                 // The delete button of each row triggers a bunch of other things | ||||||
|  |                 row.find("a." + options.deleteCssClass).on('click', function(e1) { | ||||||
|  |                     e1.preventDefault(); | ||||||
|  |                     // Remove the parent form containing this button: | ||||||
|  |                     row.remove(); | ||||||
|  |                     nextIndex -= 1; | ||||||
|  |                     // If a post-delete callback was provided, call it with the deleted form: | ||||||
|  |                     if (options.removed) { | ||||||
|  |                         options.removed(row); | ||||||
|  |                     } | ||||||
|  |                     $(document).trigger('formset:removed', [row, options.prefix]); | ||||||
|  |                     // Update the TOTAL_FORMS form count. | ||||||
|  |                     var forms = $("." + options.formCssClass); | ||||||
|  |                     $("#id_" + options.prefix + "-TOTAL_FORMS").val(forms.length); | ||||||
|  |                     // Show add button again once we drop below max | ||||||
|  |                     if ((maxForms.val() === '') || (maxForms.val() - forms.length) > 0) { | ||||||
|  |                         addButton.parent().show(); | ||||||
|  |                     } | ||||||
|  |                     // Also, update names and ids for all remaining form controls | ||||||
|  |                     // so they remain in sequence: | ||||||
|  |                     var i, formCount; | ||||||
|  |                     var updateElementCallback = function() { | ||||||
|  |                         updateElementIndex(this, options.prefix, i); | ||||||
|  |                     }; | ||||||
|  |                     for (i = 0, formCount = forms.length; i < formCount; i++) { | ||||||
|  |                         updateElementIndex($(forms).get(i), options.prefix, i); | ||||||
|  |                         $(forms.get(i)).find("*").each(updateElementCallback); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |                 // If a post-add callback was supplied, call it with the added form: | ||||||
|  |                 if (options.added) { | ||||||
|  |                     options.added(row); | ||||||
|  |                 } | ||||||
|  |                 $(document).trigger('formset:added', [row, options.prefix]); | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |         return this; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     /* Setup plugin defaults */ | ||||||
|  |     $.fn.formset.defaults = { | ||||||
|  |         prefix: "form", // The form prefix for your django formset | ||||||
|  |         addText: "add another", // Text for the add link | ||||||
|  |         deleteText: "remove", // Text for the delete link | ||||||
|  |         addCssClass: "add-row", // CSS class applied to the add link | ||||||
|  |         deleteCssClass: "delete-row", // CSS class applied to the delete link | ||||||
|  |         emptyCssClass: "empty-row", // CSS class applied to the empty row | ||||||
|  |         formCssClass: "dynamic-form", // CSS class applied to each form in a formset | ||||||
|  |         added: null, // Function called each time a new form is added | ||||||
|  |         removed: null, // Function called each time a form is deleted | ||||||
|  |         addButton: null // Existing add button to use | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     // Tabular inlines --------------------------------------------------------- | ||||||
|  |     $.fn.tabularFormset = function(selector, options) { | ||||||
|  |         var $rows = $(this); | ||||||
|  |         var alternatingRows = function(row) { | ||||||
|  |             $(selector).not(".add-row").removeClass("row1 row2") | ||||||
|  |                 .filter(":even").addClass("row1").end() | ||||||
|  |                 .filter(":odd").addClass("row2"); | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         var reinitDateTimeShortCuts = function() { | ||||||
|  |             // Reinitialize the calendar and clock widgets by force | ||||||
|  |             if (typeof DateTimeShortcuts !== "undefined") { | ||||||
|  |                 $(".datetimeshortcuts").remove(); | ||||||
|  |                 DateTimeShortcuts.init(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         var updateSelectFilter = function() { | ||||||
|  |             // If any SelectFilter widgets are a part of the new form, | ||||||
|  |             // instantiate a new SelectFilter instance for it. | ||||||
|  |             if (typeof SelectFilter !== 'undefined') { | ||||||
|  |                 $('.selectfilter').each(function(index, value) { | ||||||
|  |                     var namearr = value.name.split('-'); | ||||||
|  |                     SelectFilter.init(value.id, namearr[namearr.length - 1], false); | ||||||
|  |                 }); | ||||||
|  |                 $('.selectfilterstacked').each(function(index, value) { | ||||||
|  |                     var namearr = value.name.split('-'); | ||||||
|  |                     SelectFilter.init(value.id, namearr[namearr.length - 1], true); | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         var initPrepopulatedFields = function(row) { | ||||||
|  |             row.find('.prepopulated_field').each(function() { | ||||||
|  |                 var field = $(this), | ||||||
|  |                     input = field.find('input, select, textarea'), | ||||||
|  |                     dependency_list = input.data('dependency_list') || [], | ||||||
|  |                     dependencies = []; | ||||||
|  |                 $.each(dependency_list, function(i, field_name) { | ||||||
|  |                     dependencies.push('#' + row.find('.field-' + field_name).find('input, select, textarea').attr('id')); | ||||||
|  |                 }); | ||||||
|  |                 if (dependencies.length) { | ||||||
|  |                     input.prepopulate(dependencies, input.attr('maxlength')); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         $rows.formset({ | ||||||
|  |             prefix: options.prefix, | ||||||
|  |             addText: options.addText, | ||||||
|  |             formCssClass: "dynamic-" + options.prefix, | ||||||
|  |             deleteCssClass: "inline-deletelink", | ||||||
|  |             deleteText: options.deleteText, | ||||||
|  |             emptyCssClass: "empty-form", | ||||||
|  |             removed: alternatingRows, | ||||||
|  |             added: function(row) { | ||||||
|  |                 initPrepopulatedFields(row); | ||||||
|  |                 reinitDateTimeShortCuts(); | ||||||
|  |                 updateSelectFilter(); | ||||||
|  |                 alternatingRows(row); | ||||||
|  |             }, | ||||||
|  |             addButton: options.addButton | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         return $rows; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // Stacked inlines --------------------------------------------------------- | ||||||
|  |     $.fn.stackedFormset = function(selector, options) { | ||||||
|  |         var $rows = $(this); | ||||||
|  |         var updateInlineLabel = function(row) { | ||||||
|  |             $(selector).find(".inline_label").each(function(i) { | ||||||
|  |                 var count = i + 1; | ||||||
|  |                 $(this).html($(this).html().replace(/(#\d+)/g, "#" + count)); | ||||||
|  |             }); | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         var reinitDateTimeShortCuts = function() { | ||||||
|  |             // Reinitialize the calendar and clock widgets by force, yuck. | ||||||
|  |             if (typeof DateTimeShortcuts !== "undefined") { | ||||||
|  |                 $(".datetimeshortcuts").remove(); | ||||||
|  |                 DateTimeShortcuts.init(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         var updateSelectFilter = function() { | ||||||
|  |             // If any SelectFilter widgets were added, instantiate a new instance. | ||||||
|  |             if (typeof SelectFilter !== "undefined") { | ||||||
|  |                 $(".selectfilter").each(function(index, value) { | ||||||
|  |                     var namearr = value.name.split('-'); | ||||||
|  |                     SelectFilter.init(value.id, namearr[namearr.length - 1], false); | ||||||
|  |                 }); | ||||||
|  |                 $(".selectfilterstacked").each(function(index, value) { | ||||||
|  |                     var namearr = value.name.split('-'); | ||||||
|  |                     SelectFilter.init(value.id, namearr[namearr.length - 1], true); | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         var initPrepopulatedFields = function(row) { | ||||||
|  |             row.find('.prepopulated_field').each(function() { | ||||||
|  |                 var field = $(this), | ||||||
|  |                     input = field.find('input, select, textarea'), | ||||||
|  |                     dependency_list = input.data('dependency_list') || [], | ||||||
|  |                     dependencies = []; | ||||||
|  |                 $.each(dependency_list, function(i, field_name) { | ||||||
|  |                     dependencies.push('#' + row.find('.form-row .field-' + field_name).find('input, select, textarea').attr('id')); | ||||||
|  |                 }); | ||||||
|  |                 if (dependencies.length) { | ||||||
|  |                     input.prepopulate(dependencies, input.attr('maxlength')); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         $rows.formset({ | ||||||
|  |             prefix: options.prefix, | ||||||
|  |             addText: options.addText, | ||||||
|  |             formCssClass: "dynamic-" + options.prefix, | ||||||
|  |             deleteCssClass: "inline-deletelink", | ||||||
|  |             deleteText: options.deleteText, | ||||||
|  |             emptyCssClass: "empty-form", | ||||||
|  |             removed: updateInlineLabel, | ||||||
|  |             added: function(row) { | ||||||
|  |                 initPrepopulatedFields(row); | ||||||
|  |                 reinitDateTimeShortCuts(); | ||||||
|  |                 updateSelectFilter(); | ||||||
|  |                 updateInlineLabel(row); | ||||||
|  |             }, | ||||||
|  |             addButton: options.addButton | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         return $rows; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     $(document).ready(function() { | ||||||
|  |         $(".js-inline-admin-formset").each(function() { | ||||||
|  |             var data = $(this).data(), | ||||||
|  |                 inlineOptions = data.inlineFormset, | ||||||
|  |                 selector; | ||||||
|  |             switch(data.inlineType) { | ||||||
|  |             case "stacked": | ||||||
|  |                 selector = inlineOptions.name + "-group .inline-related"; | ||||||
|  |                 $(selector).stackedFormset(selector, inlineOptions.options); | ||||||
|  |                 break; | ||||||
|  |             case "tabular": | ||||||
|  |                 selector = inlineOptions.name + "-group .tabular.inline-related tbody:first > tr"; | ||||||
|  |                 $(selector).tabularFormset(selector, inlineOptions.options); | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										10
									
								
								CalibreWebCompanion/static/admin/js/inlines.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | (function(b){b.fn.formset=function(c){var a=b.extend({},b.fn.formset.defaults,c),d=b(this);c=d.parent();var h=function(a,e,f){var c=new RegExp("("+e+"-(\\d+|__prefix__))");e=e+"-"+f;b(a).prop("for")&&b(a).prop("for",b(a).prop("for").replace(c,e));a.id&&(a.id=a.id.replace(c,e));a.name&&(a.name=a.name.replace(c,e))},g=b("#id_"+a.prefix+"-TOTAL_FORMS").prop("autocomplete","off"),k=parseInt(g.val(),10),e=b("#id_"+a.prefix+"-MAX_NUM_FORMS").prop("autocomplete","off"),f=""===e.val()||0<e.val()-g.val(); | ||||||
|  | d.each(function(e){b(this).not("."+a.emptyCssClass).addClass(a.formCssClass)});if(d.length&&f){var m=a.addButton;null===m&&("TR"===d.prop("tagName")?(d=this.eq(-1).children().length,c.append('<tr class="'+a.addCssClass+'"><td colspan="'+d+'"><a href="#">'+a.addText+"</a></tr>"),m=c.find("tr:last a")):(d.filter(":last").after('<div class="'+a.addCssClass+'"><a href="#">'+a.addText+"</a></div>"),m=d.filter(":last").next().find("a")));m.on("click",function(f){f.preventDefault();f=b("#"+a.prefix+"-empty"); | ||||||
|  | var c=f.clone(!0);c.removeClass(a.emptyCssClass).addClass(a.formCssClass).attr("id",a.prefix+"-"+k);c.is("tr")?c.children(":last").append('<div><a class="'+a.deleteCssClass+'" href="#">'+a.deleteText+"</a></div>"):c.is("ul")||c.is("ol")?c.append('<li><a class="'+a.deleteCssClass+'" href="#">'+a.deleteText+"</a></li>"):c.children(":first").append('<span><a class="'+a.deleteCssClass+'" href="#">'+a.deleteText+"</a></span>");c.find("*").each(function(){h(this,a.prefix,g.val())});c.insertBefore(b(f)); | ||||||
|  | b(g).val(parseInt(g.val(),10)+1);k+=1;""!==e.val()&&0>=e.val()-g.val()&&m.parent().hide();c.find("a."+a.deleteCssClass).on("click",function(f){f.preventDefault();c.remove();--k;a.removed&&a.removed(c);b(document).trigger("formset:removed",[c,a.prefix]);f=b("."+a.formCssClass);b("#id_"+a.prefix+"-TOTAL_FORMS").val(f.length);(""===e.val()||0<e.val()-f.length)&&m.parent().show();var d,g=function(){h(this,a.prefix,l)};var l=0;for(d=f.length;l<d;l++)h(b(f).get(l),a.prefix,l),b(f.get(l)).find("*").each(g)}); | ||||||
|  | a.added&&a.added(c);b(document).trigger("formset:added",[c,a.prefix])})}return this};b.fn.formset.defaults={prefix:"form",addText:"add another",deleteText:"remove",addCssClass:"add-row",deleteCssClass:"delete-row",emptyCssClass:"empty-row",formCssClass:"dynamic-form",added:null,removed:null,addButton:null};b.fn.tabularFormset=function(c,a){var d=b(this),h=function(a){b(c).not(".add-row").removeClass("row1 row2").filter(":even").addClass("row1").end().filter(":odd").addClass("row2")},g=function(){"undefined"!== | ||||||
|  | typeof SelectFilter&&(b(".selectfilter").each(function(a,b){a=b.name.split("-");SelectFilter.init(b.id,a[a.length-1],!1)}),b(".selectfilterstacked").each(function(a,b){a=b.name.split("-");SelectFilter.init(b.id,a[a.length-1],!0)}))},k=function(a){a.find(".prepopulated_field").each(function(){var c=b(this).find("input, select, textarea"),e=c.data("dependency_list")||[],d=[];b.each(e,function(b,c){d.push("#"+a.find(".field-"+c).find("input, select, textarea").attr("id"))});d.length&&c.prepopulate(d, | ||||||
|  | c.attr("maxlength"))})};d.formset({prefix:a.prefix,addText:a.addText,formCssClass:"dynamic-"+a.prefix,deleteCssClass:"inline-deletelink",deleteText:a.deleteText,emptyCssClass:"empty-form",removed:h,added:function(a){k(a);"undefined"!==typeof DateTimeShortcuts&&(b(".datetimeshortcuts").remove(),DateTimeShortcuts.init());g();h(a)},addButton:a.addButton});return d};b.fn.stackedFormset=function(c,a){var d=b(this),h=function(a){b(c).find(".inline_label").each(function(a){a+=1;b(this).html(b(this).html().replace(/(#\d+)/g, | ||||||
|  | "#"+a))})},g=function(){"undefined"!==typeof SelectFilter&&(b(".selectfilter").each(function(a,b){a=b.name.split("-");SelectFilter.init(b.id,a[a.length-1],!1)}),b(".selectfilterstacked").each(function(a,b){a=b.name.split("-");SelectFilter.init(b.id,a[a.length-1],!0)}))},k=function(a){a.find(".prepopulated_field").each(function(){var c=b(this).find("input, select, textarea"),d=c.data("dependency_list")||[],e=[];b.each(d,function(b,c){e.push("#"+a.find(".form-row .field-"+c).find("input, select, textarea").attr("id"))}); | ||||||
|  | e.length&&c.prepopulate(e,c.attr("maxlength"))})};d.formset({prefix:a.prefix,addText:a.addText,formCssClass:"dynamic-"+a.prefix,deleteCssClass:"inline-deletelink",deleteText:a.deleteText,emptyCssClass:"empty-form",removed:h,added:function(a){k(a);"undefined"!==typeof DateTimeShortcuts&&(b(".datetimeshortcuts").remove(),DateTimeShortcuts.init());g();h(a)},addButton:a.addButton});return d};b(document).ready(function(){b(".js-inline-admin-formset").each(function(){var c=b(this).data(),a=c.inlineFormset; | ||||||
|  | switch(c.inlineType){case "stacked":c=a.name+"-group .inline-related";b(c).stackedFormset(c,a.options);break;case "tabular":c=a.name+"-group .tabular.inline-related tbody:first > tr",b(c).tabularFormset(c,a.options)}})})})(django.jQuery); | ||||||
							
								
								
									
										8
									
								
								CalibreWebCompanion/static/admin/js/jquery.init.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | |||||||
|  | /*global django:true, jQuery:false*/ | ||||||
|  | /* Puts the included jQuery into our own namespace using noConflict and passing | ||||||
|  |  * it 'true'. This ensures that the included jQuery doesn't pollute the global | ||||||
|  |  * namespace (i.e. this preserves pre-existing values for both window.$ and | ||||||
|  |  * window.jQuery). | ||||||
|  |  */ | ||||||
|  | var django = django || {}; | ||||||
|  | django.jQuery = jQuery.noConflict(true); | ||||||
							
								
								
									
										16
									
								
								CalibreWebCompanion/static/admin/js/popup_response.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | |||||||
|  | /*global opener */ | ||||||
|  | (function() { | ||||||
|  |     'use strict'; | ||||||
|  |     var initData = JSON.parse(document.getElementById('django-admin-popup-response-constants').dataset.popupResponse); | ||||||
|  |     switch(initData.action) { | ||||||
|  |     case 'change': | ||||||
|  |         opener.dismissChangeRelatedObjectPopup(window, initData.value, initData.obj, initData.new_value); | ||||||
|  |         break; | ||||||
|  |     case 'delete': | ||||||
|  |         opener.dismissDeleteRelatedObjectPopup(window, initData.value); | ||||||
|  |         break; | ||||||
|  |     default: | ||||||
|  |         opener.dismissAddRelatedObjectPopup(window, initData.value, initData.obj); | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | })(); | ||||||
							
								
								
									
										42
									
								
								CalibreWebCompanion/static/admin/js/prepopulate.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | |||||||
|  | /*global URLify*/ | ||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     $.fn.prepopulate = function(dependencies, maxLength, allowUnicode) { | ||||||
|  |         /* | ||||||
|  |             Depends on urlify.js | ||||||
|  |             Populates a selected field with the values of the dependent fields, | ||||||
|  |             URLifies and shortens the string. | ||||||
|  |             dependencies - array of dependent fields ids | ||||||
|  |             maxLength - maximum length of the URLify'd string | ||||||
|  |             allowUnicode - Unicode support of the URLify'd string | ||||||
|  |         */ | ||||||
|  |         return this.each(function() { | ||||||
|  |             var prepopulatedField = $(this); | ||||||
|  |  | ||||||
|  |             var populate = function() { | ||||||
|  |                 // Bail if the field's value has been changed by the user | ||||||
|  |                 if (prepopulatedField.data('_changed')) { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 var values = []; | ||||||
|  |                 $.each(dependencies, function(i, field) { | ||||||
|  |                     field = $(field); | ||||||
|  |                     if (field.val().length > 0) { | ||||||
|  |                         values.push(field.val()); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  |                 prepopulatedField.val(URLify(values.join(' '), maxLength, allowUnicode)); | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             prepopulatedField.data('_changed', false); | ||||||
|  |             prepopulatedField.on('change', function() { | ||||||
|  |                 prepopulatedField.data('_changed', true); | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             if (!prepopulatedField.val()) { | ||||||
|  |                 $(dependencies.join(',')).on('keyup change focus', populate); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     }; | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										1
									
								
								CalibreWebCompanion/static/admin/js/prepopulate.min.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | (function(b){b.fn.prepopulate=function(d,f,g){return this.each(function(){var a=b(this),h=function(){if(!a.data("_changed")){var e=[];b.each(d,function(a,c){c=b(c);0<c.val().length&&e.push(c.val())});a.val(URLify(e.join(" "),f,g))}};a.data("_changed",!1);a.on("change",function(){a.data("_changed",!0)});if(!a.val())b(d.join(",")).on("keyup change focus",h)})}})(django.jQuery); | ||||||
							
								
								
									
										10
									
								
								CalibreWebCompanion/static/admin/js/prepopulate_init.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | (function($) { | ||||||
|  |     'use strict'; | ||||||
|  |     var fields = $('#django-admin-prepopulated-fields-constants').data('prepopulatedFields'); | ||||||
|  |     $.each(fields, function(index, field) { | ||||||
|  |         $('.empty-form .form-row .field-' + field.name + ', .empty-form.form-row .field-' + field.name).addClass('prepopulated_field'); | ||||||
|  |         $(field.id).data('dependency_list', field.dependency_list).prepopulate( | ||||||
|  |             field.dependency_ids, field.maxLength, field.allowUnicode | ||||||
|  |         ); | ||||||
|  |     }); | ||||||
|  | })(django.jQuery); | ||||||
							
								
								
									
										195
									
								
								CalibreWebCompanion/static/admin/js/urlify.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,195 @@ | |||||||
|  | /*global XRegExp*/ | ||||||
|  | (function() { | ||||||
|  |     'use strict'; | ||||||
|  |  | ||||||
|  |     var LATIN_MAP = { | ||||||
|  |         'À': 'A', 'Á': 'A', 'Â': 'A', 'Ã': 'A', 'Ä': 'A', 'Å': 'A', 'Æ': 'AE', | ||||||
|  |         'Ç': 'C', 'È': 'E', 'É': 'E', 'Ê': 'E', 'Ë': 'E', 'Ì': 'I', 'Í': 'I', | ||||||
|  |         'Î': 'I', 'Ï': 'I', 'Ð': 'D', 'Ñ': 'N', 'Ò': 'O', 'Ó': 'O', 'Ô': 'O', | ||||||
|  |         'Õ': 'O', 'Ö': 'O', 'Ő': 'O', 'Ø': 'O', 'Ù': 'U', 'Ú': 'U', 'Û': 'U', | ||||||
|  |         'Ü': 'U', 'Ű': 'U', 'Ý': 'Y', 'Þ': 'TH', 'Ÿ': 'Y', 'ß': 'ss', 'à': 'a', | ||||||
|  |         'á': 'a', 'â': 'a', 'ã': 'a', 'ä': 'a', 'å': 'a', 'æ': 'ae', 'ç': 'c', | ||||||
|  |         'è': 'e', 'é': 'e', 'ê': 'e', 'ë': 'e', 'ì': 'i', 'í': 'i', 'î': 'i', | ||||||
|  |         'ï': 'i', 'ð': 'd', 'ñ': 'n', 'ò': 'o', 'ó': 'o', 'ô': 'o', 'õ': 'o', | ||||||
|  |         'ö': 'o', 'ő': 'o', 'ø': 'o', 'ù': 'u', 'ú': 'u', 'û': 'u', 'ü': 'u', | ||||||
|  |         'ű': 'u', 'ý': 'y', 'þ': 'th', 'ÿ': 'y' | ||||||
|  |     }; | ||||||
|  |     var LATIN_SYMBOLS_MAP = { | ||||||
|  |         '©': '(c)' | ||||||
|  |     }; | ||||||
|  |     var GREEK_MAP = { | ||||||
|  |         'α': 'a', 'β': 'b', 'γ': 'g', 'δ': 'd', 'ε': 'e', 'ζ': 'z', 'η': 'h', | ||||||
|  |         'θ': '8', 'ι': 'i', 'κ': 'k', 'λ': 'l', 'μ': 'm', 'ν': 'n', 'ξ': '3', | ||||||
|  |         'ο': 'o', 'π': 'p', 'ρ': 'r', 'σ': 's', 'τ': 't', 'υ': 'y', 'φ': 'f', | ||||||
|  |         'χ': 'x', 'ψ': 'ps', 'ω': 'w', 'ά': 'a', 'έ': 'e', 'ί': 'i', 'ό': 'o', | ||||||
|  |         'ύ': 'y', 'ή': 'h', 'ώ': 'w', 'ς': 's', 'ϊ': 'i', 'ΰ': 'y', 'ϋ': 'y', | ||||||
|  |         'ΐ': 'i', 'Α': 'A', 'Β': 'B', 'Γ': 'G', 'Δ': 'D', 'Ε': 'E', 'Ζ': 'Z', | ||||||
|  |         'Η': 'H', 'Θ': '8', 'Ι': 'I', 'Κ': 'K', 'Λ': 'L', 'Μ': 'M', 'Ν': 'N', | ||||||
|  |         'Ξ': '3', 'Ο': 'O', 'Π': 'P', 'Ρ': 'R', 'Σ': 'S', 'Τ': 'T', 'Υ': 'Y', | ||||||
|  |         'Φ': 'F', 'Χ': 'X', 'Ψ': 'PS', 'Ω': 'W', 'Ά': 'A', 'Έ': 'E', 'Ί': 'I', | ||||||
|  |         'Ό': 'O', 'Ύ': 'Y', 'Ή': 'H', 'Ώ': 'W', 'Ϊ': 'I', 'Ϋ': 'Y' | ||||||
|  |     }; | ||||||
|  |     var TURKISH_MAP = { | ||||||
|  |         'ş': 's', 'Ş': 'S', 'ı': 'i', 'İ': 'I', 'ç': 'c', 'Ç': 'C', 'ü': 'u', | ||||||
|  |         'Ü': 'U', 'ö': 'o', 'Ö': 'O', 'ğ': 'g', 'Ğ': 'G' | ||||||
|  |     }; | ||||||
|  |     var ROMANIAN_MAP = { | ||||||
|  |         'ă': 'a', 'î': 'i', 'ș': 's', 'ț': 't', 'â': 'a', | ||||||
|  |         'Ă': 'A', 'Î': 'I', 'Ș': 'S', 'Ț': 'T', 'Â': 'A' | ||||||
|  |     }; | ||||||
|  |     var RUSSIAN_MAP = { | ||||||
|  |         'а': 'a', 'б': 'b', 'в': 'v', 'г': 'g', 'д': 'd', 'е': 'e', 'ё': 'yo', | ||||||
|  |         'ж': 'zh', 'з': 'z', 'и': 'i', 'й': 'j', 'к': 'k', 'л': 'l', 'м': 'm', | ||||||
|  |         'н': 'n', 'о': 'o', 'п': 'p', 'р': 'r', 'с': 's', 'т': 't', 'у': 'u', | ||||||
|  |         'ф': 'f', 'х': 'h', 'ц': 'c', 'ч': 'ch', 'ш': 'sh', 'щ': 'sh', 'ъ': '', | ||||||
|  |         'ы': 'y', 'ь': '', 'э': 'e', 'ю': 'yu', 'я': 'ya', | ||||||
|  |         'А': 'A', 'Б': 'B', 'В': 'V', 'Г': 'G', 'Д': 'D', 'Е': 'E', 'Ё': 'Yo', | ||||||
|  |         'Ж': 'Zh', 'З': 'Z', 'И': 'I', 'Й': 'J', 'К': 'K', 'Л': 'L', 'М': 'M', | ||||||
|  |         'Н': 'N', 'О': 'O', 'П': 'P', 'Р': 'R', 'С': 'S', 'Т': 'T', 'У': 'U', | ||||||
|  |         'Ф': 'F', 'Х': 'H', 'Ц': 'C', 'Ч': 'Ch', 'Ш': 'Sh', 'Щ': 'Sh', 'Ъ': '', | ||||||
|  |         'Ы': 'Y', 'Ь': '', 'Э': 'E', 'Ю': 'Yu', 'Я': 'Ya' | ||||||
|  |     }; | ||||||
|  |     var UKRAINIAN_MAP = { | ||||||
|  |         'Є': 'Ye', 'І': 'I', 'Ї': 'Yi', 'Ґ': 'G', 'є': 'ye', 'і': 'i', | ||||||
|  |         'ї': 'yi', 'ґ': 'g' | ||||||
|  |     }; | ||||||
|  |     var CZECH_MAP = { | ||||||
|  |         'č': 'c', 'ď': 'd', 'ě': 'e', 'ň': 'n', 'ř': 'r', 'š': 's', 'ť': 't', | ||||||
|  |         'ů': 'u', 'ž': 'z', 'Č': 'C', 'Ď': 'D', 'Ě': 'E', 'Ň': 'N', 'Ř': 'R', | ||||||
|  |         'Š': 'S', 'Ť': 'T', 'Ů': 'U', 'Ž': 'Z' | ||||||
|  |     }; | ||||||
|  |     var SLOVAK_MAP = { | ||||||
|  |         'á': 'a', 'ä': 'a', 'č': 'c', 'ď': 'd', 'é': 'e', 'í': 'i', 'ľ': 'l', | ||||||
|  |         'ĺ': 'l', 'ň': 'n', 'ó': 'o', 'ô': 'o', 'ŕ': 'r', 'š': 's', 'ť': 't', | ||||||
|  |         'ú': 'u', 'ý': 'y', 'ž': 'z', | ||||||
|  |         'Á': 'a', 'Ä': 'A', 'Č': 'C', 'Ď': 'D', 'É': 'E', 'Í': 'I', 'Ľ': 'L', | ||||||
|  |         'Ĺ': 'L', 'Ň': 'N', 'Ó': 'O', 'Ô': 'O', 'Ŕ': 'R', 'Š': 'S', 'Ť': 'T', | ||||||
|  |         'Ú': 'U', 'Ý': 'Y', 'Ž': 'Z' | ||||||
|  |     }; | ||||||
|  |     var POLISH_MAP = { | ||||||
|  |         'ą': 'a', 'ć': 'c', 'ę': 'e', 'ł': 'l', 'ń': 'n', 'ó': 'o', 'ś': 's', | ||||||
|  |         'ź': 'z', 'ż': 'z', | ||||||
|  |         'Ą': 'A', 'Ć': 'C', 'Ę': 'E', 'Ł': 'L', 'Ń': 'N', 'Ó': 'O', 'Ś': 'S', | ||||||
|  |         'Ź': 'Z', 'Ż': 'Z' | ||||||
|  |     }; | ||||||
|  |     var LATVIAN_MAP = { | ||||||
|  |         'ā': 'a', 'č': 'c', 'ē': 'e', 'ģ': 'g', 'ī': 'i', 'ķ': 'k', 'ļ': 'l', | ||||||
|  |         'ņ': 'n', 'š': 's', 'ū': 'u', 'ž': 'z', | ||||||
|  |         'Ā': 'A', 'Č': 'C', 'Ē': 'E', 'Ģ': 'G', 'Ī': 'I', 'Ķ': 'K', 'Ļ': 'L', | ||||||
|  |         'Ņ': 'N', 'Š': 'S', 'Ū': 'U', 'Ž': 'Z' | ||||||
|  |     }; | ||||||
|  |     var ARABIC_MAP = { | ||||||
|  |         'أ': 'a', 'ب': 'b', 'ت': 't', 'ث': 'th', 'ج': 'g', 'ح': 'h', 'خ': 'kh', 'د': 'd', | ||||||
|  |         'ذ': 'th', 'ر': 'r', 'ز': 'z', 'س': 's', 'ش': 'sh', 'ص': 's', 'ض': 'd', 'ط': 't', | ||||||
|  |         'ظ': 'th', 'ع': 'aa', 'غ': 'gh', 'ف': 'f', 'ق': 'k', 'ك': 'k', 'ل': 'l', 'م': 'm', | ||||||
|  |         'ن': 'n', 'ه': 'h', 'و': 'o', 'ي': 'y' | ||||||
|  |     }; | ||||||
|  |     var LITHUANIAN_MAP = { | ||||||
|  |         'ą': 'a', 'č': 'c', 'ę': 'e', 'ė': 'e', 'į': 'i', 'š': 's', 'ų': 'u', | ||||||
|  |         'ū': 'u', 'ž': 'z', | ||||||
|  |         'Ą': 'A', 'Č': 'C', 'Ę': 'E', 'Ė': 'E', 'Į': 'I', 'Š': 'S', 'Ų': 'U', | ||||||
|  |         'Ū': 'U', 'Ž': 'Z' | ||||||
|  |     }; | ||||||
|  |     var SERBIAN_MAP = { | ||||||
|  |         'ђ': 'dj', 'ј': 'j', 'љ': 'lj', 'њ': 'nj', 'ћ': 'c', 'џ': 'dz', | ||||||
|  |         'đ': 'dj', 'Ђ': 'Dj', 'Ј': 'j', 'Љ': 'Lj', 'Њ': 'Nj', 'Ћ': 'C', | ||||||
|  |         'Џ': 'Dz', 'Đ': 'Dj' | ||||||
|  |     }; | ||||||
|  |     var AZERBAIJANI_MAP = { | ||||||
|  |         'ç': 'c', 'ə': 'e', 'ğ': 'g', 'ı': 'i', 'ö': 'o', 'ş': 's', 'ü': 'u', | ||||||
|  |         'Ç': 'C', 'Ə': 'E', 'Ğ': 'G', 'İ': 'I', 'Ö': 'O', 'Ş': 'S', 'Ü': 'U' | ||||||
|  |     }; | ||||||
|  |     var GEORGIAN_MAP = { | ||||||
|  |         'ა': 'a', 'ბ': 'b', 'გ': 'g', 'დ': 'd', 'ე': 'e', 'ვ': 'v', 'ზ': 'z', | ||||||
|  |         'თ': 't', 'ი': 'i', 'კ': 'k', 'ლ': 'l', 'მ': 'm', 'ნ': 'n', 'ო': 'o', | ||||||
|  |         'პ': 'p', 'ჟ': 'j', 'რ': 'r', 'ს': 's', 'ტ': 't', 'უ': 'u', 'ფ': 'f', | ||||||
|  |         'ქ': 'q', 'ღ': 'g', 'ყ': 'y', 'შ': 'sh', 'ჩ': 'ch', 'ც': 'c', 'ძ': 'dz', | ||||||
|  |         'წ': 'w', 'ჭ': 'ch', 'ხ': 'x', 'ჯ': 'j', 'ჰ': 'h' | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     var ALL_DOWNCODE_MAPS = [ | ||||||
|  |         LATIN_MAP, | ||||||
|  |         LATIN_SYMBOLS_MAP, | ||||||
|  |         GREEK_MAP, | ||||||
|  |         TURKISH_MAP, | ||||||
|  |         ROMANIAN_MAP, | ||||||
|  |         RUSSIAN_MAP, | ||||||
|  |         UKRAINIAN_MAP, | ||||||
|  |         CZECH_MAP, | ||||||
|  |         SLOVAK_MAP, | ||||||
|  |         POLISH_MAP, | ||||||
|  |         LATVIAN_MAP, | ||||||
|  |         ARABIC_MAP, | ||||||
|  |         LITHUANIAN_MAP, | ||||||
|  |         SERBIAN_MAP, | ||||||
|  |         AZERBAIJANI_MAP, | ||||||
|  |         GEORGIAN_MAP | ||||||
|  |     ]; | ||||||
|  |  | ||||||
|  |     var Downcoder = { | ||||||
|  |         'Initialize': function() { | ||||||
|  |             if (Downcoder.map) { // already made | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             Downcoder.map = {}; | ||||||
|  |             Downcoder.chars = []; | ||||||
|  |             for (var i = 0; i < ALL_DOWNCODE_MAPS.length; i++) { | ||||||
|  |                 var lookup = ALL_DOWNCODE_MAPS[i]; | ||||||
|  |                 for (var c in lookup) { | ||||||
|  |                     if (lookup.hasOwnProperty(c)) { | ||||||
|  |                         Downcoder.map[c] = lookup[c]; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             for (var k in Downcoder.map) { | ||||||
|  |                 if (Downcoder.map.hasOwnProperty(k)) { | ||||||
|  |                     Downcoder.chars.push(k); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             Downcoder.regex = new RegExp(Downcoder.chars.join('|'), 'g'); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     function downcode(slug) { | ||||||
|  |         Downcoder.Initialize(); | ||||||
|  |         return slug.replace(Downcoder.regex, function(m) { | ||||||
|  |             return Downcoder.map[m]; | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     function URLify(s, num_chars, allowUnicode) { | ||||||
|  |         // changes, e.g., "Petty theft" to "petty-theft" | ||||||
|  |         // remove all these words from the string before urlifying | ||||||
|  |         if (!allowUnicode) { | ||||||
|  |             s = downcode(s); | ||||||
|  |         } | ||||||
|  |         var hasUnicodeChars = /[^\u0000-\u007f]/.test(s); | ||||||
|  |         // Remove English words only if the string contains ASCII (English) | ||||||
|  |         // characters. | ||||||
|  |         if (!hasUnicodeChars) { | ||||||
|  |             var removeList = [ | ||||||
|  |                 "a", "an", "as", "at", "before", "but", "by", "for", "from", | ||||||
|  |                 "is", "in", "into", "like", "of", "off", "on", "onto", "per", | ||||||
|  |                 "since", "than", "the", "this", "that", "to", "up", "via", | ||||||
|  |                 "with" | ||||||
|  |             ]; | ||||||
|  |             var r = new RegExp('\\b(' + removeList.join('|') + ')\\b', 'gi'); | ||||||
|  |             s = s.replace(r, ''); | ||||||
|  |         } | ||||||
|  |         // if downcode doesn't hit, the char will be stripped here | ||||||
|  |         if (allowUnicode) { | ||||||
|  |             // Keep Unicode letters including both lowercase and uppercase | ||||||
|  |             // characters, whitespace, and dash; remove other characters. | ||||||
|  |             s = XRegExp.replace(s, XRegExp('[^-_\\p{L}\\p{N}\\s]', 'g'), ''); | ||||||
|  |         } else { | ||||||
|  |             s = s.replace(/[^-\w\s]/g, ''); // remove unneeded chars | ||||||
|  |         } | ||||||
|  |         s = s.replace(/^\s+|\s+$/g, ''); // trim leading/trailing spaces | ||||||
|  |         s = s.replace(/[-\s]+/g, '-'); // convert spaces to hyphens | ||||||
|  |         s = s.substring(0, num_chars); // trim to first num_chars chars | ||||||
|  |         s = s.replace(/-+$/g, ''); // trim any trailing hyphens | ||||||
|  |         return s.toLowerCase(); // convert to lowercase | ||||||
|  |     } | ||||||
|  |     window.URLify = URLify; | ||||||
|  | })(); | ||||||
							
								
								
									
										20
									
								
								CalibreWebCompanion/static/admin/js/vendor/jquery/LICENSE.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | |||||||
|  | Copyright JS Foundation and other contributors, https://js.foundation/ | ||||||
|  |  | ||||||
|  | Permission is hereby granted, free of charge, to any person obtaining | ||||||
|  | a copy of this software and associated documentation files (the | ||||||
|  | "Software"), to deal in the Software without restriction, including | ||||||
|  | without limitation the rights to use, copy, modify, merge, publish, | ||||||
|  | distribute, sublicense, and/or sell copies of the Software, and to | ||||||
|  | permit persons to whom the Software is furnished to do so, subject to | ||||||
|  | the following conditions: | ||||||
|  |  | ||||||
|  | The above copyright notice and this permission notice shall be | ||||||
|  | included in all copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||||||
|  | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | ||||||
|  | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||||||
|  | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE | ||||||
|  | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION | ||||||
|  | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION | ||||||
|  | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | ||||||