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.
Files changed (49) hide show
  1. {fastadmin-0.1.32 → fastadmin-0.1.34}/PKG-INFO +1 -1
  2. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/api.py +2 -2
  3. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/api.py +1 -2
  4. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/api.py +2 -2
  5. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/helpers.py +8 -33
  6. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/service.py +34 -1
  7. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/base.py +31 -33
  8. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/helpers.py +8 -1
  9. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/django.py +11 -2
  10. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/ponyorm.py +5 -2
  11. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/sqlalchemy.py +5 -2
  12. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/tortoise.py +5 -2
  13. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/schemas.py +5 -0
  14. fastadmin-0.1.34/fastadmin/static/css/main.min.css +8 -0
  15. fastadmin-0.1.34/fastadmin/static/css/main.min.css.map +1 -0
  16. fastadmin-0.1.34/fastadmin/static/js/main.min.js +3 -0
  17. fastadmin-0.1.34/fastadmin/static/js/main.min.js.map +1 -0
  18. {fastadmin-0.1.32 → fastadmin-0.1.34}/pyproject.toml +1 -1
  19. fastadmin-0.1.32/fastadmin/static/css/main.min.css +0 -7
  20. fastadmin-0.1.32/fastadmin/static/css/main.min.css.map +0 -1
  21. fastadmin-0.1.32/fastadmin/static/js/main.min.js +0 -3
  22. fastadmin-0.1.32/fastadmin/static/js/main.min.js.map +0 -1
  23. {fastadmin-0.1.32 → fastadmin-0.1.34}/LICENSE +0 -0
  24. {fastadmin-0.1.32 → fastadmin-0.1.34}/README.md +0 -0
  25. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/__init__.py +0 -0
  26. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/__init__.py +0 -0
  27. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/exceptions.py +0 -0
  28. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/__init__.py +0 -0
  29. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/__init__.py +0 -0
  30. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/__init__.py +0 -0
  31. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/urls.py +0 -0
  32. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/django/app/views.py +0 -0
  33. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/__init__.py +0 -0
  34. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/app.py +0 -0
  35. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/fastapi/views.py +0 -0
  36. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/__init__.py +0 -0
  37. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/app.py +0 -0
  38. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/frameworks/flask/views.py +0 -0
  39. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/api/schemas.py +0 -0
  40. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/__init__.py +0 -0
  41. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/decorators.py +0 -0
  42. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/models/orms/__init__.py +0 -0
  43. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/settings.py +0 -0
  44. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/images/favicon.png +0 -0
  45. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/images/header-logo.svg +0 -0
  46. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/images/sign-in-logo.svg +0 -0
  47. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/js/787.cda612ba.chunk.js +0 -0
  48. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/static/js/787.cda612ba.chunk.js.map +0 -0
  49. {fastadmin-0.1.32 → fastadmin-0.1.34}/fastadmin/templates/index.html +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: fastadmin
3
- Version: 0.1.32
3
+ Version: 0.1.34
4
4
  Summary:
5
5
  Home-page: https://github.com/vsdudakov/fastadmin
6
6
  License: MIT
@@ -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 get_user_id_from_session_id, is_valid_id
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 get_user_id_from_session_id, is_valid_id
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
- from datetime import datetime
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
- async def get_user_id_from_session_id(session_id: str | None) -> UUID | int | None:
79
- """This method is used to get user id from session_id.
74
+ def is_valid_base64(s: str) -> bool:
75
+ """Check if s is a valid base64.
80
76
 
81
- :param session_id: A session id.
82
- :return: A user id or None.
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
- token_payload = jwt.decode(session_id, settings.ADMIN_SECRET_KEY, algorithms=["HS256"])
93
- except jwt.PyJWTError:
94
- return None
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 get_user_id_from_session_id, sanitize_filter_key, sanitize_filter_value
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.table_fields_widths.get(field_name, None),
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
- is_upload = field_type in ("FileField", "ImageField")
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.form_fields_widgets.get(
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
- is_upload = False
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.form_fields_widgets.get(
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
- is_upload = False
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.form_fields_widgets.get(
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
- is_upload = False
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.form_fields_widgets.get(
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*/