fastadmin 0.1.32__tar.gz → 0.1.34__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {fastadmin-0.1.32 → fastadmin-0.1.34}/PKG-INFO +1 -1
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/api.py +2 -2
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/api.py +1 -2
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/api.py +2 -2
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/helpers.py +8 -33
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/service.py +34 -1
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/base.py +31 -33
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/helpers.py +8 -1
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/django.py +11 -2
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/ponyorm.py +5 -2
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/sqlalchemy.py +5 -2
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/tortoise.py +5 -2
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/schemas.py +5 -0
- fastadmin-0.1.34/fastadmin/static/css/main.min.css +8 -0
- fastadmin-0.1.34/fastadmin/static/css/main.min.css.map +1 -0
- fastadmin-0.1.34/fastadmin/static/js/main.min.js +3 -0
- fastadmin-0.1.34/fastadmin/static/js/main.min.js.map +1 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/pyproject.toml +1 -1
- fastadmin-0.1.32/fastadmin/static/css/main.min.css +0 -7
- fastadmin-0.1.32/fastadmin/static/css/main.min.css.map +0 -1
- fastadmin-0.1.32/fastadmin/static/js/main.min.js +0 -3
- fastadmin-0.1.32/fastadmin/static/js/main.min.js.map +0 -1
- {fastadmin-0.1.32 → fastadmin-0.1.34}/LICENSE +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/README.md +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/exceptions.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/urls.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/views.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/app.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/views.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/app.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/views.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/schemas.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/decorators.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/__init__.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/settings.py +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/images/favicon.png +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/images/header-logo.svg +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/images/sign-in-logo.svg +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/js/787.cda612ba.chunk.js +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/js/787.cda612ba.chunk.js.map +0 -0
- {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/templates/index.html +0 -0
@@ -11,9 +11,9 @@ from django.http import StreamingHttpResponse
|
|
11
11
|
from django.http.request import HttpRequest
|
12
12
|
|
13
13
|
from fastadmin.api.exceptions import AdminApiException
|
14
|
-
from fastadmin.api.helpers import
|
14
|
+
from fastadmin.api.helpers import is_valid_id
|
15
15
|
from fastadmin.api.schemas import ActionInputSchema, ExportInputSchema, ListQuerySchema, SignInInputSchema
|
16
|
-
from fastadmin.api.service import ApiService
|
16
|
+
from fastadmin.api.service import ApiService, get_user_id_from_session_id
|
17
17
|
from fastadmin.settings import settings
|
18
18
|
|
19
19
|
logger = logging.getLogger(__name__)
|
@@ -6,9 +6,8 @@ from fastapi import APIRouter, HTTPException, Request
|
|
6
6
|
from fastapi.responses import Response, StreamingResponse
|
7
7
|
|
8
8
|
from fastadmin.api.exceptions import AdminApiException
|
9
|
-
from fastadmin.api.helpers import get_user_id_from_session_id
|
10
9
|
from fastadmin.api.schemas import ActionInputSchema, ExportInputSchema, SignInInputSchema
|
11
|
-
from fastadmin.api.service import ApiService
|
10
|
+
from fastadmin.api.service import ApiService, get_user_id_from_session_id
|
12
11
|
from fastadmin.models.schemas import ConfigurationSchema
|
13
12
|
from fastadmin.settings import settings
|
14
13
|
|
@@ -5,9 +5,9 @@ from flask import Blueprint, Response, make_response, request
|
|
5
5
|
from werkzeug.exceptions import HTTPException
|
6
6
|
|
7
7
|
from fastadmin.api.exceptions import AdminApiException
|
8
|
-
from fastadmin.api.helpers import
|
8
|
+
from fastadmin.api.helpers import is_valid_id
|
9
9
|
from fastadmin.api.schemas import ActionInputSchema, ExportInputSchema, SignInInputSchema
|
10
|
-
from fastadmin.api.service import ApiService
|
10
|
+
from fastadmin.api.service import ApiService, get_user_id_from_session_id
|
11
11
|
from fastadmin.settings import settings
|
12
12
|
|
13
13
|
logger = logging.getLogger(__name__)
|
@@ -1,11 +1,7 @@
|
|
1
|
-
|
1
|
+
import base64
|
2
2
|
from uuid import UUID
|
3
3
|
|
4
|
-
import jwt
|
5
|
-
|
6
|
-
from fastadmin.models.helpers import get_admin_model
|
7
4
|
from fastadmin.models.schemas import ModelFieldWidgetSchema
|
8
|
-
from fastadmin.settings import settings
|
9
5
|
|
10
6
|
|
11
7
|
def sanitize_filter_value(value: str) -> bool | None | str:
|
@@ -75,34 +71,13 @@ def is_valid_id(id: UUID | int) -> bool:
|
|
75
71
|
return is_digit(str(id)) or is_valid_uuid(str(id))
|
76
72
|
|
77
73
|
|
78
|
-
|
79
|
-
"""
|
74
|
+
def is_valid_base64(s: str) -> bool:
|
75
|
+
"""Check if s is a valid base64.
|
80
76
|
|
81
|
-
:param
|
82
|
-
:return:
|
77
|
+
:param s: A string to test.
|
78
|
+
:return: True if s is a valid base64, False otherwise.
|
83
79
|
"""
|
84
|
-
if not session_id:
|
85
|
-
return None
|
86
|
-
|
87
|
-
admin_model = get_admin_model(settings.ADMIN_USER_MODEL)
|
88
|
-
if not admin_model:
|
89
|
-
return None
|
90
|
-
|
91
80
|
try:
|
92
|
-
|
93
|
-
except
|
94
|
-
return
|
95
|
-
|
96
|
-
session_expired_at = token_payload.get("session_expired_at")
|
97
|
-
if not session_expired_at:
|
98
|
-
return None
|
99
|
-
|
100
|
-
if datetime.fromisoformat(session_expired_at) < datetime.utcnow():
|
101
|
-
return None
|
102
|
-
|
103
|
-
user_id = token_payload.get("user_id")
|
104
|
-
|
105
|
-
if not user_id or not await admin_model.get_obj(user_id):
|
106
|
-
return None
|
107
|
-
|
108
|
-
return user_id
|
81
|
+
return base64.b64encode(base64.b64decode(s)) == s
|
82
|
+
except Exception:
|
83
|
+
return False
|
@@ -9,7 +9,7 @@ import jwt
|
|
9
9
|
from asgiref.sync import sync_to_async
|
10
10
|
|
11
11
|
from fastadmin.api.exceptions import AdminApiException
|
12
|
-
from fastadmin.api.helpers import
|
12
|
+
from fastadmin.api.helpers import sanitize_filter_key, sanitize_filter_value
|
13
13
|
from fastadmin.api.schemas import (
|
14
14
|
ActionInputSchema,
|
15
15
|
ChangePasswordInputSchema,
|
@@ -28,6 +28,39 @@ from fastadmin.models.schemas import ConfigurationSchema, ModelSchema
|
|
28
28
|
from fastadmin.settings import settings
|
29
29
|
|
30
30
|
|
31
|
+
async def get_user_id_from_session_id(session_id: str | None) -> UUID | int | None:
|
32
|
+
"""This method is used to get user id from session_id.
|
33
|
+
|
34
|
+
:param session_id: A session id.
|
35
|
+
:return: A user id or None.
|
36
|
+
"""
|
37
|
+
if not session_id:
|
38
|
+
return None
|
39
|
+
|
40
|
+
admin_model = get_admin_model(settings.ADMIN_USER_MODEL)
|
41
|
+
if not admin_model:
|
42
|
+
return None
|
43
|
+
|
44
|
+
try:
|
45
|
+
token_payload = jwt.decode(session_id, settings.ADMIN_SECRET_KEY, algorithms=["HS256"])
|
46
|
+
except jwt.PyJWTError:
|
47
|
+
return None
|
48
|
+
|
49
|
+
session_expired_at = token_payload.get("session_expired_at")
|
50
|
+
if not session_expired_at:
|
51
|
+
return None
|
52
|
+
|
53
|
+
if datetime.fromisoformat(session_expired_at) < datetime.utcnow():
|
54
|
+
return None
|
55
|
+
|
56
|
+
user_id = token_payload.get("user_id")
|
57
|
+
|
58
|
+
if not user_id or not await admin_model.get_obj(user_id):
|
59
|
+
return None
|
60
|
+
|
61
|
+
return user_id
|
62
|
+
|
63
|
+
|
31
64
|
class ApiService:
|
32
65
|
async def sign_in(
|
33
66
|
self,
|
@@ -8,6 +8,7 @@ from uuid import UUID
|
|
8
8
|
|
9
9
|
from asgiref.sync import sync_to_async
|
10
10
|
|
11
|
+
from fastadmin.api.helpers import is_valid_base64
|
11
12
|
from fastadmin.api.schemas import ExportFormat
|
12
13
|
from fastadmin.models.schemas import ModelFieldWidgetSchema, WidgetType
|
13
14
|
|
@@ -18,21 +19,6 @@ class BaseModelAdmin:
|
|
18
19
|
# Use it only if you use several orms in your project.
|
19
20
|
model_name_prefix: str | None = None
|
20
21
|
|
21
|
-
# A dictionary containing the field names and the corresponding widget type and widget props for the form view.
|
22
|
-
# Example of usage:
|
23
|
-
# form_fields_widgets = {
|
24
|
-
# "description": (WidgetType.RichTextArea, {})
|
25
|
-
# }
|
26
|
-
form_fields_widgets: dict[str, tuple[WidgetType, dict]] = {}
|
27
|
-
|
28
|
-
# A dictionary containing the field names and the corresponding widget type and
|
29
|
-
# column widths (px, %) for the list view.
|
30
|
-
# Example of usage:
|
31
|
-
# table_fields_widths = {
|
32
|
-
# "id": "100px",
|
33
|
-
# }
|
34
|
-
table_fields_widths: dict[str, str] = {}
|
35
|
-
|
36
22
|
# A list of actions to make available on the change list page.
|
37
23
|
# You have to implement methods with names like <action_name> in your ModelAdmin class and decorate them with @action decorator. # noqa: E501
|
38
24
|
# Example of usage:
|
@@ -76,6 +62,10 @@ class BaseModelAdmin:
|
|
76
62
|
# Example of usage: fields = ("id", "mobile_number", "email", "is_superuser", "is_active", "created_at")
|
77
63
|
fields: Sequence[str] = ()
|
78
64
|
|
65
|
+
# Set fieldsets to control the layout of admin “add” and “change” pages.
|
66
|
+
# fieldsets is a list of two-tuples, in which each two-tuple represents a <fieldset> on the admin form page. (A <fieldset> is a “section” of the form.) # noqa: E501
|
67
|
+
fieldsets: Sequence[tuple[str | None, dict[str, Sequence[str]]]] = ()
|
68
|
+
|
79
69
|
# By default, a ManyToManyField is displayed in the admin dashboard with a <select multiple>.
|
80
70
|
# However, multiple-select boxes can be difficult to use when selecting many items.
|
81
71
|
# Adding a ManyToManyField to this list will instead use a nifty unobtrusive JavaScript “filter” interface that allows searching within the options. # noqa: E501
|
@@ -87,11 +77,35 @@ class BaseModelAdmin:
|
|
87
77
|
# Example of usage: filter_vertical = ("groups", "user_permissions")
|
88
78
|
filter_vertical: Sequence[str] = ()
|
89
79
|
|
80
|
+
# Not supported setting
|
81
|
+
# form
|
82
|
+
|
83
|
+
# This provides a quick-and-dirty way to override some of the Field options for use in the admin.
|
84
|
+
# formfield_overrides is a dictionary mapping a field class to a dict
|
85
|
+
# of arguments to pass to the field at construction time.
|
86
|
+
# Example of usage:
|
87
|
+
# formfield_overrides = {
|
88
|
+
# "description": (WidgetType.RichTextArea, {})
|
89
|
+
# }
|
90
|
+
formfield_overrides: dict[str, tuple[WidgetType, dict]] = {}
|
91
|
+
|
90
92
|
# Set list_display to control which fields are displayed on the list page of the admin.
|
91
93
|
# If you don’t set list_display, the admin site will display a single column that displays the __str__() representation of each object # noqa: E501
|
92
94
|
# Example of usage: list_display = ("id", "mobile_number", "email", "is_superuser", "is_active", "created_at")
|
93
95
|
list_display: Sequence[str] = ()
|
94
96
|
|
97
|
+
# Use list_display_links to control if and which fields in list_display should be linked to the “change” page for an object. # noqa: E501
|
98
|
+
# Example of usage: list_display_links = ("id", "mobile_number", "email")
|
99
|
+
list_display_links: Sequence[str] = ()
|
100
|
+
|
101
|
+
# A dictionary containing the field names and the corresponding widget type and
|
102
|
+
# column widths (px, %) for the list view.
|
103
|
+
# Example of usage:
|
104
|
+
# list_display_widths = {
|
105
|
+
# "id": "100px",
|
106
|
+
# }
|
107
|
+
list_display_widths: dict[str, str] = {}
|
108
|
+
|
95
109
|
# Set list_filter to activate filters in the tabel columns of the list page of the admin.
|
96
110
|
# Example of usage: list_filter = ("is_superuser", "is_active", "created_at")
|
97
111
|
list_filter: Sequence[str] = ()
|
@@ -420,7 +434,7 @@ class BaseModelAdmin:
|
|
420
434
|
return None
|
421
435
|
|
422
436
|
for upload_field in upload_fields:
|
423
|
-
if upload_field.name in payload:
|
437
|
+
if upload_field.name in payload and is_valid_base64(payload[upload_field.name]):
|
424
438
|
await self.orm_save_upload_field(obj, upload_field.column_name, payload[upload_field.name])
|
425
439
|
|
426
440
|
for m2m_field in m2m_fields:
|
@@ -544,23 +558,6 @@ class InlineModelAdmin(BaseModelAdmin):
|
|
544
558
|
class ModelAdmin(BaseModelAdmin):
|
545
559
|
"""This class is used to create admin model class."""
|
546
560
|
|
547
|
-
# Use list_display_links to control if and which fields in list_display should be linked to the “change” page for an object. # noqa: E501
|
548
|
-
# Example of usage: list_display_links = ("id", "mobile_number", "email")
|
549
|
-
list_display_links: Sequence[str] = ()
|
550
|
-
|
551
|
-
# Set fieldsets to control the layout of admin “add” and “change” pages.
|
552
|
-
# fieldsets is a list of two-tuples, in which each two-tuple represents a <fieldset> on the admin form page. (A <fieldset> is a “section” of the form.) # noqa: E501
|
553
|
-
fieldsets: Sequence[tuple[str | None, dict[str, Sequence[str]]]] = ()
|
554
|
-
|
555
|
-
# Not supported setting
|
556
|
-
# form
|
557
|
-
|
558
|
-
# Not supported setting
|
559
|
-
# inlines
|
560
|
-
|
561
|
-
# Not supported setting
|
562
|
-
# formfield_overrides
|
563
|
-
|
564
561
|
# Normally, objects have three save options: “Save”, “Save and continue editing”, and “Save and add another”.
|
565
562
|
# If save_as is True, “Save and add another” will be replaced
|
566
563
|
# by a “Save as new” button that creates a new object (with a new ID) rather than updating the existing object.
|
@@ -582,6 +579,7 @@ class ModelAdmin(BaseModelAdmin):
|
|
582
579
|
# Example of usage: view_on_site = "http://example.com"
|
583
580
|
view_on_site: str | None = None
|
584
581
|
|
582
|
+
# Inlines
|
585
583
|
inlines: Sequence[type[InlineModelAdmin]] = ()
|
586
584
|
|
587
585
|
async def authenticate(self, username: str, password: str) -> UUID | int | None:
|
@@ -135,7 +135,7 @@ def generate_models_schema(
|
|
135
135
|
empty_value_display=admin_model_obj.empty_value_display,
|
136
136
|
filter_widget_type=filter_widget_type,
|
137
137
|
filter_widget_props=filter_widget_props,
|
138
|
-
width=admin_model_obj.
|
138
|
+
width=admin_model_obj.list_display_widths.get(field_name, None),
|
139
139
|
)
|
140
140
|
|
141
141
|
add_configuration = None
|
@@ -284,6 +284,13 @@ def generate_models_schema(
|
|
284
284
|
|
285
285
|
|
286
286
|
def getattrs(obj: Any, attrs: str, default: Any | None = None) -> Any:
|
287
|
+
"""Get attributes from an object.
|
288
|
+
|
289
|
+
:param obj: An object.
|
290
|
+
:param attrs: A string of attributes separated by dots.
|
291
|
+
:param default: A default value to return if an attribute is not found.
|
292
|
+
:return: The value of the last attribute.
|
293
|
+
"""
|
287
294
|
try:
|
288
295
|
return attrgetter(attrs)(obj)
|
289
296
|
except (TypeError, AttributeError):
|
@@ -43,7 +43,8 @@ class DjangoORMMixin:
|
|
43
43
|
column_name = f"{field_name}_id"
|
44
44
|
|
45
45
|
is_m2m = field_type in "ManyToManyField"
|
46
|
-
|
46
|
+
w_type, _ = self.formfield_overrides.get(field_name, (None, None))
|
47
|
+
is_upload = field_type in ("FileField", "ImageField") or w_type == WidgetType.Upload
|
47
48
|
if with_m2m is not None and not with_m2m and is_m2m:
|
48
49
|
continue
|
49
50
|
if with_m2m is not None and with_m2m and not is_m2m:
|
@@ -134,6 +135,14 @@ class DjangoORMMixin:
|
|
134
135
|
filter_widget_props["showTime"] = True
|
135
136
|
case "FileField" | "ImageField":
|
136
137
|
form_widget_type = WidgetType.Upload
|
138
|
+
case "URLField":
|
139
|
+
form_widget_type = WidgetType.UrlInput
|
140
|
+
case "EmailField":
|
141
|
+
form_widget_type = WidgetType.EmailInput
|
142
|
+
case "JSONField":
|
143
|
+
form_widget_type = WidgetType.JsonTextArea
|
144
|
+
case "SlugField":
|
145
|
+
form_widget_type = WidgetType.SlugInput
|
137
146
|
|
138
147
|
# relations
|
139
148
|
if field_type in ("ForeignKey", "OneToOneField", "ManyToManyField"):
|
@@ -197,7 +206,7 @@ class DjangoORMMixin:
|
|
197
206
|
filter_widget_type = WidgetType.AsyncSelect
|
198
207
|
filter_widget_props["mode"] = "multiple"
|
199
208
|
|
200
|
-
form_widget_type, form_widget_props = self.
|
209
|
+
form_widget_type, form_widget_props = self.formfield_overrides.get(
|
201
210
|
field_name, (form_widget_type, form_widget_props)
|
202
211
|
)
|
203
212
|
fields.append(
|
@@ -45,7 +45,8 @@ class PonyORMMixin:
|
|
45
45
|
field_type = "fk"
|
46
46
|
|
47
47
|
is_m2m = field_type in "m2m"
|
48
|
-
|
48
|
+
w_type, _ = self.formfield_overrides.get(field_name, (None, None))
|
49
|
+
is_upload = w_type == WidgetType.Upload
|
49
50
|
if with_m2m is not None and not with_m2m and is_m2m:
|
50
51
|
continue
|
51
52
|
if with_m2m is not None and with_m2m and not is_m2m:
|
@@ -129,6 +130,8 @@ class PonyORMMixin:
|
|
129
130
|
form_widget_type = WidgetType.Select
|
130
131
|
filter_widget_type = WidgetType.Select
|
131
132
|
filter_widget_props["mode"] = "multiple"
|
133
|
+
case "Json":
|
134
|
+
form_widget_type = WidgetType.JsonTextArea
|
132
135
|
|
133
136
|
# relations
|
134
137
|
if field_type in ("fk", "o2o", "m2m"):
|
@@ -192,7 +195,7 @@ class PonyORMMixin:
|
|
192
195
|
filter_widget_type = WidgetType.AsyncSelect
|
193
196
|
filter_widget_props["mode"] = "multiple"
|
194
197
|
|
195
|
-
form_widget_type, form_widget_props = self.
|
198
|
+
form_widget_type, form_widget_props = self.formfield_overrides.get(
|
196
199
|
field_name, (form_widget_type, form_widget_props)
|
197
200
|
)
|
198
201
|
fields.append(
|
@@ -54,7 +54,8 @@ class SqlAlchemyMixin:
|
|
54
54
|
continue
|
55
55
|
|
56
56
|
is_m2m = field_type == "MANYTOMANY"
|
57
|
-
|
57
|
+
w_type, _ = self.formfield_overrides.get(field_name, (None, None))
|
58
|
+
is_upload = w_type == WidgetType.Upload
|
58
59
|
if with_m2m is not None and not with_m2m and is_m2m:
|
59
60
|
continue
|
60
61
|
if with_m2m is not None and with_m2m and not is_m2m:
|
@@ -141,6 +142,8 @@ class SqlAlchemyMixin:
|
|
141
142
|
form_widget_type = WidgetType.Select
|
142
143
|
filter_widget_type = WidgetType.Select
|
143
144
|
filter_widget_props["mode"] = "multiple"
|
145
|
+
case "JSON":
|
146
|
+
form_widget_type = WidgetType.JsonTextArea
|
144
147
|
|
145
148
|
# relations
|
146
149
|
if field_type in (
|
@@ -208,7 +211,7 @@ class SqlAlchemyMixin:
|
|
208
211
|
filter_widget_type = WidgetType.AsyncSelect
|
209
212
|
filter_widget_props["mode"] = "multiple"
|
210
213
|
|
211
|
-
form_widget_type, form_widget_props = self.
|
214
|
+
form_widget_type, form_widget_props = self.formfield_overrides.get(
|
212
215
|
field_name, (form_widget_type, form_widget_props)
|
213
216
|
)
|
214
217
|
fields.append(
|
@@ -47,7 +47,8 @@ class TortoiseMixin:
|
|
47
47
|
column_name = f"{column_name}_id"
|
48
48
|
|
49
49
|
is_m2m = field_type == "ManyToManyFieldInstance"
|
50
|
-
|
50
|
+
w_type, _ = self.formfield_overrides.get(field_name, (None, None))
|
51
|
+
is_upload = w_type == WidgetType.Upload
|
51
52
|
if with_m2m is not None and not with_m2m and is_m2m:
|
52
53
|
continue
|
53
54
|
if with_m2m is not None and with_m2m and not is_m2m:
|
@@ -134,6 +135,8 @@ class TortoiseMixin:
|
|
134
135
|
form_widget_type = WidgetType.Select
|
135
136
|
filter_widget_type = WidgetType.Select
|
136
137
|
filter_widget_props["mode"] = "multiple"
|
138
|
+
case "JSONField":
|
139
|
+
form_widget_type = WidgetType.JsonTextArea
|
137
140
|
|
138
141
|
# relations
|
139
142
|
if field_type in ("ForeignKeyFieldInstance", "OneToOneFieldInstance", "ManyToManyFieldInstance"):
|
@@ -197,7 +200,7 @@ class TortoiseMixin:
|
|
197
200
|
filter_widget_type = WidgetType.AsyncSelect
|
198
201
|
filter_widget_props["mode"] = "multiple"
|
199
202
|
|
200
|
-
form_widget_type, form_widget_props = self.
|
203
|
+
form_widget_type, form_widget_props = self.formfield_overrides.get(
|
201
204
|
field_name, (form_widget_type, form_widget_props)
|
202
205
|
)
|
203
206
|
fields.append(
|
@@ -9,9 +9,14 @@ class WidgetType(str, Enum):
|
|
9
9
|
|
10
10
|
Input = "Input"
|
11
11
|
InputNumber = "InputNumber"
|
12
|
+
SlugInput = "SlugInput"
|
13
|
+
EmailInput = "EmailInput"
|
14
|
+
PhoneInput = "PhoneInput"
|
15
|
+
UrlInput = "UrlInput"
|
12
16
|
PasswordInput = "PasswordInput"
|
13
17
|
TextArea = "TextArea"
|
14
18
|
RichTextArea = "RichTextArea"
|
19
|
+
JsonTextArea = "JsonTextArea"
|
15
20
|
Select = "Select"
|
16
21
|
AsyncSelect = "AsyncSelect"
|
17
22
|
AsyncTransfer = "AsyncTransfer"
|
@@ -0,0 +1,8 @@
|
|
1
|
+
.react-tel-input{font-family:Roboto,sans-serif;font-size:15px;position:relative;width:100%}.react-tel-input :disabled{cursor:not-allowed}.react-tel-input .flag{background-image:url();height:11px;width:16px}.react-tel-input .ad{background-position:-16px 0}.react-tel-input .ae{background-position:-32px 0}.react-tel-input .af{background-position:-48px 0}.react-tel-input .ag{background-position:-64px 0}.react-tel-input .ai{background-position:-80px 0}.react-tel-input .al{background-position:-96px 0}.react-tel-input .am{background-position:-112px 0}.react-tel-input .ao{background-position:-128px 0}.react-tel-input .ar{background-position:-144px 0}.react-tel-input .as{background-position:-160px 0}.react-tel-input .at{background-position:-176px 0}.react-tel-input .au{background-position:-192px 0}.react-tel-input .aw{background-position:-208px 0}.react-tel-input .az{background-position:-224px 0}.react-tel-input .ba{background-position:-240px 0}.react-tel-input .bb{background-position:0 -11px}.react-tel-input .bd{background-position:-16px -11px}.react-tel-input .be{background-position:-32px -11px}.react-tel-input .bf{background-position:-48px -11px}.react-tel-input .bg{background-position:-64px -11px}.react-tel-input .bh{background-position:-80px -11px}.react-tel-input .bi{background-position:-96px -11px}.react-tel-input .bj{background-position:-112px -11px}.react-tel-input .bm{background-position:-128px -11px}.react-tel-input .bn{background-position:-144px -11px}.react-tel-input .bo{background-position:-160px -11px}.react-tel-input .br{background-position:-176px -11px}.react-tel-input .bs{background-position:-192px -11px}.react-tel-input .bt{background-position:-208px -11px}.react-tel-input .bw{background-position:-224px -11px}.react-tel-input .by{background-position:-240px -11px}.react-tel-input .bz{background-position:0 -22px}.react-tel-input .ca{background-position:-16px -22px}.react-tel-input .cd{background-position:-32px -22px}.react-tel-input .cf{background-position:-48px -22px}.react-tel-input .cg{background-position:-64px -22px}.react-tel-input .ch{background-position:-80px -22px}.react-tel-input .ci{background-position:-96px -22px}.react-tel-input .ck{background-position:-112px -22px}.react-tel-input .cl{background-position:-128px -22px}.react-tel-input .cm{background-position:-144px -22px}.react-tel-input .cn{background-position:-160px -22px}.react-tel-input .co{background-position:-176px -22px}.react-tel-input .cr{background-position:-192px -22px}.react-tel-input .cu{background-position:-208px -22px}.react-tel-input .cv{background-position:-224px -22px}.react-tel-input .cw{background-position:-240px -22px}.react-tel-input .cy{background-position:0 -33px}.react-tel-input .cz{background-position:-16px -33px}.react-tel-input .de{background-position:-32px -33px}.react-tel-input .dj{background-position:-48px -33px}.react-tel-input .dk{background-position:-64px -33px}.react-tel-input .dm{background-position:-80px -33px}.react-tel-input .do{background-position:-96px -33px}.react-tel-input .dz{background-position:-112px -33px}.react-tel-input .ec{background-position:-128px -33px}.react-tel-input .ee{background-position:-144px -33px}.react-tel-input .eg{background-position:-160px -33px}.react-tel-input .er{background-position:-176px -33px}.react-tel-input .es{background-position:-192px -33px}.react-tel-input .et{background-position:-208px -33px}.react-tel-input .fi{background-position:-224px -33px}.react-tel-input .fj{background-position:-240px -33px}.react-tel-input .fk{background-position:0 -44px}.react-tel-input .fm{background-position:-16px -44px}.react-tel-input .fo{background-position:-32px -44px}.react-tel-input .bl,.react-tel-input .fr,.react-tel-input .mf{background-position:-48px -44px}.react-tel-input .ga{background-position:-64px -44px}.react-tel-input .gb{background-position:-80px -44px}.react-tel-input .gd{background-position:-96px -44px}.react-tel-input .ge{background-position:-112px -44px}.react-tel-input .gf{background-position:-128px -44px}.react-tel-input .gh{background-position:-144px -44px}.react-tel-input .gi{background-position:-160px -44px}.react-tel-input .gl{background-position:-176px -44px}.react-tel-input .gm{background-position:-192px -44px}.react-tel-input .gn{background-position:-208px -44px}.react-tel-input .gp{background-position:-224px -44px}.react-tel-input .gq{background-position:-240px -44px}.react-tel-input .gr{background-position:0 -55px}.react-tel-input .gt{background-position:-16px -55px}.react-tel-input .gu{background-position:-32px -55px}.react-tel-input .gw{background-position:-48px -55px}.react-tel-input .gy{background-position:-64px -55px}.react-tel-input .hk{background-position:-80px -55px}.react-tel-input .hn{background-position:-96px -55px}.react-tel-input .hr{background-position:-112px -55px}.react-tel-input .ht{background-position:-128px -55px}.react-tel-input .hu{background-position:-144px -55px}.react-tel-input .id{background-position:-160px -55px}.react-tel-input .ie{background-position:-176px -55px}.react-tel-input .il{background-position:-192px -55px}.react-tel-input .in{background-position:-208px -55px}.react-tel-input .io{background-position:-224px -55px}.react-tel-input .iq{background-position:-240px -55px}.react-tel-input .ir{background-position:0 -66px}.react-tel-input .is{background-position:-16px -66px}.react-tel-input .it{background-position:-32px -66px}.react-tel-input .je{background-position:-144px -154px}.react-tel-input .jm{background-position:-48px -66px}.react-tel-input .jo{background-position:-64px -66px}.react-tel-input .jp{background-position:-80px -66px}.react-tel-input .ke{background-position:-96px -66px}.react-tel-input .kg{background-position:-112px -66px}.react-tel-input .kh{background-position:-128px -66px}.react-tel-input .ki{background-position:-144px -66px}.react-tel-input .xk{background-position:-128px -154px}.react-tel-input .km{background-position:-160px -66px}.react-tel-input .kn{background-position:-176px -66px}.react-tel-input .kp{background-position:-192px -66px}.react-tel-input .kr{background-position:-208px -66px}.react-tel-input .kw{background-position:-224px -66px}.react-tel-input .ky{background-position:-240px -66px}.react-tel-input .kz{background-position:0 -77px}.react-tel-input .la{background-position:-16px -77px}.react-tel-input .lb{background-position:-32px -77px}.react-tel-input .lc{background-position:-48px -77px}.react-tel-input .li{background-position:-64px -77px}.react-tel-input .lk{background-position:-80px -77px}.react-tel-input .lr{background-position:-96px -77px}.react-tel-input .ls{background-position:-112px -77px}.react-tel-input .lt{background-position:-128px -77px}.react-tel-input .lu{background-position:-144px -77px}.react-tel-input .lv{background-position:-160px -77px}.react-tel-input .ly{background-position:-176px -77px}.react-tel-input .ma{background-position:-192px -77px}.react-tel-input .mc{background-position:-208px -77px}.react-tel-input .md{background-position:-224px -77px}.react-tel-input .me{background-position:-112px -154px;height:12px}.react-tel-input .mg{background-position:0 -88px}.react-tel-input .mh{background-position:-16px -88px}.react-tel-input .mk{background-position:-32px -88px}.react-tel-input .ml{background-position:-48px -88px}.react-tel-input .mm{background-position:-64px -88px}.react-tel-input .mn{background-position:-80px -88px}.react-tel-input .mo{background-position:-96px -88px}.react-tel-input .mp{background-position:-112px -88px}.react-tel-input .mq{background-position:-128px -88px}.react-tel-input .mr{background-position:-144px -88px}.react-tel-input .ms{background-position:-160px -88px}.react-tel-input .mt{background-position:-176px -88px}.react-tel-input .mu{background-position:-192px -88px}.react-tel-input .mv{background-position:-208px -88px}.react-tel-input .mw{background-position:-224px -88px}.react-tel-input .mx{background-position:-240px -88px}.react-tel-input .my{background-position:0 -99px}.react-tel-input .mz{background-position:-16px -99px}.react-tel-input .na{background-position:-32px -99px}.react-tel-input .nc{background-position:-48px -99px}.react-tel-input .ne{background-position:-64px -99px}.react-tel-input .nf{background-position:-80px -99px}.react-tel-input .ng{background-position:-96px -99px}.react-tel-input .ni{background-position:-112px -99px}.react-tel-input .bq,.react-tel-input .nl{background-position:-128px -99px}.react-tel-input .no{background-position:-144px -99px}.react-tel-input .np{background-position:-160px -99px}.react-tel-input .nr{background-position:-176px -99px}.react-tel-input .nu{background-position:-192px -99px}.react-tel-input .nz{background-position:-208px -99px}.react-tel-input .om{background-position:-224px -99px}.react-tel-input .pa{background-position:-240px -99px}.react-tel-input .pe{background-position:0 -110px}.react-tel-input .pf{background-position:-16px -110px}.react-tel-input .pg{background-position:-32px -110px}.react-tel-input .ph{background-position:-48px -110px}.react-tel-input .pk{background-position:-64px -110px}.react-tel-input .pl{background-position:-80px -110px}.react-tel-input .pm{background-position:-96px -110px}.react-tel-input .pr{background-position:-112px -110px}.react-tel-input .ps{background-position:-128px -110px}.react-tel-input .pt{background-position:-144px -110px}.react-tel-input .pw{background-position:-160px -110px}.react-tel-input .py{background-position:-176px -110px}.react-tel-input .qa{background-position:-192px -110px}.react-tel-input .re{background-position:-208px -110px}.react-tel-input .ro{background-position:-224px -110px}.react-tel-input .rs{background-position:-240px -110px}.react-tel-input .ru{background-position:0 -121px}.react-tel-input .rw{background-position:-16px -121px}.react-tel-input .sa{background-position:-32px -121px}.react-tel-input .sb{background-position:-48px -121px}.react-tel-input .sc{background-position:-64px -121px}.react-tel-input .sd{background-position:-80px -121px}.react-tel-input .se{background-position:-96px -121px}.react-tel-input .sg{background-position:-112px -121px}.react-tel-input .sh{background-position:-128px -121px}.react-tel-input .si{background-position:-144px -121px}.react-tel-input .sk{background-position:-160px -121px}.react-tel-input .sl{background-position:-176px -121px}.react-tel-input .sm{background-position:-192px -121px}.react-tel-input .sn{background-position:-208px -121px}.react-tel-input .so{background-position:-224px -121px}.react-tel-input .sr{background-position:-240px -121px}.react-tel-input .ss{background-position:0 -132px}.react-tel-input .st{background-position:-16px -132px}.react-tel-input .sv{background-position:-32px -132px}.react-tel-input .sx{background-position:-48px -132px}.react-tel-input .sy{background-position:-64px -132px}.react-tel-input .sz{background-position:-80px -132px}.react-tel-input .tc{background-position:-96px -132px}.react-tel-input .td{background-position:-112px -132px}.react-tel-input .tg{background-position:-128px -132px}.react-tel-input .th{background-position:-144px -132px}.react-tel-input .tj{background-position:-160px -132px}.react-tel-input .tk{background-position:-176px -132px}.react-tel-input .tl{background-position:-192px -132px}.react-tel-input .tm{background-position:-208px -132px}.react-tel-input .tn{background-position:-224px -132px}.react-tel-input .to{background-position:-240px -132px}.react-tel-input .tr{background-position:0 -143px}.react-tel-input .tt{background-position:-16px -143px}.react-tel-input .tv{background-position:-32px -143px}.react-tel-input .tw{background-position:-48px -143px}.react-tel-input .tz{background-position:-64px -143px}.react-tel-input .ua{background-position:-80px -143px}.react-tel-input .ug{background-position:-96px -143px}.react-tel-input .us{background-position:-112px -143px}.react-tel-input .uy{background-position:-128px -143px}.react-tel-input .uz{background-position:-144px -143px}.react-tel-input .va{background-position:-160px -143px}.react-tel-input .vc{background-position:-176px -143px}.react-tel-input .ve{background-position:-192px -143px}.react-tel-input .vg{background-position:-208px -143px}.react-tel-input .vi{background-position:-224px -143px}.react-tel-input .vn{background-position:-240px -143px}.react-tel-input .vu{background-position:0 -154px}.react-tel-input .wf{background-position:-16px -154px}.react-tel-input .ws{background-position:-32px -154px}.react-tel-input .ye{background-position:-48px -154px}.react-tel-input .za{background-position:-64px -154px}.react-tel-input .zm{background-position:-80px -154px}.react-tel-input .zw{background-position:-96px -154px}.react-tel-input *{box-sizing:border-box;-moz-box-sizing:border-box}.react-tel-input .hide{display:none}.react-tel-input .v-hide{visibility:hidden}.react-tel-input .form-control{background:#fff;border:1px solid #cacaca;border-radius:5px;font-size:14px;height:35px;letter-spacing:.01rem;line-height:25px;margin-bottom:0!important;margin-left:0;margin-top:0!important;outline:none;padding-left:48px;position:relative;width:300px}.react-tel-input .form-control.invalid-number,.react-tel-input .form-control.invalid-number:focus{background-color:#faf0f0;border:1px solid #d79f9f;border-left-color:#cacaca}.react-tel-input .flag-dropdown{background-color:#f5f5f5;border:1px solid #cacaca;border-radius:3px 0 0 3px;bottom:0;padding:0;position:absolute;top:0}.react-tel-input .flag-dropdown:focus,.react-tel-input .flag-dropdown:hover{cursor:pointer}.react-tel-input .flag-dropdown.invalid-number{border-color:#d79f9f}.react-tel-input .flag-dropdown.open{z-index:2}.react-tel-input .flag-dropdown.open,.react-tel-input .flag-dropdown.open .selected-flag{background:#fff;border-radius:3px 0 0 0}.react-tel-input input[disabled]+.flag-dropdown:hover{cursor:default}.react-tel-input input[disabled]+.flag-dropdown:hover .selected-flag{background-color:initial}.react-tel-input .selected-flag{border-radius:3px 0 0 3px;height:100%;outline:none;padding:0 0 0 8px;position:relative;width:38px}.react-tel-input .selected-flag:focus,.react-tel-input .selected-flag:hover{background-color:#fff}.react-tel-input .selected-flag .flag{margin-top:-5px;position:absolute;top:50%}.react-tel-input .selected-flag .arrow{border-left:3px solid transparent;border-right:3px solid transparent;border-top:4px solid #555;height:0;left:20px;margin-top:-2px;position:relative;top:50%;width:0}.react-tel-input .selected-flag .arrow.up{border-bottom:4px solid #555;border-top:none}.react-tel-input .country-list{background-color:#fff;border-radius:0 0 3px 3px;box-shadow:1px 2px 10px rgba(0,0,0,.35);list-style:none;margin:10px 0 10px -1px;max-height:200px;outline:none;overflow-y:scroll;padding:0;position:absolute;width:300px;z-index:1}.react-tel-input .country-list .flag{display:inline-block}.react-tel-input .country-list .divider{border-bottom:1px solid #ccc;margin-bottom:5px;padding-bottom:5px}.react-tel-input .country-list .country{padding:7px 9px}.react-tel-input .country-list .country .dial-code{color:#6b6b6b}.react-tel-input .country-list .country.highlight,.react-tel-input .country-list .country:hover{background-color:#f1f1f1}.react-tel-input .country-list .flag{margin-right:7px;margin-top:2px}.react-tel-input .country-list .country-name{margin-right:6px}.react-tel-input .country-list .search{background-color:#fff;padding:10px 0 6px 10px;position:-webkit-sticky;position:sticky;top:0}.react-tel-input .country-list .search-emoji{font-size:15px}.react-tel-input .country-list .search-box{border:1px solid #cacaca;border-radius:3px;font-size:15px;line-height:15px;margin-left:6px;outline:none;padding:3px 8px 5px}.react-tel-input .country-list .no-entries-message{opacity:.7;padding:7px 10px 11px}.react-tel-input .invalid-number-message{color:#de0000}.react-tel-input .invalid-number-message,.react-tel-input .special-label{background:#fff;font-size:13px;left:46px;padding:0 2px;position:absolute;top:-8px;z-index:1}.react-tel-input .special-label{display:none;white-space:nowrap}
|
2
|
+
/*!
|
3
|
+
* Quill Editor v1.3.7
|
4
|
+
* https://quilljs.com/
|
5
|
+
* Copyright (c) 2014, Jason Chen
|
6
|
+
* Copyright (c) 2013, salesforce.com
|
7
|
+
*/.ql-container{box-sizing:border-box;font-family:Helvetica,Arial,sans-serif;font-size:13px;height:100%;margin:0;position:relative}.ql-container.ql-disabled .ql-tooltip{visibility:hidden}.ql-container.ql-disabled .ql-editor ul[data-checked]>li:before{pointer-events:none}.ql-clipboard{height:1px;left:-100000px;overflow-y:hidden;position:absolute;top:50%}.ql-clipboard p{margin:0;padding:0}.ql-editor{word-wrap:break-word;box-sizing:border-box;height:100%;line-height:1.42;outline:none;overflow-y:auto;padding:12px 15px;tab-size:4;-moz-tab-size:4;text-align:left;white-space:pre-wrap}.ql-editor>*{cursor:text}.ql-editor blockquote,.ql-editor h1,.ql-editor h2,.ql-editor h3,.ql-editor h4,.ql-editor h5,.ql-editor h6,.ql-editor ol,.ql-editor p,.ql-editor pre,.ql-editor ul{counter-reset:list-1 list-2 list-3 list-4 list-5 list-6 list-7 list-8 list-9;margin:0;padding:0}.ql-editor ol,.ql-editor ul{padding-left:1.5em}.ql-editor ol>li,.ql-editor ul>li{list-style-type:none}.ql-editor ul>li:before{content:"\2022"}.ql-editor ul[data-checked=false],.ql-editor ul[data-checked=true]{pointer-events:none}.ql-editor ul[data-checked=false]>li *,.ql-editor ul[data-checked=true]>li *{pointer-events:all}.ql-editor ul[data-checked=false]>li:before,.ql-editor ul[data-checked=true]>li:before{color:#777;cursor:pointer;pointer-events:all}.ql-editor ul[data-checked=true]>li:before{content:"\2611"}.ql-editor ul[data-checked=false]>li:before{content:"\2610"}.ql-editor li:before{display:inline-block;white-space:nowrap;width:1.2em}.ql-editor li:not(.ql-direction-rtl):before{margin-left:-1.5em;margin-right:.3em;text-align:right}.ql-editor li.ql-direction-rtl:before{margin-left:.3em;margin-right:-1.5em}.ql-editor ol li:not(.ql-direction-rtl),.ql-editor ul li:not(.ql-direction-rtl){padding-left:1.5em}.ql-editor ol li.ql-direction-rtl,.ql-editor ul li.ql-direction-rtl{padding-right:1.5em}.ql-editor ol li{counter-increment:list-0;counter-reset:list-1 list-2 list-3 list-4 list-5 list-6 list-7 list-8 list-9}.ql-editor ol li:before{content:counter(list-0,decimal) ". "}.ql-editor ol li.ql-indent-1{counter-increment:list-1}.ql-editor ol li.ql-indent-1:before{content:counter(list-1,lower-alpha) ". "}.ql-editor ol li.ql-indent-1{counter-reset:list-2 list-3 list-4 list-5 list-6 list-7 list-8 list-9}.ql-editor ol li.ql-indent-2{counter-increment:list-2}.ql-editor ol li.ql-indent-2:before{content:counter(list-2,lower-roman) ". "}.ql-editor ol li.ql-indent-2{counter-reset:list-3 list-4 list-5 list-6 list-7 list-8 list-9}.ql-editor ol li.ql-indent-3{counter-increment:list-3}.ql-editor ol li.ql-indent-3:before{content:counter(list-3,decimal) ". "}.ql-editor ol li.ql-indent-3{counter-reset:list-4 list-5 list-6 list-7 list-8 list-9}.ql-editor ol li.ql-indent-4{counter-increment:list-4}.ql-editor ol li.ql-indent-4:before{content:counter(list-4,lower-alpha) ". "}.ql-editor ol li.ql-indent-4{counter-reset:list-5 list-6 list-7 list-8 list-9}.ql-editor ol li.ql-indent-5{counter-increment:list-5}.ql-editor ol li.ql-indent-5:before{content:counter(list-5,lower-roman) ". "}.ql-editor ol li.ql-indent-5{counter-reset:list-6 list-7 list-8 list-9}.ql-editor ol li.ql-indent-6{counter-increment:list-6}.ql-editor ol li.ql-indent-6:before{content:counter(list-6,decimal) ". "}.ql-editor ol li.ql-indent-6{counter-reset:list-7 list-8 list-9}.ql-editor ol li.ql-indent-7{counter-increment:list-7}.ql-editor ol li.ql-indent-7:before{content:counter(list-7,lower-alpha) ". "}.ql-editor ol li.ql-indent-7{counter-reset:list-8 list-9}.ql-editor ol li.ql-indent-8{counter-increment:list-8}.ql-editor ol li.ql-indent-8:before{content:counter(list-8,lower-roman) ". "}.ql-editor ol li.ql-indent-8{counter-reset:list-9}.ql-editor ol li.ql-indent-9{counter-increment:list-9}.ql-editor ol li.ql-indent-9:before{content:counter(list-9,decimal) ". "}.ql-editor .ql-indent-1:not(.ql-direction-rtl){padding-left:3em}.ql-editor li.ql-indent-1:not(.ql-direction-rtl){padding-left:4.5em}.ql-editor .ql-indent-1.ql-direction-rtl.ql-align-right{padding-right:3em}.ql-editor li.ql-indent-1.ql-direction-rtl.ql-align-right{padding-right:4.5em}.ql-editor .ql-indent-2:not(.ql-direction-rtl){padding-left:6em}.ql-editor li.ql-indent-2:not(.ql-direction-rtl){padding-left:7.5em}.ql-editor .ql-indent-2.ql-direction-rtl.ql-align-right{padding-right:6em}.ql-editor li.ql-indent-2.ql-direction-rtl.ql-align-right{padding-right:7.5em}.ql-editor .ql-indent-3:not(.ql-direction-rtl){padding-left:9em}.ql-editor li.ql-indent-3:not(.ql-direction-rtl){padding-left:10.5em}.ql-editor .ql-indent-3.ql-direction-rtl.ql-align-right{padding-right:9em}.ql-editor li.ql-indent-3.ql-direction-rtl.ql-align-right{padding-right:10.5em}.ql-editor .ql-indent-4:not(.ql-direction-rtl){padding-left:12em}.ql-editor li.ql-indent-4:not(.ql-direction-rtl){padding-left:13.5em}.ql-editor .ql-indent-4.ql-direction-rtl.ql-align-right{padding-right:12em}.ql-editor li.ql-indent-4.ql-direction-rtl.ql-align-right{padding-right:13.5em}.ql-editor .ql-indent-5:not(.ql-direction-rtl){padding-left:15em}.ql-editor li.ql-indent-5:not(.ql-direction-rtl){padding-left:16.5em}.ql-editor .ql-indent-5.ql-direction-rtl.ql-align-right{padding-right:15em}.ql-editor li.ql-indent-5.ql-direction-rtl.ql-align-right{padding-right:16.5em}.ql-editor .ql-indent-6:not(.ql-direction-rtl){padding-left:18em}.ql-editor li.ql-indent-6:not(.ql-direction-rtl){padding-left:19.5em}.ql-editor .ql-indent-6.ql-direction-rtl.ql-align-right{padding-right:18em}.ql-editor li.ql-indent-6.ql-direction-rtl.ql-align-right{padding-right:19.5em}.ql-editor .ql-indent-7:not(.ql-direction-rtl){padding-left:21em}.ql-editor li.ql-indent-7:not(.ql-direction-rtl){padding-left:22.5em}.ql-editor .ql-indent-7.ql-direction-rtl.ql-align-right{padding-right:21em}.ql-editor li.ql-indent-7.ql-direction-rtl.ql-align-right{padding-right:22.5em}.ql-editor .ql-indent-8:not(.ql-direction-rtl){padding-left:24em}.ql-editor li.ql-indent-8:not(.ql-direction-rtl){padding-left:25.5em}.ql-editor .ql-indent-8.ql-direction-rtl.ql-align-right{padding-right:24em}.ql-editor li.ql-indent-8.ql-direction-rtl.ql-align-right{padding-right:25.5em}.ql-editor .ql-indent-9:not(.ql-direction-rtl){padding-left:27em}.ql-editor li.ql-indent-9:not(.ql-direction-rtl){padding-left:28.5em}.ql-editor .ql-indent-9.ql-direction-rtl.ql-align-right{padding-right:27em}.ql-editor li.ql-indent-9.ql-direction-rtl.ql-align-right{padding-right:28.5em}.ql-editor .ql-video{display:block;max-width:100%}.ql-editor .ql-video.ql-align-center{margin:0 auto}.ql-editor .ql-video.ql-align-right{margin:0 0 0 auto}.ql-editor .ql-bg-black{background-color:#000}.ql-editor .ql-bg-red{background-color:#e60000}.ql-editor .ql-bg-orange{background-color:#f90}.ql-editor .ql-bg-yellow{background-color:#ff0}.ql-editor .ql-bg-green{background-color:#008a00}.ql-editor .ql-bg-blue{background-color:#06c}.ql-editor .ql-bg-purple{background-color:#93f}.ql-editor .ql-color-white{color:#fff}.ql-editor .ql-color-red{color:#e60000}.ql-editor .ql-color-orange{color:#f90}.ql-editor .ql-color-yellow{color:#ff0}.ql-editor .ql-color-green{color:#008a00}.ql-editor .ql-color-blue{color:#06c}.ql-editor .ql-color-purple{color:#93f}.ql-editor .ql-font-serif{font-family:Georgia,Times New Roman,serif}.ql-editor .ql-font-monospace{font-family:Monaco,Courier New,monospace}.ql-editor .ql-size-small{font-size:.75em}.ql-editor .ql-size-large{font-size:1.5em}.ql-editor .ql-size-huge{font-size:2.5em}.ql-editor .ql-direction-rtl{direction:rtl;text-align:inherit}.ql-editor .ql-align-center{text-align:center}.ql-editor .ql-align-justify{text-align:justify}.ql-editor .ql-align-right{text-align:right}.ql-editor.ql-blank:before{color:rgba(0,0,0,.6);content:attr(data-placeholder);font-style:italic;left:15px;pointer-events:none;position:absolute;right:15px}.ql-snow .ql-toolbar:after,.ql-snow.ql-toolbar:after{clear:both;content:"";display:table}.ql-snow .ql-toolbar button,.ql-snow.ql-toolbar button{background:none;border:none;cursor:pointer;display:inline-block;float:left;height:24px;padding:3px 5px;width:28px}.ql-snow .ql-toolbar button svg,.ql-snow.ql-toolbar button svg{float:left;height:100%}.ql-snow .ql-toolbar button:active:hover,.ql-snow.ql-toolbar button:active:hover{outline:none}.ql-snow .ql-toolbar input.ql-image[type=file],.ql-snow.ql-toolbar input.ql-image[type=file]{display:none}.ql-snow .ql-toolbar .ql-picker-item.ql-selected,.ql-snow .ql-toolbar .ql-picker-item:hover,.ql-snow .ql-toolbar .ql-picker-label.ql-active,.ql-snow .ql-toolbar .ql-picker-label:hover,.ql-snow .ql-toolbar button.ql-active,.ql-snow .ql-toolbar button:focus,.ql-snow .ql-toolbar button:hover,.ql-snow.ql-toolbar .ql-picker-item.ql-selected,.ql-snow.ql-toolbar .ql-picker-item:hover,.ql-snow.ql-toolbar .ql-picker-label.ql-active,.ql-snow.ql-toolbar .ql-picker-label:hover,.ql-snow.ql-toolbar button.ql-active,.ql-snow.ql-toolbar button:focus,.ql-snow.ql-toolbar button:hover{color:#06c}.ql-snow .ql-toolbar .ql-picker-item.ql-selected .ql-fill,.ql-snow .ql-toolbar .ql-picker-item.ql-selected .ql-stroke.ql-fill,.ql-snow .ql-toolbar .ql-picker-item:hover .ql-fill,.ql-snow .ql-toolbar .ql-picker-item:hover .ql-stroke.ql-fill,.ql-snow .ql-toolbar .ql-picker-label.ql-active .ql-fill,.ql-snow .ql-toolbar .ql-picker-label.ql-active .ql-stroke.ql-fill,.ql-snow .ql-toolbar .ql-picker-label:hover .ql-fill,.ql-snow .ql-toolbar .ql-picker-label:hover .ql-stroke.ql-fill,.ql-snow .ql-toolbar button.ql-active .ql-fill,.ql-snow .ql-toolbar button.ql-active .ql-stroke.ql-fill,.ql-snow .ql-toolbar button:focus .ql-fill,.ql-snow .ql-toolbar button:focus .ql-stroke.ql-fill,.ql-snow .ql-toolbar button:hover .ql-fill,.ql-snow .ql-toolbar button:hover .ql-stroke.ql-fill,.ql-snow.ql-toolbar .ql-picker-item.ql-selected .ql-fill,.ql-snow.ql-toolbar .ql-picker-item.ql-selected .ql-stroke.ql-fill,.ql-snow.ql-toolbar .ql-picker-item:hover .ql-fill,.ql-snow.ql-toolbar .ql-picker-item:hover .ql-stroke.ql-fill,.ql-snow.ql-toolbar .ql-picker-label.ql-active .ql-fill,.ql-snow.ql-toolbar .ql-picker-label.ql-active .ql-stroke.ql-fill,.ql-snow.ql-toolbar .ql-picker-label:hover .ql-fill,.ql-snow.ql-toolbar .ql-picker-label:hover .ql-stroke.ql-fill,.ql-snow.ql-toolbar button.ql-active .ql-fill,.ql-snow.ql-toolbar button.ql-active .ql-stroke.ql-fill,.ql-snow.ql-toolbar button:focus .ql-fill,.ql-snow.ql-toolbar button:focus .ql-stroke.ql-fill,.ql-snow.ql-toolbar button:hover .ql-fill,.ql-snow.ql-toolbar button:hover .ql-stroke.ql-fill{fill:#06c}.ql-snow .ql-toolbar .ql-picker-item.ql-selected .ql-stroke,.ql-snow .ql-toolbar .ql-picker-item.ql-selected .ql-stroke-miter,.ql-snow .ql-toolbar .ql-picker-item:hover .ql-stroke,.ql-snow .ql-toolbar .ql-picker-item:hover .ql-stroke-miter,.ql-snow .ql-toolbar .ql-picker-label.ql-active .ql-stroke,.ql-snow .ql-toolbar .ql-picker-label.ql-active .ql-stroke-miter,.ql-snow .ql-toolbar .ql-picker-label:hover .ql-stroke,.ql-snow .ql-toolbar .ql-picker-label:hover .ql-stroke-miter,.ql-snow .ql-toolbar button.ql-active .ql-stroke,.ql-snow .ql-toolbar button.ql-active .ql-stroke-miter,.ql-snow .ql-toolbar button:focus .ql-stroke,.ql-snow .ql-toolbar button:focus .ql-stroke-miter,.ql-snow .ql-toolbar button:hover .ql-stroke,.ql-snow .ql-toolbar button:hover .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-item.ql-selected .ql-stroke,.ql-snow.ql-toolbar .ql-picker-item.ql-selected .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-item:hover .ql-stroke,.ql-snow.ql-toolbar .ql-picker-item:hover .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-label.ql-active .ql-stroke,.ql-snow.ql-toolbar .ql-picker-label.ql-active .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-label:hover .ql-stroke,.ql-snow.ql-toolbar .ql-picker-label:hover .ql-stroke-miter,.ql-snow.ql-toolbar button.ql-active .ql-stroke,.ql-snow.ql-toolbar button.ql-active .ql-stroke-miter,.ql-snow.ql-toolbar button:focus .ql-stroke,.ql-snow.ql-toolbar button:focus .ql-stroke-miter,.ql-snow.ql-toolbar button:hover .ql-stroke,.ql-snow.ql-toolbar button:hover .ql-stroke-miter{stroke:#06c}@media (pointer:coarse){.ql-snow .ql-toolbar button:hover:not(.ql-active),.ql-snow.ql-toolbar button:hover:not(.ql-active){color:#444}.ql-snow .ql-toolbar button:hover:not(.ql-active) .ql-fill,.ql-snow .ql-toolbar button:hover:not(.ql-active) .ql-stroke.ql-fill,.ql-snow.ql-toolbar button:hover:not(.ql-active) .ql-fill,.ql-snow.ql-toolbar button:hover:not(.ql-active) .ql-stroke.ql-fill{fill:#444}.ql-snow .ql-toolbar button:hover:not(.ql-active) .ql-stroke,.ql-snow .ql-toolbar button:hover:not(.ql-active) .ql-stroke-miter,.ql-snow.ql-toolbar button:hover:not(.ql-active) .ql-stroke,.ql-snow.ql-toolbar button:hover:not(.ql-active) .ql-stroke-miter{stroke:#444}}.ql-snow,.ql-snow *{box-sizing:border-box}.ql-snow .ql-hidden{display:none}.ql-snow .ql-out-bottom,.ql-snow .ql-out-top{visibility:hidden}.ql-snow .ql-tooltip{position:absolute;-webkit-transform:translateY(10px);transform:translateY(10px)}.ql-snow .ql-tooltip a{cursor:pointer;text-decoration:none}.ql-snow .ql-tooltip.ql-flip{-webkit-transform:translateY(-10px);transform:translateY(-10px)}.ql-snow .ql-formats{display:inline-block;vertical-align:middle}.ql-snow .ql-formats:after{clear:both;content:"";display:table}.ql-snow .ql-stroke{fill:none;stroke:#444;stroke-linecap:round;stroke-linejoin:round;stroke-width:2}.ql-snow .ql-stroke-miter{fill:none;stroke:#444;stroke-miterlimit:10;stroke-width:2}.ql-snow .ql-fill,.ql-snow .ql-stroke.ql-fill{fill:#444}.ql-snow .ql-empty{fill:none}.ql-snow .ql-even{fill-rule:evenodd}.ql-snow .ql-stroke.ql-thin,.ql-snow .ql-thin{stroke-width:1}.ql-snow .ql-transparent{opacity:.4}.ql-snow .ql-direction svg:last-child{display:none}.ql-snow .ql-direction.ql-active svg:last-child{display:inline}.ql-snow .ql-direction.ql-active svg:first-child{display:none}.ql-snow .ql-editor h1{font-size:2em}.ql-snow .ql-editor h2{font-size:1.5em}.ql-snow .ql-editor h3{font-size:1.17em}.ql-snow .ql-editor h4{font-size:1em}.ql-snow .ql-editor h5{font-size:.83em}.ql-snow .ql-editor h6{font-size:.67em}.ql-snow .ql-editor a{text-decoration:underline}.ql-snow .ql-editor blockquote{border-left:4px solid #ccc;margin-bottom:5px;margin-top:5px;padding-left:16px}.ql-snow .ql-editor code,.ql-snow .ql-editor pre{background-color:#f0f0f0;border-radius:3px}.ql-snow .ql-editor pre{margin-bottom:5px;margin-top:5px;padding:5px 10px;white-space:pre-wrap}.ql-snow .ql-editor code{font-size:85%;padding:2px 4px}.ql-snow .ql-editor pre.ql-syntax{background-color:#23241f;color:#f8f8f2;overflow:visible}.ql-snow .ql-editor img{max-width:100%}.ql-snow .ql-picker{color:#444;display:inline-block;float:left;font-size:14px;font-weight:500;height:24px;position:relative;vertical-align:middle}.ql-snow .ql-picker-label{cursor:pointer;display:inline-block;height:100%;padding-left:8px;padding-right:2px;position:relative;width:100%}.ql-snow .ql-picker-label:before{display:inline-block;line-height:22px}.ql-snow .ql-picker-options{background-color:#fff;display:none;min-width:100%;padding:4px 8px;position:absolute;white-space:nowrap}.ql-snow .ql-picker-options .ql-picker-item{cursor:pointer;display:block;padding-bottom:5px;padding-top:5px}.ql-snow .ql-picker.ql-expanded .ql-picker-label{color:#ccc;z-index:2}.ql-snow .ql-picker.ql-expanded .ql-picker-label .ql-fill{fill:#ccc}.ql-snow .ql-picker.ql-expanded .ql-picker-label .ql-stroke{stroke:#ccc}.ql-snow .ql-picker.ql-expanded .ql-picker-options{display:block;margin-top:-1px;top:100%;z-index:1}.ql-snow .ql-color-picker,.ql-snow .ql-icon-picker{width:28px}.ql-snow .ql-color-picker .ql-picker-label,.ql-snow .ql-icon-picker .ql-picker-label{padding:2px 4px}.ql-snow .ql-color-picker .ql-picker-label svg,.ql-snow .ql-icon-picker .ql-picker-label svg{right:4px}.ql-snow .ql-icon-picker .ql-picker-options{padding:4px 0}.ql-snow .ql-icon-picker .ql-picker-item{height:24px;padding:2px 4px;width:24px}.ql-snow .ql-color-picker .ql-picker-options{padding:3px 5px;width:152px}.ql-snow .ql-color-picker .ql-picker-item{border:1px solid transparent;float:left;height:16px;margin:2px;padding:0;width:16px}.ql-snow .ql-picker:not(.ql-color-picker):not(.ql-icon-picker) svg{margin-top:-9px;position:absolute;right:0;top:50%;width:18px}.ql-snow .ql-picker.ql-font .ql-picker-item[data-label]:not([data-label=""]):before,.ql-snow .ql-picker.ql-font .ql-picker-label[data-label]:not([data-label=""]):before,.ql-snow .ql-picker.ql-header .ql-picker-item[data-label]:not([data-label=""]):before,.ql-snow .ql-picker.ql-header .ql-picker-label[data-label]:not([data-label=""]):before,.ql-snow .ql-picker.ql-size .ql-picker-item[data-label]:not([data-label=""]):before,.ql-snow .ql-picker.ql-size .ql-picker-label[data-label]:not([data-label=""]):before{content:attr(data-label)}.ql-snow .ql-picker.ql-header{width:98px}.ql-snow .ql-picker.ql-header .ql-picker-item:before,.ql-snow .ql-picker.ql-header .ql-picker-label:before{content:"Normal"}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="1"]:before,.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="1"]:before{content:"Heading 1"}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="2"]:before,.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="2"]:before{content:"Heading 2"}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="3"]:before,.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="3"]:before{content:"Heading 3"}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="4"]:before,.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="4"]:before{content:"Heading 4"}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="5"]:before,.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="5"]:before{content:"Heading 5"}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="6"]:before,.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="6"]:before{content:"Heading 6"}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="1"]:before{font-size:2em}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="2"]:before{font-size:1.5em}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="3"]:before{font-size:1.17em}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="4"]:before{font-size:1em}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="5"]:before{font-size:.83em}.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="6"]:before{font-size:.67em}.ql-snow .ql-picker.ql-font{width:108px}.ql-snow .ql-picker.ql-font .ql-picker-item:before,.ql-snow .ql-picker.ql-font .ql-picker-label:before{content:"Sans Serif"}.ql-snow .ql-picker.ql-font .ql-picker-item[data-value=serif]:before,.ql-snow .ql-picker.ql-font .ql-picker-label[data-value=serif]:before{content:"Serif"}.ql-snow .ql-picker.ql-font .ql-picker-item[data-value=monospace]:before,.ql-snow .ql-picker.ql-font .ql-picker-label[data-value=monospace]:before{content:"Monospace"}.ql-snow .ql-picker.ql-font .ql-picker-item[data-value=serif]:before{font-family:Georgia,Times New Roman,serif}.ql-snow .ql-picker.ql-font .ql-picker-item[data-value=monospace]:before{font-family:Monaco,Courier New,monospace}.ql-snow .ql-picker.ql-size{width:98px}.ql-snow .ql-picker.ql-size .ql-picker-item:before,.ql-snow .ql-picker.ql-size .ql-picker-label:before{content:"Normal"}.ql-snow .ql-picker.ql-size .ql-picker-item[data-value=small]:before,.ql-snow .ql-picker.ql-size .ql-picker-label[data-value=small]:before{content:"Small"}.ql-snow .ql-picker.ql-size .ql-picker-item[data-value=large]:before,.ql-snow .ql-picker.ql-size .ql-picker-label[data-value=large]:before{content:"Large"}.ql-snow .ql-picker.ql-size .ql-picker-item[data-value=huge]:before,.ql-snow .ql-picker.ql-size .ql-picker-label[data-value=huge]:before{content:"Huge"}.ql-snow .ql-picker.ql-size .ql-picker-item[data-value=small]:before{font-size:10px}.ql-snow .ql-picker.ql-size .ql-picker-item[data-value=large]:before{font-size:18px}.ql-snow .ql-picker.ql-size .ql-picker-item[data-value=huge]:before{font-size:32px}.ql-snow .ql-color-picker.ql-background .ql-picker-item{background-color:#fff}.ql-snow .ql-color-picker.ql-color .ql-picker-item{background-color:#000}.ql-toolbar.ql-snow{border:1px solid #ccc;box-sizing:border-box;font-family:Helvetica Neue,Helvetica,Arial,sans-serif;padding:8px}.ql-toolbar.ql-snow .ql-formats{margin-right:15px}.ql-toolbar.ql-snow .ql-picker-label{border:1px solid transparent}.ql-toolbar.ql-snow .ql-picker-options{border:1px solid transparent;box-shadow:0 2px 8px rgba(0,0,0,.2)}.ql-toolbar.ql-snow .ql-picker.ql-expanded .ql-picker-label,.ql-toolbar.ql-snow .ql-picker.ql-expanded .ql-picker-options{border-color:#ccc}.ql-toolbar.ql-snow .ql-color-picker .ql-picker-item.ql-selected,.ql-toolbar.ql-snow .ql-color-picker .ql-picker-item:hover{border-color:#000}.ql-toolbar.ql-snow+.ql-container.ql-snow{border-top:0}.ql-snow .ql-tooltip{background-color:#fff;border:1px solid #ccc;box-shadow:0 0 5px #ddd;color:#444;padding:5px 12px;white-space:nowrap}.ql-snow .ql-tooltip:before{content:"Visit URL:";line-height:26px;margin-right:8px}.ql-snow .ql-tooltip input[type=text]{border:1px solid #ccc;display:none;font-size:13px;height:26px;margin:0;padding:3px 5px;width:170px}.ql-snow .ql-tooltip a.ql-preview{display:inline-block;max-width:200px;overflow-x:hidden;text-overflow:ellipsis;vertical-align:top}.ql-snow .ql-tooltip a.ql-action:after{border-right:1px solid #ccc;content:"Edit";margin-left:16px;padding-right:8px}.ql-snow .ql-tooltip a.ql-remove:before{content:"Remove";margin-left:8px}.ql-snow .ql-tooltip a{line-height:26px}.ql-snow .ql-tooltip.ql-editing a.ql-preview,.ql-snow .ql-tooltip.ql-editing a.ql-remove{display:none}.ql-snow .ql-tooltip.ql-editing input[type=text]{display:inline-block}.ql-snow .ql-tooltip.ql-editing a.ql-action:after{border-right:0;content:"Save";padding-right:0}.ql-snow .ql-tooltip[data-mode=link]:before{content:"Enter link:"}.ql-snow .ql-tooltip[data-mode=formula]:before{content:"Enter formula:"}.ql-snow .ql-tooltip[data-mode=video]:before{content:"Enter video:"}.ql-snow a{color:#06c}.ql-container.ql-snow{border:1px solid #ccc}body{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,Cantarell,Fira Sans,Droid Sans,Helvetica Neue,sans-serif;margin:0}code{font-family:source-code-pro,Menlo,Monaco,Consolas,Courier New,monospace}.table-row-selected{background-color:#f0f0f0}.quill{background-color:#fff!important}.ql-snow .ql-toolbar .ql-picker-item.ql-selected,.ql-snow .ql-toolbar .ql-picker-item:hover,.ql-snow .ql-toolbar .ql-picker-label.ql-active,.ql-snow .ql-toolbar .ql-picker-label:hover,.ql-snow .ql-toolbar button.ql-active,.ql-snow .ql-toolbar button:focus,.ql-snow .ql-toolbar button:hover,.ql-snow.ql-toolbar .ql-picker-item.ql-selected,.ql-snow.ql-toolbar .ql-picker-item:hover,.ql-snow.ql-toolbar .ql-picker-label.ql-active,.ql-snow.ql-toolbar .ql-picker-label:hover,.ql-snow.ql-toolbar button.ql-active,.ql-snow.ql-toolbar button:focus,.ql-snow.ql-toolbar button:hover{color:#009485!important}.ql-snow .ql-toolbar .ql-picker-item.ql-selected .ql-stroke,.ql-snow .ql-toolbar .ql-picker-item.ql-selected .ql-stroke-miter,.ql-snow .ql-toolbar .ql-picker-item:hover .ql-stroke,.ql-snow .ql-toolbar .ql-picker-item:hover .ql-stroke-miter,.ql-snow .ql-toolbar .ql-picker-label.ql-active .ql-stroke,.ql-snow .ql-toolbar .ql-picker-label.ql-active .ql-stroke-miter,.ql-snow .ql-toolbar .ql-picker-label:hover .ql-stroke,.ql-snow .ql-toolbar .ql-picker-label:hover .ql-stroke-miter,.ql-snow .ql-toolbar button.ql-active .ql-stroke,.ql-snow .ql-toolbar button.ql-active .ql-stroke-miter,.ql-snow .ql-toolbar button:focus .ql-stroke,.ql-snow .ql-toolbar button:focus .ql-stroke-miter,.ql-snow .ql-toolbar button:hover .ql-stroke,.ql-snow .ql-toolbar button:hover .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-item.ql-selected .ql-stroke,.ql-snow.ql-toolbar .ql-picker-item.ql-selected .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-item:hover .ql-stroke,.ql-snow.ql-toolbar .ql-picker-item:hover .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-label.ql-active .ql-stroke,.ql-snow.ql-toolbar .ql-picker-label.ql-active .ql-stroke-miter,.ql-snow.ql-toolbar .ql-picker-label:hover .ql-stroke,.ql-snow.ql-toolbar .ql-picker-label:hover .ql-stroke-miter,.ql-snow.ql-toolbar button.ql-active .ql-stroke,.ql-snow.ql-toolbar button.ql-active .ql-stroke-miter,.ql-snow.ql-toolbar button:focus .ql-stroke,.ql-snow.ql-toolbar button:focus .ql-stroke-miter,.ql-snow.ql-toolbar button:hover .ql-stroke,.ql-snow.ql-toolbar button:hover .ql-stroke-miter{stroke:#009485!important}.ql-snow a{color:#009485!important}.ql-toolbar{border-top-left-radius:5px;border-top-right-radius:5px}.ql-container{border-bottom-left-radius:5px;border-bottom-right-radius:5px}
|
8
|
+
/*# sourceMappingURL=main.805d4533.css.map*/
|