diff --git a/.env.template b/.env.template index 36c56980..97f5604c 100644 --- a/.env.template +++ b/.env.template @@ -20,6 +20,10 @@ POSTGRES_USER=djangouser POSTGRES_PASSWORD= POSTGRES_DB=djangodb +# database connection string, when used overrides other database settings. +# format might vary depending on backend +# DATABASE_URL = engine://username:password@host:port/dbname + # the default value for the user preference 'fractions' (enable/disable fraction support) # default: disabled=0 FRACTION_PREF_DEFAULT=0 @@ -34,7 +38,7 @@ COMMENT_PREF_DEFAULT=1 SHOPPING_MIN_AUTOSYNC_INTERVAL=5 # Default for user setting sticky navbar -#STICKY_NAV_PREF_DEFAULT=1 +# STICKY_NAV_PREF_DEFAULT=1 # If staticfiles are stored at a different location uncomment and change accordingly # STATIC_URL=/static/ @@ -48,11 +52,54 @@ SHOPPING_MIN_AUTOSYNC_INTERVAL=5 # when unset: 1 (true) - this is temporary until an appropriate amount of time has passed for everyone to migrate GUNICORN_MEDIA=0 +# S3 Media settings: store mediafiles in s3 or any compatible storage backend (e.g. minio) +# as long as S3_ACCESS_KEY is not set S3 features are disabled +# S3_ACCESS_KEY= +# S3_SECRET_ACCESS_KEY= +# S3_BUCKET_NAME= +# S3_REGION_NAME= # default none, set your region might be required +# S3_QUERYSTRING_AUTH=1 # default true, set to 0 to serve media from a public bucket without signed urls +# S3_QUERYSTRING_EXPIRE=3600 # number of seconds querystring are valid for +# S3_ENDPOINT_URL= # when using a custom endpoint like minio + +# Email Settings, see https://docs.djangoproject.com/en/3.2/ref/settings/#email-host +# Required for email confirmation and password reset (automatically activates if host is set) +# EMAIL_HOST= +# EMAIL_PORT= +# EMAIL_HOST_USER= +# EMAIL_HOST_PASSWORD= +# EMAIL_USE_TLS=0 +# EMAIL_USE_SSL=0 +# DEFAULT_FROM_EMAIL= # email sender address (default 'webmaster@localhost') +# ACCOUNT_EMAIL_SUBJECT_PREFIX= # prefix used for account related emails (default "[Tandoor Recipes] ") + # allow authentication via reverse proxy (e.g. authelia), leave off if you dont know what you are doing # see docs for more information https://vabene1111.github.io/recipes/features/authentication/ # when unset: 0 (false) REVERSE_PROXY_AUTH=0 +# Default settings for spaces, apply per space and can be changed in the admin view +# SPACE_DEFAULT_MAX_RECIPES=0 # 0=unlimited recipes +# SPACE_DEFAULT_MAX_USERS=0 # 0=unlimited users per space +# SPACE_DEFAULT_FILES=1 # 1=can upload files (images, etc.) NOT IMPLEMENTED YET + +# allow people to create accounts on your application instance (without an invite link) +# when unset: 0 (false) +# ENABLE_SIGNUP=0 + +# If signup is enabled you might want to add a captcha to it to prevent spam +# HCAPTCHA_SITEKEY= +# HCAPTCHA_SECRET= + +# if signup is enabled you might want to provide urls to data protection policies or terms and conditions +# TERMS_URL= +# PRIVACY_URL= +# IMPRINT_URL= + +# enable serving of prometheus metrics under the /metrics path +# ATTENTION: view is not secured (as per the prometheus default way) so make sure to secure it +# trough your web server (or leave it open of you dont care if the stats are exposed) +# ENABLE_METRICS=0 # allows you to setup OAuth providers # see docs for more information https://vabene1111.github.io/recipes/features/authentication/ diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index a113ead1..1aa68a27 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -9,14 +9,14 @@ jobs: strategy: max-parallel: 4 matrix: - python-version: [3.8] + python-version: [3.9] steps: - uses: actions/checkout@v1 - - name: Set up Python 3.8 + - name: Set up Python 3.9 uses: actions/setup-python@v1 with: - python-version: 3.8 + python-version: 3.9 - name: Install dependencies run: | python -m pip install --upgrade pip diff --git a/.idea/recipes.iml b/.idea/recipes.iml index e1dd7064..6d3000d1 100644 --- a/.idea/recipes.iml +++ b/.idea/recipes.iml @@ -18,7 +18,7 @@ - + diff --git a/README.md b/README.md index b309f07a..25e4b902 100644 --- a/README.md +++ b/README.md @@ -25,6 +25,10 @@ ![Preview](docs/preview.png) +# Your Feedback + +Share some information on how you use Tandoor to help me improve the application [Google Survey](https://forms.gle/qNfLK2tWTeWHe9Qd7) + ## Features - 📦 **Sync** files with Dropbox and Nextcloud (more can easily be added) diff --git a/cookbook/admin.py b/cookbook/admin.py index 1df9f1b9..f59ce06d 100644 --- a/cookbook/admin.py +++ b/cookbook/admin.py @@ -7,7 +7,8 @@ from .models import (Comment, CookLog, Food, Ingredient, InviteLink, Keyword, RecipeBook, RecipeBookEntry, RecipeImport, ShareLink, ShoppingList, ShoppingListEntry, ShoppingListRecipe, Space, Step, Storage, Sync, SyncLog, Unit, UserPreference, - ViewLog, Supermarket, SupermarketCategory, SupermarketCategoryRelation, ImportLog, TelegramBot) + ViewLog, Supermarket, SupermarketCategory, SupermarketCategoryRelation, + ImportLog, TelegramBot, BookmarkletImport, UserFile) class CustomUserAdmin(UserAdmin): @@ -165,7 +166,7 @@ admin.site.register(ViewLog, ViewLogAdmin) class InviteLinkAdmin(admin.ModelAdmin): list_display = ( - 'username', 'group', 'valid_until', + 'group', 'valid_until', 'created_by', 'created_at', 'used_by' ) @@ -227,3 +228,17 @@ class TelegramBotAdmin(admin.ModelAdmin): admin.site.register(TelegramBot, TelegramBotAdmin) + + +class BookmarkletImportAdmin(admin.ModelAdmin): + list_display = ('id', 'url', 'created_by', 'created_at',) + + +admin.site.register(BookmarkletImport, BookmarkletImportAdmin) + + +class UserFileAdmin(admin.ModelAdmin): + list_display = ('id', 'name', 'file_size_kb', 'created_at',) + + +admin.site.register(UserFile, UserFileAdmin) diff --git a/cookbook/forms.py b/cookbook/forms.py index 56c584cc..f4f37e08 100644 --- a/cookbook/forms.py +++ b/cookbook/forms.py @@ -1,8 +1,12 @@ from django import forms +from django.conf import settings +from django.core.exceptions import ValidationError from django.forms import widgets from django.utils.translation import gettext_lazy as _ +from django_scopes import scopes_disabled from django_scopes.forms import SafeModelChoiceField, SafeModelMultipleChoiceField from emoji_picker.widgets import EmojiPickerTextInput +from hcaptcha.fields import hCaptchaField from .models import (Comment, Food, InviteLink, Keyword, MealPlan, Recipe, RecipeBook, RecipeBookEntry, Storage, Sync, Unit, User, @@ -42,10 +46,15 @@ class UserPreferenceForm(forms.ModelForm): ) help_texts = { - 'nav_color': _('Color of the top navigation bar. Not all colors work with all themes, just try them out!'), # noqa: E501 + 'nav_color': _('Color of the top navigation bar. Not all colors work with all themes, just try them out!'), + # noqa: E501 'default_unit': _('Default Unit to be used when inserting a new ingredient into a recipe.'), # noqa: E501 - 'use_fractions': _('Enables support for fractions in ingredient amounts (e.g. convert decimals to fractions automatically)'), # noqa: E501 - 'plan_share': _('Users with whom newly created meal plan/shopping list entries should be shared by default.'), # noqa: E501 + 'use_fractions': _( + 'Enables support for fractions in ingredient amounts (e.g. convert decimals to fractions automatically)'), + # noqa: E501 + 'plan_share': _( + 'Users with whom newly created meal plan/shopping list entries should be shared by default.'), + # noqa: E501 'show_recent': _('Show recently viewed recipes on search page.'), # noqa: E501 'ingredient_decimals': _('Number of decimals to round ingredients.'), # noqa: E501 'comments': _('If you want to be able to create and see comments underneath recipes.'), # noqa: E501 @@ -69,7 +78,7 @@ class UserNameForm(forms.ModelForm): fields = ('first_name', 'last_name') help_texts = { - 'first_name': _('Both fields are optional. If none are given the username will be displayed instead') # noqa: E501 + 'first_name': _('Both fields are optional. If none are given the username will be displayed instead') } @@ -113,22 +122,27 @@ class ImportExportBase(forms.Form): CHEFTAP = 'CHEFTAP' PEPPERPLATE = 'PEPPERPLATE' RECIPEKEEPER = 'RECIPEKEEPER' + RECETTETEK = 'RECETTETEK' RECIPESAGE = 'RECIPESAGE' DOMESTICA = 'DOMESTICA' MEALMASTER = 'MEALMASTER' REZKONV = 'REZKONV' + OPENEATS = 'OPENEATS' type = forms.ChoiceField(choices=( (DEFAULT, _('Default')), (PAPRIKA, 'Paprika'), (NEXTCLOUD, 'Nextcloud Cookbook'), (MEALIE, 'Mealie'), (CHOWDOWN, 'Chowdown'), (SAFRON, 'Safron'), (CHEFTAP, 'ChefTap'), - (PEPPERPLATE, 'Pepperplate'), (RECIPEKEEPER, 'Recipe Keeper'), (RECIPESAGE, 'Recipe Sage'), (DOMESTICA, 'Domestica'), - (MEALMASTER, 'MealMaster'), (REZKONV, 'RezKonv'), + (PEPPERPLATE, 'Pepperplate'), (RECETTETEK, 'RecetteTek'), (RECIPESAGE, 'Recipe Sage'), (DOMESTICA, 'Domestica'), + (MEALMASTER, 'MealMaster'), (REZKONV, 'RezKonv'), (OPENEATS, 'Openeats'), (RECIPEKEEPER, 'Recipe Keeper'), + )) class ImportForm(ImportExportBase): files = forms.FileField(required=True, widget=forms.ClearableFileInput(attrs={'multiple': True})) - duplicates = forms.BooleanField(help_text=_('To prevent duplicates recipes with the same name as existing ones are ignored. Check this box to import everything.'), required=False) + duplicates = forms.BooleanField(help_text=_( + 'To prevent duplicates recipes with the same name as existing ones are ignored. Check this box to import everything.'), + required=False) class ExportForm(ImportExportBase): @@ -251,7 +265,8 @@ class StorageForm(forms.ModelForm): fields = ('name', 'method', 'username', 'password', 'token', 'url', 'path') help_texts = { - 'url': _('Leave empty for dropbox and enter only base url for nextcloud (/remote.php/webdav/ is added automatically)'), + 'url': _( + 'Leave empty for dropbox and enter only base url for nextcloud (/remote.php/webdav/ is added automatically)'), } @@ -366,7 +381,8 @@ class MealPlanForm(forms.ModelForm): help_texts = { 'shared': _('You can list default users to share recipes with in the settings.'), # noqa: E501 - 'note': _('You can use markdown to format this field. See the docs here') # noqa: E501 + 'note': _('You can use markdown to format this field. See the docs here') + # noqa: E501 } widgets = { @@ -387,17 +403,62 @@ class InviteLinkForm(forms.ModelForm): super().__init__(*args, **kwargs) self.fields['space'].queryset = Space.objects.filter(created_by=user).all() + def clean(self): + space = self.cleaned_data['space'] + if space.max_users != 0 and (UserPreference.objects.filter(space=space).count() + InviteLink.objects.filter(space=space).count()) >= space.max_users: + raise ValidationError(_('Maximum number of users for this space reached.')) + + def clean_email(self): + email = self.cleaned_data['email'] + with scopes_disabled(): + if email != '' and User.objects.filter(email=email).exists(): + raise ValidationError(_('Email address already taken!')) + + return email + class Meta: model = InviteLink - fields = ('username', 'group', 'valid_until', 'space') + fields = ('email', 'group', 'valid_until', 'space') help_texts = { - 'username': _('A username is not required, if left blank the new user can choose one.') # noqa: E501 + 'email': _('An email address is not required but if present the invite link will be send to the user.'), } field_classes = { 'space': SafeModelChoiceField, } +class SpaceCreateForm(forms.Form): + prefix = 'create' + name = forms.CharField() + + def clean_name(self): + name = self.cleaned_data['name'] + with scopes_disabled(): + if Space.objects.filter(name=name).exists(): + raise ValidationError(_('Name already taken.')) + return name + + +class SpaceJoinForm(forms.Form): + prefix = 'join' + token = forms.CharField() + + +class AllAuthSignupForm(forms.Form): + captcha = hCaptchaField() + terms = forms.BooleanField(label=_('Accept Terms and Privacy')) + + def __init__(self, **kwargs): + super(AllAuthSignupForm, self).__init__(**kwargs) + if settings.PRIVACY_URL == '' and settings.TERMS_URL == '': + self.fields.pop('terms') + if settings.HCAPTCHA_SECRET == '': + self.fields.pop('captcha') + + def signup(self, request, user): + pass + + class UserCreateForm(forms.Form): name = forms.CharField(label='Username') password = forms.CharField( diff --git a/cookbook/helper/AllAuthCustomAdapter.py b/cookbook/helper/AllAuthCustomAdapter.py index 64bf3c4d..9587cf2a 100644 --- a/cookbook/helper/AllAuthCustomAdapter.py +++ b/cookbook/helper/AllAuthCustomAdapter.py @@ -1,6 +1,11 @@ +import datetime + from django.conf import settings from allauth.account.adapter import DefaultAccountAdapter +from django.contrib import messages +from django.core.cache import caches +from gettext import gettext as _ class AllAuthCustomAdapter(DefaultAccountAdapter): @@ -9,11 +14,19 @@ class AllAuthCustomAdapter(DefaultAccountAdapter): """ Whether to allow sign ups. """ - if request.resolver_match.view_name == 'account_signup': + if request.resolver_match.view_name == 'account_signup' and not settings.ENABLE_SIGNUP: return False else: return super(AllAuthCustomAdapter, self).is_open_for_signup(request) # disable password reset for now def send_mail(self, template_prefix, email, context): - pass + if settings.EMAIL_HOST != '': + default = datetime.datetime.now() + c = caches['default'].get_or_set(email, default, timeout=360) + if c == default: + super(AllAuthCustomAdapter, self).send_mail(template_prefix, email, context) + else: + messages.add_message(self.request, messages.ERROR, _('In order to prevent spam, the requested email was not send. Please wait a few minutes and try again.')) + else: + pass diff --git a/cookbook/helper/CustomStorageClass.py b/cookbook/helper/CustomStorageClass.py new file mode 100644 index 00000000..b2008939 --- /dev/null +++ b/cookbook/helper/CustomStorageClass.py @@ -0,0 +1,19 @@ +import hashlib + +from django.conf import settings +from django.core.cache import cache +from storages.backends.s3boto3 import S3Boto3Storage + + +class CachedS3Boto3Storage(S3Boto3Storage): + def url(self, name, **kwargs): + key = hashlib.md5(f'recipes_media_urls_{name}'.encode('utf-8')).hexdigest() + if result := cache.get(key): + return result + + result = super(CachedS3Boto3Storage, self).url(name, **kwargs) + + timeout = int(settings.AWS_QUERYSTRING_EXPIRE * .95) + cache.set(key, result, timeout) + + return result diff --git a/cookbook/helper/context_processors.py b/cookbook/helper/context_processors.py new file mode 100644 index 00000000..449ccbf5 --- /dev/null +++ b/cookbook/helper/context_processors.py @@ -0,0 +1,13 @@ +from django.conf import settings + + +def context_settings(request): + return { + 'EMAIL_ENABLED': settings.EMAIL_HOST != '', + 'SIGNUP_ENABLED': settings.ENABLE_SIGNUP, + 'CAPTCHA_ENABLED': settings.HCAPTCHA_SITEKEY != '', + 'HOSTED': settings.HOSTED, + 'TERMS_URL': settings.TERMS_URL, + 'PRIVACY_URL': settings.PRIVACY_URL, + 'IMPRINT_URL': settings.IMPRINT_URL, + } diff --git a/cookbook/helper/mdx_attributes.py b/cookbook/helper/mdx_attributes.py index 57eb44c1..2bde4a10 100644 --- a/cookbook/helper/mdx_attributes.py +++ b/cookbook/helper/mdx_attributes.py @@ -19,7 +19,7 @@ class StyleTreeprocessor(Treeprocessor): class MarkdownFormatExtension(markdown.Extension): - # md_ globals deprecated - see here: + # md_ globals deprecated - see here: def extendMarkdown(self, md): md.treeprocessors.register( StyleTreeprocessor(), diff --git a/cookbook/helper/permission_helper.py b/cookbook/helper/permission_helper.py index 794107f1..1ac842cd 100644 --- a/cookbook/helper/permission_helper.py +++ b/cookbook/helper/permission_helper.py @@ -120,9 +120,12 @@ class GroupRequiredMixin(object): def dispatch(self, request, *args, **kwargs): if not has_group_permission(request.user, self.groups_required): - messages.add_message(request, messages.ERROR, _('You do not have the required permissions to view this page!')) - return HttpResponseRedirect(reverse_lazy('index')) - + if not request.user.is_authenticated: + messages.add_message(request, messages.ERROR, _('You are not logged in and therefore cannot view this page!')) + return HttpResponseRedirect(reverse_lazy('account_login') + '?next=' + request.path) + else: + messages.add_message(request, messages.ERROR, _('You do not have the required permissions to view this page!')) + return HttpResponseRedirect(reverse_lazy('index')) try: obj = self.get_object() if obj.get_space() != request.space: diff --git a/cookbook/helper/recipe_html_import.py b/cookbook/helper/recipe_html_import.py new file mode 100644 index 00000000..3b06dc80 --- /dev/null +++ b/cookbook/helper/recipe_html_import.py @@ -0,0 +1,193 @@ +import json +import re + +from bs4 import BeautifulSoup +from bs4.element import Tag +from cookbook.helper import recipe_url_import as helper +from cookbook.helper.scrapers.scrapers import text_scraper +from json import JSONDecodeError +from recipe_scrapers._utils import get_host_name, normalize_string +from urllib.parse import unquote + + +def get_recipe_from_source(text, url, space): + def build_node(k, v): + if isinstance(v, dict): + node = { + 'name': k, + 'value': k, + 'children': get_children_dict(v) + } + elif isinstance(v, list): + node = { + 'name': k, + 'value': k, + 'children': get_children_list(v) + } + else: + node = { + 'name': k + ": " + normalize_string(str(v)), + 'value': normalize_string(str(v)) + } + return node + + def get_children_dict(children): + kid_list = [] + for k, v in children.items(): + kid_list.append(build_node(k, v)) + return kid_list + + def get_children_list(children): + kid_list = [] + for kid in children: + if type(kid) == list: + node = { + 'name': "unknown list", + 'value': "unknown list", + 'children': get_children_list(kid) + } + kid_list.append(node) + elif type(kid) == dict: + for k, v in kid.items(): + kid_list.append(build_node(k, v)) + else: + kid_list.append({ + 'name': normalize_string(str(kid)), + 'value': normalize_string(str(kid)) + }) + return kid_list + + recipe_json = { + 'name': '', + 'url': '', + 'description': '', + 'image': '', + 'keywords': [], + 'recipeIngredient': [], + 'recipeInstructions': '', + 'servings': '', + 'prepTime': '', + 'cookTime': '' + } + recipe_tree = [] + parse_list = [] + html_data = [] + images = [] + text = unquote(text) + + try: + parse_list.append(remove_graph(json.loads(text))) + if not url and 'url' in parse_list[0]: + url = parse_list[0]['url'] + scrape = text_scraper("", url=url) + + except JSONDecodeError: + soup = BeautifulSoup(text, "html.parser") + html_data = get_from_html(soup) + images += get_images_from_source(soup, url) + for el in soup.find_all('script', type='application/ld+json'): + el = remove_graph(el) + if not url and 'url' in el: + url = el['url'] + if type(el) == list: + for le in el: + parse_list.append(le) + elif type(el) == dict: + parse_list.append(el) + for el in soup.find_all(type='application/json'): + el = remove_graph(el) + if type(el) == list: + for le in el: + parse_list.append(le) + elif type(el) == dict: + parse_list.append(el) + scrape = text_scraper(text, url=url) + + recipe_json = helper.get_from_scraper(scrape, space) + + for el in parse_list: + temp_tree = [] + if isinstance(el, Tag): + try: + el = json.loads(el.string) + except TypeError: + continue + + for k, v in el.items(): + if isinstance(v, dict): + node = { + 'name': k, + 'value': k, + 'children': get_children_dict(v) + } + elif isinstance(v, list): + node = { + 'name': k, + 'value': k, + 'children': get_children_list(v) + } + else: + node = { + 'name': k + ": " + normalize_string(str(v)), + 'value': normalize_string(str(v)) + } + temp_tree.append(node) + + if '@type' in el and el['@type'] == 'Recipe': + recipe_tree += [{'name': 'ld+json', 'children': temp_tree}] + else: + recipe_tree += [{'name': 'json', 'children': temp_tree}] + + return recipe_json, recipe_tree, html_data, images + + +def get_from_html(soup): + INVISIBLE_ELEMS = ('style', 'script', 'head', 'title') + html = [] + for s in soup.strings: + if ((s.parent.name not in INVISIBLE_ELEMS) and (len(s.strip()) > 0)): + html.append(s) + return html + + +def get_images_from_source(soup, url): + sources = ['src', 'srcset', 'data-src'] + images = [] + img_tags = soup.find_all('img') + if url: + site = get_host_name(url) + prot = url.split(':')[0] + + urls = [] + for img in img_tags: + for src in sources: + try: + urls.append(img[src]) + except KeyError: + pass + + for u in urls: + u = u.split('?')[0] + filename = re.search(r'/([\w_-]+[.](jpg|jpeg|gif|png))$', u) + if filename: + if (('http' not in u) and (url)): + # sometimes an image source can be relative + # if it is provide the base url + u = '{}://{}{}'.format(prot, site, u) + if 'http' in u: + images.append(u) + return images + + +def remove_graph(el): + # recipes type might be wrapped in @graph type + if isinstance(el, Tag): + try: + el = json.loads(el.string) + if '@graph' in el: + for x in el['@graph']: + if '@type' in x and x['@type'] == 'Recipe': + el = x + except TypeError: + pass + return el diff --git a/cookbook/helper/recipe_search.py b/cookbook/helper/recipe_search.py index 821184f5..5f6d4199 100644 --- a/cookbook/helper/recipe_search.py +++ b/cookbook/helper/recipe_search.py @@ -1,12 +1,15 @@ +from datetime import datetime, timedelta from functools import reduce from django.contrib.postgres.search import TrigramSimilarity -from django.db.models import Q +from django.db.models import Q, Case, When, Value +from django.forms import IntegerField +from cookbook.models import ViewLog from recipes import settings -def search_recipes(queryset, params): +def search_recipes(request, queryset, params): search_string = params.get('query', '') search_keywords = params.getlist('keywords', []) search_foods = params.getlist('foods', []) @@ -18,9 +21,22 @@ def search_recipes(queryset, params): search_internal = params.get('internal', None) search_random = params.get('random', False) + search_last_viewed = int(params.get('last_viewed', 0)) - if settings.DATABASES['default']['ENGINE'] in ['django.db.backends.postgresql_psycopg2', 'django.db.backends.postgresql']: - queryset = queryset.annotate(similarity=TrigramSimilarity('name', search_string), ).filter(Q(similarity__gt=0.1) | Q(name__unaccent__icontains=search_string)).order_by('-similarity') + if search_last_viewed > 0: + last_viewed_recipes = ViewLog.objects.filter(created_by=request.user, space=request.space, + created_at__gte=datetime.now() - timedelta(days=14)).order_by('pk').values_list('recipe__pk', flat=True).distinct() + + return queryset.filter(pk__in=last_viewed_recipes[len(last_viewed_recipes) - min(len(last_viewed_recipes), search_last_viewed):]) + + queryset = queryset.annotate( + new_recipe=Case(When(created_at__gte=(datetime.now() - timedelta(days=7)), then=Value(100)), + default=Value(0), )).order_by('-new_recipe', 'name') + + if settings.DATABASES['default']['ENGINE'] in ['django.db.backends.postgresql_psycopg2', + 'django.db.backends.postgresql']: + queryset = queryset.annotate(similarity=TrigramSimilarity('name', search_string), ).filter( + Q(similarity__gt=0.1) | Q(name__unaccent__icontains=search_string)).order_by('-similarity') else: queryset = queryset.filter(name__icontains=search_string) diff --git a/cookbook/helper/recipe_url_import.py b/cookbook/helper/recipe_url_import.py index 0eae969a..4facaa57 100644 --- a/cookbook/helper/recipe_url_import.py +++ b/cookbook/helper/recipe_url_import.py @@ -1,269 +1,97 @@ -import json import random import re -from json import JSONDecodeError +from isodate import parse_duration as iso_parse_duration +from isodate.isoerror import ISO8601Error +from recipe_scrapers._exceptions import ElementNotFoundInHtml -import microdata -from bs4 import BeautifulSoup -from cookbook.helper.ingredient_parser import parse as parse_ingredient +from cookbook.helper.ingredient_parser import parse as parse_single_ingredient from cookbook.models import Keyword -from django.http import JsonResponse from django.utils.dateparse import parse_duration -from django.utils.translation import gettext as _ -from recipe_scrapers import _utils - - -def get_from_html(html_text, url, space): - soup = BeautifulSoup(html_text, "html.parser") - - # first try finding ld+json as its most common - for ld in soup.find_all('script', type='application/ld+json'): - try: - ld_json = json.loads(ld.string.replace('\n', '')) - if type(ld_json) != list: - ld_json = [ld_json] - - for ld_json_item in ld_json: - # recipes type might be wrapped in @graph type - if '@graph' in ld_json_item: - for x in ld_json_item['@graph']: - if '@type' in x and x['@type'] == 'Recipe': - ld_json_item = x - - if ('@type' in ld_json_item - and ld_json_item['@type'] == 'Recipe'): - return JsonResponse(find_recipe_json(ld_json_item, url, space)) - except JSONDecodeError: - return JsonResponse( - { - 'error': True, - 'msg': _('The requested site provided malformed data and cannot be read.') # noqa: E501 - }, - status=400) - - # now try to find microdata - items = microdata.get_items(html_text) - for i in items: - md_json = json.loads(i.json()) - if 'schema.org/Recipe' in str(md_json['type']): - return JsonResponse(find_recipe_json(md_json['properties'], url, space)) - - return JsonResponse( - { - 'error': True, - 'msg': _('The requested site does not provide any recognized data format to import the recipe from.') # noqa: E501 - }, - status=400) - - -def find_recipe_json(ld_json, url, space): - if type(ld_json['name']) == list: - try: - ld_json['name'] = ld_json['name'][0] - except Exception: - ld_json['name'] = 'ERROR' - - # some sites use ingredients instead of recipeIngredients - if 'recipeIngredient' not in ld_json and 'ingredients' in ld_json: - ld_json['recipeIngredient'] = ld_json['ingredients'] - - if 'recipeIngredient' in ld_json: - # some pages have comma separated ingredients in a single array entry - if (len(ld_json['recipeIngredient']) == 1 - and type(ld_json['recipeIngredient']) == list): - ld_json['recipeIngredient'] = ld_json['recipeIngredient'][0].split(',') # noqa: E501 - elif type(ld_json['recipeIngredient']) == str: - ld_json['recipeIngredient'] = ld_json['recipeIngredient'].split(',') - - for x in ld_json['recipeIngredient']: - if '\n' in x: - ld_json['recipeIngredient'].remove(x) - for i in x.split('\n'): - ld_json['recipeIngredient'].insert(0, i) - - ingredients = [] - - for x in ld_json['recipeIngredient']: - if x.replace(' ', '') != '': - x = x.replace('½', "0.5").replace('¼', "0.25").replace('¾', "0.75") - try: - amount, unit, ingredient, note = parse_ingredient(x) - if ingredient: - ingredients.append( - { - 'amount': amount, - 'unit': { - 'text': unit, - 'id': random.randrange(10000, 99999) - }, - 'ingredient': { - 'text': ingredient, - 'id': random.randrange(10000, 99999) - }, - 'note': note, - 'original': x - } - ) - except Exception: - ingredients.append( - { - 'amount': 0, - 'unit': { - 'text': '', - 'id': random.randrange(10000, 99999) - }, - 'ingredient': { - 'text': x, - 'id': random.randrange(10000, 99999) - }, - 'note': '', - 'original': x - } - ) - - ld_json['recipeIngredient'] = ingredients - else: - ld_json['recipeIngredient'] = [] - - if 'keywords' in ld_json: - ld_json['keywords'] = parse_keywords(listify_keywords(ld_json['keywords']), space) - - if 'recipeInstructions' in ld_json: - instructions = '' - - # flatten instructions if they are in a list - if type(ld_json['recipeInstructions']) == list: - for i in ld_json['recipeInstructions']: - if type(i) == str: - instructions += i - else: - if 'text' in i: - instructions += i['text'] + '\n\n' - elif 'itemListElement' in i: - for ile in i['itemListElement']: - if type(ile) == str: - instructions += ile + '\n\n' - elif 'text' in ile: - instructions += ile['text'] + '\n\n' - else: - instructions += str(i) - ld_json['recipeInstructions'] = instructions - - ld_json['recipeInstructions'] = re.sub(r'\n\s*\n', '\n\n', ld_json['recipeInstructions']) # noqa: E501 - ld_json['recipeInstructions'] = re.sub(' +', ' ', ld_json['recipeInstructions']) # noqa: E501 - ld_json['recipeInstructions'] = ld_json['recipeInstructions'].replace('

', '') # noqa: E501 - ld_json['recipeInstructions'] = ld_json['recipeInstructions'].replace('

', '') # noqa: E501 - else: - ld_json['recipeInstructions'] = '' - - if url != '': - ld_json['recipeInstructions'] += '\n\n' + _('Imported from') + ' ' + url - - if 'image' in ld_json: - # check if list of images is returned, take first if so - if (type(ld_json['image'])) == list: - if type(ld_json['image'][0]) == str: - ld_json['image'] = ld_json['image'][0] - elif 'url' in ld_json['image'][0]: - ld_json['image'] = ld_json['image'][0]['url'] - - # ignore relative image paths - if 'http' not in ld_json['image']: - ld_json['image'] = '' - - if 'cookTime' in ld_json: - try: - if (type(ld_json['cookTime']) == list - and len(ld_json['cookTime']) > 0): - ld_json['cookTime'] = ld_json['cookTime'][0] - ld_json['cookTime'] = round( - parse_duration( - ld_json['cookTime'] - ).seconds / 60 - ) - except TypeError: - ld_json['cookTime'] = 0 - else: - ld_json['cookTime'] = 0 - - if 'prepTime' in ld_json: - try: - if (type(ld_json['prepTime']) == list - and len(ld_json['prepTime']) > 0): - ld_json['prepTime'] = ld_json['prepTime'][0] - ld_json['prepTime'] = round( - parse_duration( - ld_json['prepTime'] - ).seconds / 60 - ) - except TypeError: - ld_json['prepTime'] = 0 - else: - ld_json['prepTime'] = 0 - - ld_json['servings'] = 1 - try: - if 'recipeYield' in ld_json: - if type(ld_json['recipeYield']) == str: - ld_json['servings'] = int(re.findall(r'\b\d+\b', ld_json['recipeYield'])[0]) - elif type(ld_json['recipeYield']) == list: - ld_json['servings'] = int(re.findall(r'\b\d+\b', ld_json['recipeYield'][0])[0]) - except Exception as e: - print(e) - - for key in list(ld_json): - if key not in [ - 'prepTime', 'cookTime', 'image', 'recipeInstructions', - 'keywords', 'name', 'recipeIngredient', 'servings', 'description' - ]: - ld_json.pop(key, None) - - return ld_json +from html import unescape +from recipe_scrapers._schemaorg import SchemaOrgException +from recipe_scrapers._utils import get_minutes def get_from_scraper(scrape, space): # converting the scrape_me object to the existing json format based on ld+json - recipe_json = {} - recipe_json['name'] = scrape.title() + try: + recipe_json['name'] = parse_name(scrape.title() or None) + except Exception: + recipe_json['name'] = None + if not recipe_json['name']: + try: + recipe_json['name'] = scrape.schema.data.get('name') or '' + except Exception: + recipe_json['name'] = '' try: description = scrape.schema.data.get("description") or '' - recipe_json['prepTime'] = _utils.get_minutes(scrape.schema.data.get("prepTime")) or 0 - recipe_json['cookTime'] = _utils.get_minutes(scrape.schema.data.get("cookTime")) or 0 - except AttributeError: + except Exception: description = '' - recipe_json['prepTime'] = 0 - recipe_json['cookTime'] = 0 - recipe_json['description'] = description + recipe_json['description'] = parse_description(description) try: - servings = scrape.yields() - servings = int(re.findall(r'\b\d+\b', servings)[0]) - except (AttributeError, ValueError, IndexError): - servings = 1 - recipe_json['servings'] = servings + servings = scrape.yields() or None + except Exception: + servings = None + if not servings: + try: + servings = scrape.schema.data.get('recipeYield') or 1 + except Exception: + servings = 1 + if type(servings) != int: + try: + servings = int(re.findall(r'\b\d+\b', servings)[0]) + except Exception: + servings = 1 + recipe_json['servings'] = max(servings, 1) + try: + recipe_json['prepTime'] = get_minutes(scrape.schema.data.get("prepTime")) or 0 + except Exception: + recipe_json['prepTime'] = 0 + try: + recipe_json['cookTime'] = get_minutes(scrape.schema.data.get("cookTime")) or 0 + except Exception: + recipe_json['cookTime'] = 0 if recipe_json['cookTime'] + recipe_json['prepTime'] == 0: try: - recipe_json['prepTime'] = scrape.total_time() - except AttributeError: - pass + recipe_json['prepTime'] = get_minutes(scrape.total_time()) or 0 + except Exception: + try: + get_minutes(scrape.schema.data.get("totalTime")) or 0 + except Exception: + pass try: - recipe_json['image'] = scrape.image() - except AttributeError: - pass + recipe_json['image'] = parse_image(scrape.image()) or None + except Exception: + recipe_json['image'] = None + if not recipe_json['image']: + try: + recipe_json['image'] = parse_image(scrape.schema.data.get('image')) or '' + except Exception: + recipe_json['image'] = '' keywords = [] try: if scrape.schema.data.get("keywords"): keywords += listify_keywords(scrape.schema.data.get("keywords")) + except Exception: + pass + try: if scrape.schema.data.get('recipeCategory'): keywords += listify_keywords(scrape.schema.data.get("recipeCategory")) + except Exception: + pass + try: if scrape.schema.data.get('recipeCuisine'): keywords += listify_keywords(scrape.schema.data.get("recipeCuisine")) + except Exception: + pass + try: recipe_json['keywords'] = parse_keywords(list(set(map(str.casefold, keywords))), space) except AttributeError: recipe_json['keywords'] = keywords @@ -272,23 +100,22 @@ def get_from_scraper(scrape, space): ingredients = [] for x in scrape.ingredients(): try: - amount, unit, ingredient, note = parse_ingredient(x) - if ingredient: - ingredients.append( - { - 'amount': amount, - 'unit': { - 'text': unit, - 'id': random.randrange(10000, 99999) - }, - 'ingredient': { - 'text': ingredient, - 'id': random.randrange(10000, 99999) - }, - 'note': note, - 'original': x - } - ) + amount, unit, ingredient, note = parse_single_ingredient(x) + ingredients.append( + { + 'amount': amount, + 'unit': { + 'text': unit, + 'id': random.randrange(10000, 99999) + }, + 'ingredient': { + 'text': ingredient, + 'id': random.randrange(10000, 99999) + }, + 'note': note, + 'original': x + } + ) except Exception: ingredients.append( { @@ -306,38 +133,228 @@ def get_from_scraper(scrape, space): } ) recipe_json['recipeIngredient'] = ingredients - except AttributeError: + except Exception: recipe_json['recipeIngredient'] = ingredients try: - recipe_json['recipeInstructions'] = scrape.instructions() - except AttributeError: + recipe_json['recipeInstructions'] = parse_instructions(scrape.instructions()) + except Exception: recipe_json['recipeInstructions'] = "" - recipe_json['recipeInstructions'] += "\n\nImported from " + scrape.url + if scrape.url: + recipe_json['url'] = scrape.url + recipe_json['recipeInstructions'] += "\n\nImported from " + scrape.url return recipe_json +def parse_name(name): + if type(name) == list: + try: + name = name[0] + except Exception: + name = 'ERROR' + return normalize_string(name) + + +def parse_ingredients(ingredients): + # some pages have comma separated ingredients in a single array entry + try: + if type(ingredients[0]) == dict: + return ingredients + except (KeyError, IndexError): + pass + + if (len(ingredients) == 1 and type(ingredients) == list): + ingredients = ingredients[0].split(',') + elif type(ingredients) == str: + ingredients = ingredients.split(',') + + for x in ingredients: + if '\n' in x: + ingredients.remove(x) + for i in x.split('\n'): + ingredients.insert(0, i) + + ingredient_list = [] + + for x in ingredients: + if x.replace(' ', '') != '': + x = x.replace('½', "0.5").replace('¼', "0.25").replace('¾', "0.75") + try: + amount, unit, ingredient, note = parse_single_ingredient(x) + if ingredient: + ingredient_list.append( + { + 'amount': amount, + 'unit': { + 'text': unit, + 'id': random.randrange(10000, 99999) + }, + 'ingredient': { + 'text': ingredient, + 'id': random.randrange(10000, 99999) + }, + 'note': note, + 'original': x + } + ) + except Exception: + ingredient_list.append( + { + 'amount': 0, + 'unit': { + 'text': '', + 'id': random.randrange(10000, 99999) + }, + 'ingredient': { + 'text': x, + 'id': random.randrange(10000, 99999) + }, + 'note': '', + 'original': x + } + ) + + ingredients = ingredient_list + else: + ingredients = [] + return ingredients + + +def parse_description(description): + return normalize_string(description) + + +def parse_instructions(instructions): + instruction_text = '' + + # flatten instructions if they are in a list + if type(instructions) == list: + for i in instructions: + if type(i) == str: + instruction_text += i + else: + if 'text' in i: + instruction_text += i['text'] + '\n\n' + elif 'itemListElement' in i: + for ile in i['itemListElement']: + if type(ile) == str: + instruction_text += ile + '\n\n' + elif 'text' in ile: + instruction_text += ile['text'] + '\n\n' + else: + instruction_text += str(i) + instructions = instruction_text + + return normalize_string(instructions) + + +def parse_image(image): + # check if list of images is returned, take first if so + if not image: + return None + if type(image) == list: + for pic in image: + if (type(pic) == str) and (pic[:4] == 'http'): + image = pic + elif 'url' in pic: + image = pic['url'] + elif type(image) == dict: + if 'url' in image: + image = image['url'] + + # ignore relative image paths + if image[:4] != 'http': + image = '' + return image + + +def parse_servings(servings): + if type(servings) == str: + try: + servings = int(re.search(r'\d+', servings).group()) + except AttributeError: + servings = 1 + elif type(servings) == list: + try: + servings = int(re.findall(r'\b\d+\b', servings[0])[0]) + except KeyError: + servings = 1 + return servings + + +def parse_cooktime(cooktime): + if type(cooktime) not in [int, float]: + try: + cooktime = float(re.search(r'\d+', cooktime).group()) + except (ValueError, AttributeError): + try: + cooktime = round(iso_parse_duration(cooktime).seconds / 60) + except ISO8601Error: + try: + if (type(cooktime) == list and len(cooktime) > 0): + cooktime = cooktime[0] + cooktime = round(parse_duration(cooktime).seconds / 60) + except AttributeError: + cooktime = 0 + + return cooktime + + +def parse_preptime(preptime): + if type(preptime) not in [int, float]: + try: + preptime = float(re.search(r'\d+', preptime).group()) + except ValueError: + try: + preptime = round(iso_parse_duration(preptime).seconds / 60) + except ISO8601Error: + try: + if (type(preptime) == list and len(preptime) > 0): + preptime = preptime[0] + preptime = round(parse_duration(preptime).seconds / 60) + except AttributeError: + preptime = 0 + + return preptime + + def parse_keywords(keyword_json, space): keywords = [] # keywords as list for kw in keyword_json: - if k := Keyword.objects.filter(name=kw, space=space).first(): - keywords.append({'id': str(k.id), 'text': str(k)}) - else: - keywords.append({'id': random.randrange(1111111, 9999999, 1), 'text': kw}) + kw = normalize_string(kw) + if len(kw) != 0: + if k := Keyword.objects.filter(name=kw, space=space).first(): + keywords.append({'id': str(k.id), 'text': str(k)}) + else: + keywords.append({'id': random.randrange(1111111, 9999999, 1), 'text': kw}) return keywords def listify_keywords(keyword_list): # keywords as string + try: + if type(keyword_list[0]) == dict: + return keyword_list + except (KeyError, IndexError): + pass if type(keyword_list) == str: keyword_list = keyword_list.split(',') # keywords as string in list - if (type(keyword_list) == list - and len(keyword_list) == 1 - and ',' in keyword_list[0]): + if (type(keyword_list) == list and len(keyword_list) == 1 and ',' in keyword_list[0]): keyword_list = keyword_list[0].split(',') return [x.strip() for x in keyword_list] + + +def normalize_string(string): + # Convert all named and numeric character references (e.g. >, >) + unescaped_string = unescape(string) + unescaped_string = re.sub('<[^<]+?>', '', unescaped_string) + unescaped_string = re.sub(' +', ' ', unescaped_string) + unescaped_string = re.sub('

', '\n', unescaped_string) + unescaped_string = re.sub(r'\n\s*\n', '\n\n', unescaped_string) + unescaped_string = unescaped_string.replace("\xa0", " ").replace("\t", " ").strip() + return unescaped_string diff --git a/cookbook/helper/scope_middleware.py b/cookbook/helper/scope_middleware.py index 6b5191df..809a7eb1 100644 --- a/cookbook/helper/scope_middleware.py +++ b/cookbook/helper/scope_middleware.py @@ -16,6 +16,12 @@ class ScopeMiddleware: with scopes_disabled(): return self.get_response(request) + if request.path.startswith('/signup/') or request.path.startswith('/invite/'): + return self.get_response(request) + + if request.path.startswith('/accounts/'): + return self.get_response(request) + with scopes_disabled(): if request.user.userpreference.space is None and not reverse('account_logout') in request.path: return views.no_space(request) diff --git a/cookbook/helper/scrapers/cooksillustrated.py b/cookbook/helper/scrapers/cooksillustrated.py new file mode 100644 index 00000000..e1e54a97 --- /dev/null +++ b/cookbook/helper/scrapers/cooksillustrated.py @@ -0,0 +1,68 @@ +import json +from recipe_scrapers._abstract import AbstractScraper + + +class CooksIllustrated(AbstractScraper): + @classmethod + def host(cls, site='cooksillustrated'): + return { + 'cooksillustrated': f"{site}.com", + 'americastestkitchen': f"{site}.com", + 'cookscountry': f"{site}.com", + }.get(site) + + def title(self): + return self.schema.title() + + def image(self): + return self.schema.image() + + def total_time(self): + if not self.recipe: + self.get_recipe() + return self.recipe['recipeTimeNote'] + + def yields(self): + if not self.recipe: + self.get_recipe() + return self.recipe['yields'] + + def ingredients(self): + if not self.recipe: + self.get_recipe() + ingredients = [] + for group in self.recipe['ingredientGroups']: + ingredients += group['fields']['recipeIngredientItems'] + return [ + "{} {} {}{}".format( + i['fields']['qty'] or '', + i['fields']['measurement'] or '', + i['fields']['ingredient']['fields']['title'] or '', + i['fields']['postText'] or '' + ) + for i in ingredients + ] + + def instructions(self): + if not self.recipe: + self.get_recipe() + if self.recipe.get('headnote', False): + i = ['Note: ' + self.recipe.get('headnote', '')] + else: + i = [] + return "\n".join( + i + + [self.recipe.get('whyThisWorks', '')] + + [ + instruction['fields']['content'] + for instruction in self.recipe['instructions'] + ] + ) + + def nutrients(self): + raise NotImplementedError("This should be implemented.") + + def get_recipe(self): + j = json.loads(self.soup.find(type='application/json').string) + name = list(j['props']['initialState']['content']['documents'])[0] + self.recipe = j['props']['initialState']['content']['documents'][name] diff --git a/cookbook/helper/scrapers/scrapers.py b/cookbook/helper/scrapers/scrapers.py new file mode 100644 index 00000000..4c41474e --- /dev/null +++ b/cookbook/helper/scrapers/scrapers.py @@ -0,0 +1,43 @@ +from bs4 import BeautifulSoup +from json import JSONDecodeError +from recipe_scrapers import SCRAPERS, get_host_name +from recipe_scrapers._factory import SchemaScraperFactory +from recipe_scrapers._schemaorg import SchemaOrg + +from .cooksillustrated import CooksIllustrated + +CUSTOM_SCRAPERS = { + CooksIllustrated.host(site="cooksillustrated"): CooksIllustrated, + CooksIllustrated.host(site="americastestkitchen"): CooksIllustrated, + CooksIllustrated.host(site="cookscountry"): CooksIllustrated, +} +SCRAPERS.update(CUSTOM_SCRAPERS) + + +def text_scraper(text, url=None): + domain = None + if url: + domain = get_host_name(url) + if domain in SCRAPERS: + scraper_class = SCRAPERS[domain] + else: + scraper_class = SchemaScraperFactory.SchemaScraper + + class TextScraper(scraper_class): + def __init__( + self, + page_data, + url=None + ): + self.wild_mode = False + # self.exception_handling = None # TODO add new method here, old one was deprecated + self.meta_http_equiv = False + self.soup = BeautifulSoup(page_data, "html.parser") + self.url = url + self.recipe = None + try: + self.schema = SchemaOrg(page_data) + except (JSONDecodeError, AttributeError): + pass + + return TextScraper(text, url) diff --git a/cookbook/integration/integration.py b/cookbook/integration/integration.py index ba55c58d..ef24372a 100644 --- a/cookbook/integration/integration.py +++ b/cookbook/integration/integration.py @@ -57,9 +57,8 @@ class Integration: recipe_stream.write(data) recipe_zip_obj.writestr(filename, recipe_stream.getvalue()) recipe_stream.close() - try: - recipe_zip_obj.write(r.image.path, 'image.png') + recipe_zip_obj.writestr('image.png', r.image.file.read()) except ValueError: pass @@ -120,29 +119,51 @@ class Integration: import_zip = ZipFile(f['file']) for z in import_zip.filelist: if self.import_file_name_filter(z): - recipe = self.get_recipe_from_file(BytesIO(import_zip.read(z.filename))) - recipe.keywords.add(self.keyword) - il.msg += f'{recipe.pk} - {recipe.name} \n' - self.handle_duplicates(recipe, import_duplicates) - + try: + recipe = self.get_recipe_from_file(BytesIO(import_zip.read(z.filename))) + recipe.keywords.add(self.keyword) + il.msg += f'{recipe.pk} - {recipe.name} \n' + self.handle_duplicates(recipe, import_duplicates) + except Exception as e: + il.msg += f'-------------------- \n ERROR \n{e}\n--------------------\n' import_zip.close() elif '.json' in f['name'] or '.txt' in f['name']: data_list = self.split_recipe_file(f['file']) for d in data_list: - recipe = self.get_recipe_from_file(d) - recipe.keywords.add(self.keyword) - il.msg += f'{recipe.pk} - {recipe.name} \n' - self.handle_duplicates(recipe, import_duplicates) + try: + recipe = self.get_recipe_from_file(d) + recipe.keywords.add(self.keyword) + il.msg += f'{recipe.pk} - {recipe.name} \n' + self.handle_duplicates(recipe, import_duplicates) + except Exception as e: + il.msg += f'-------------------- \n ERROR \n{e}\n--------------------\n' + elif '.rtk' in f['name']: + import_zip = ZipFile(f['file']) + for z in import_zip.filelist: + if self.import_file_name_filter(z): + data_list = self.split_recipe_file(import_zip.read(z.filename).decode('utf-8')) + for d in data_list: + try: + recipe = self.get_recipe_from_file(d) + recipe.keywords.add(self.keyword) + il.msg += f'{recipe.pk} - {recipe.name} \n' + self.handle_duplicates(recipe, import_duplicates) + except Exception as e: + il.msg += f'-------------------- \n ERROR \n{e}\n--------------------\n' + import_zip.close() else: recipe = self.get_recipe_from_file(f['file']) recipe.keywords.add(self.keyword) il.msg += f'{recipe.pk} - {recipe.name} \n' self.handle_duplicates(recipe, import_duplicates) except BadZipFile: - il.msg += 'ERROR ' + _('Importer expected a .zip file. Did you choose the correct importer type for your data ?') + '\n' + il.msg += 'ERROR ' + _( + 'Importer expected a .zip file. Did you choose the correct importer type for your data ?') + '\n' if len(self.ignored_recipes) > 0: - il.msg += '\n' + _('The following recipes were ignored because they already existed:') + ' ' + ', '.join(self.ignored_recipes) + '\n\n' + il.msg += '\n' + _( + 'The following recipes were ignored because they already existed:') + ' ' + ', '.join( + self.ignored_recipes) + '\n\n' il.keyword = self.keyword il.msg += (_('Imported %s recipes.') % Recipe.objects.filter(keywords=self.keyword).count()) + '\n' diff --git a/cookbook/integration/mealie.py b/cookbook/integration/mealie.py index 198883c8..a67640e0 100644 --- a/cookbook/integration/mealie.py +++ b/cookbook/integration/mealie.py @@ -16,8 +16,10 @@ class Mealie(Integration): def get_recipe_from_file(self, file): recipe_json = json.loads(file.getvalue().decode("utf-8")) + description = '' if len(recipe_json['description'].strip()) > 500 else recipe_json['description'].strip() + recipe = Recipe.objects.create( - name=recipe_json['name'].strip(), description=recipe_json['description'].strip(), + name=recipe_json['name'].strip(), description=description, created_by=self.request.user, internal=True, space=self.request.space) # TODO parse times (given in PT2H3M ) @@ -30,6 +32,9 @@ class Mealie(Integration): if not ingredients_added: ingredients_added = True + if len(recipe_json['description'].strip()) > 500: + step.instruction = recipe_json['description'].strip() + '\n\n' + step.instruction + for ingredient in recipe_json['recipeIngredient']: amount, unit, ingredient, note = parse(ingredient) f = get_food(ingredient, self.request.space) diff --git a/cookbook/integration/nextcloud_cookbook.py b/cookbook/integration/nextcloud_cookbook.py index 6857032a..e52e383f 100644 --- a/cookbook/integration/nextcloud_cookbook.py +++ b/cookbook/integration/nextcloud_cookbook.py @@ -11,13 +11,15 @@ from cookbook.models import Recipe, Step, Food, Unit, Ingredient class NextcloudCookbook(Integration): def import_file_name_filter(self, zip_info_object): - return re.match(r'^Recipes/([A-Za-z\d\s])+/recipe.json$', zip_info_object.filename) + return zip_info_object.filename.endswith('.json') def get_recipe_from_file(self, file): recipe_json = json.loads(file.getvalue().decode("utf-8")) + description = '' if len(recipe_json['description'].strip()) > 500 else recipe_json['description'].strip() + recipe = Recipe.objects.create( - name=recipe_json['name'].strip(), description=recipe_json['description'].strip(), + name=recipe_json['name'].strip(), description=description, created_by=self.request.user, internal=True, servings=recipe_json['recipeYield'], space=self.request.space) @@ -30,6 +32,9 @@ class NextcloudCookbook(Integration): instruction=s ) if not ingredients_added: + if len(recipe_json['description'].strip()) > 500: + step.instruction = recipe_json['description'].strip() + '\n\n' + step.instruction + ingredients_added = True for ingredient in recipe_json['recipeIngredient']: diff --git a/cookbook/integration/openeats.py b/cookbook/integration/openeats.py new file mode 100644 index 00000000..09edef68 --- /dev/null +++ b/cookbook/integration/openeats.py @@ -0,0 +1,71 @@ +import json +import re + +from django.utils.translation import gettext as _ + +from cookbook.helper.ingredient_parser import parse, get_food, get_unit +from cookbook.integration.integration import Integration +from cookbook.models import Recipe, Step, Food, Unit, Ingredient + + +class OpenEats(Integration): + + def get_recipe_from_file(self, file): + recipe = Recipe.objects.create(name=file['name'].strip(), created_by=self.request.user, internal=True, + servings=file['servings'], space=self.request.space, waiting_time=file['cook_time'], working_time=file['prep_time']) + + instructions = '' + if file["info"] != '': + instructions += file["info"] + + if file["directions"] != '': + instructions += file["directions"] + + if file["source"] != '': + instructions += file["source"] + + step = Step.objects.create(instruction=instructions) + + for ingredient in file['ingredients']: + f = get_food(ingredient['food'], self.request.space) + u = get_unit(ingredient['unit'], self.request.space) + step.ingredients.add(Ingredient.objects.create( + food=f, unit=u, amount=ingredient['amount'] + )) + recipe.steps.add(step) + + return recipe + + def split_recipe_file(self, file): + recipe_json = json.loads(file.read()) + recipe_dict = {} + ingredient_group_dict = {} + + for o in recipe_json: + if o['model'] == 'recipe.recipe': + recipe_dict[o['pk']] = { + 'name': o['fields']['title'], + 'info': o['fields']['info'], + 'directions': o['fields']['directions'], + 'source': o['fields']['source'], + 'prep_time': o['fields']['prep_time'], + 'cook_time': o['fields']['cook_time'], + 'servings': o['fields']['servings'], + 'ingredients': [], + } + if o['model'] == 'ingredient.ingredientgroup': + ingredient_group_dict[o['pk']] = o['fields']['recipe'] + + for o in recipe_json: + if o['model'] == 'ingredient.ingredient': + ingredient = { + 'food': o['fields']['title'], + 'unit': o['fields']['measurement'], + 'amount': round(o['fields']['numerator'] / o['fields']['denominator'], 2), + } + recipe_dict[ingredient_group_dict[o['fields']['ingredient_group']]]['ingredients'].append(ingredient) + + return list(recipe_dict.values()) + + def get_file_from_recipe(self, recipe): + raise NotImplementedError('Method not implemented in storage integration') diff --git a/cookbook/integration/paprika.py b/cookbook/integration/paprika.py index 2e0b6d10..40dcc0e8 100644 --- a/cookbook/integration/paprika.py +++ b/cookbook/integration/paprika.py @@ -20,11 +20,13 @@ class Paprika(Integration): recipe_json = json.loads(recipe_zip.read().decode("utf-8")) recipe = Recipe.objects.create( - name=recipe_json['name'].strip(), description=recipe_json['description'].strip(), - created_by=self.request.user, internal=True, space=self.request.space) + name=recipe_json['name'].strip(), created_by=self.request.user, internal=True, space=self.request.space) + + if 'description' in recipe_json: + recipe.description = '' if len(recipe_json['description'].strip()) > 500 else recipe_json['description'].strip() try: - if re.match(r'([0-9])+\s(.)*', recipe_json['servings'] ): + if re.match(r'([0-9])+\s(.)*', recipe_json['servings']): s = recipe_json['servings'].split(' ') recipe.servings = s[0] recipe.servings_text = s[1] @@ -40,34 +42,45 @@ class Paprika(Integration): recipe.save() instructions = recipe_json['directions'] - if len(recipe_json['notes'].strip()) > 0: + if recipe_json['notes'] and len(recipe_json['notes'].strip()) > 0: instructions += '\n\n### ' + _('Notes') + ' \n' + recipe_json['notes'] - if len(recipe_json['nutritional_info'].strip()) > 0: + if recipe_json['nutritional_info'] and len(recipe_json['nutritional_info'].strip()) > 0: instructions += '\n\n### ' + _('Nutritional Information') + ' \n' + recipe_json['nutritional_info'] - if len(recipe_json['source'].strip()) > 0 or len(recipe_json['source_url'].strip()) > 0: - instructions += '\n\n### ' + _('Source') + ' \n' + recipe_json['source'].strip() + ' \n' + recipe_json['source_url'].strip() + try: + if len(recipe_json['source'].strip()) > 0 or len(recipe_json['source_url'].strip()) > 0: + instructions += '\n\n### ' + _('Source') + ' \n' + recipe_json['source'].strip() + ' \n' + recipe_json['source_url'].strip() + except AttributeError: + pass step = Step.objects.create( instruction=instructions ) + if len(recipe_json['description'].strip()) > 500: + step.instruction = recipe_json['description'].strip() + '\n\n' + step.instruction + if 'categories' in recipe_json: for c in recipe_json['categories']: keyword, created = Keyword.objects.get_or_create(name=c.strip(), space=self.request.space) recipe.keywords.add(keyword) - for ingredient in recipe_json['ingredients'].split('\n'): - if len(ingredient.strip()) > 0: - amount, unit, ingredient, note = parse(ingredient) - f = get_food(ingredient, self.request.space) - u = get_unit(unit, self.request.space) - step.ingredients.add(Ingredient.objects.create( - food=f, unit=u, amount=amount, note=note - )) + try: + for ingredient in recipe_json['ingredients'].split('\n'): + if len(ingredient.strip()) > 0: + amount, unit, ingredient, note = parse(ingredient) + f = get_food(ingredient, self.request.space) + u = get_unit(unit, self.request.space) + step.ingredients.add(Ingredient.objects.create( + food=f, unit=u, amount=amount, note=note + )) + except AttributeError: + pass recipe.steps.add(step) - self.import_recipe_image(recipe, BytesIO(base64.b64decode(recipe_json['photo_data']))) + if recipe_json.get("photo_data", None): + self.import_recipe_image(recipe, BytesIO(base64.b64decode(recipe_json['photo_data']))) + return recipe diff --git a/cookbook/integration/recettetek.py b/cookbook/integration/recettetek.py new file mode 100644 index 00000000..2f51f47c --- /dev/null +++ b/cookbook/integration/recettetek.py @@ -0,0 +1,135 @@ +import re +import json +import base64 +import requests +from io import BytesIO +from zipfile import ZipFile +import imghdr +from django.utils.translation import gettext as _ + +from cookbook.helper.ingredient_parser import parse, get_food, get_unit +from cookbook.integration.integration import Integration +from cookbook.models import Recipe, Step, Food, Unit, Ingredient, Keyword + + +class RecetteTek(Integration): + + def import_file_name_filter(self, zip_info_object): + print("testing", zip_info_object.filename) + return re.match(r'^recipes_0.json$', zip_info_object.filename) or re.match(r'^recipes.json$', zip_info_object.filename) + + def split_recipe_file(self, file): + + recipe_json = json.loads(file) + + recipe_list = [r for r in recipe_json] + + return recipe_list + + def get_recipe_from_file(self, file): + + # Create initial recipe with just a title and a decription + recipe = Recipe.objects.create(name=file['title'], created_by=self.request.user, internal=True, space=self.request.space, ) + + # set the description as an empty string for later use for the source URL, incase there is no description text. + recipe.description = '' + + try: + if file['description'] != '': + recipe.description = file['description'].strip() + except Exception as e: + print(recipe.name, ': failed to parse recipe description ', str(e)) + + instructions = file['instructions'] + if not instructions: + instructions = '' + + step = Step.objects.create(instruction=instructions) + + # Append the original import url to the step (if it exists) + try: + if file['url'] != '': + step.instruction += '\n\nImported from: ' + file['url'] + step.save() + except Exception as e: + print(recipe.name, ': failed to import source url ', str(e)) + + try: + # Process the ingredients. Assumes 1 ingredient per line. + for ingredient in file['ingredients'].split('\n'): + if len(ingredient.strip()) > 0: + amount, unit, ingredient, note = parse(ingredient) + f = get_food(ingredient, self.request.space) + u = get_unit(unit, self.request.space) + step.ingredients.add(Ingredient.objects.create( + food=f, unit=u, amount=amount, note=note + )) + except Exception as e: + print(recipe.name, ': failed to parse recipe ingredients ', str(e)) + recipe.steps.add(step) + + # Attempt to import prep/cooking times + # quick hack, this assumes only one number in the quantity field. + try: + if file['quantity'] != '': + for item in file['quantity'].split(' '): + if item.isdigit(): + recipe.servings = int(item) + break + except Exception as e: + print(recipe.name, ': failed to parse quantity ', str(e)) + + try: + if file['totalTime'] != '': + recipe.waiting_time = int(file['totalTime']) + except Exception as e: + print(recipe.name, ': failed to parse total times ', str(e)) + + try: + if file['preparationTime'] != '': + recipe.working_time = int(file['preparationTime']) + except Exception as e: + print(recipe.name, ': failed to parse prep time ', str(e)) + + try: + if file['cookingTime'] != '': + recipe.waiting_time = int(file['cookingTime']) + except Exception as e: + print(recipe.name, ': failed to parse cooking time ', str(e)) + + recipe.save() + + # Import the recipe keywords + try: + if file['keywords'] != '': + for keyword in file['keywords'].split(';'): + k, created = Keyword.objects.get_or_create(name=keyword.strip(), space=self.request.space) + recipe.keywords.add(k) + recipe.save() + except Exception as e: + pass + + # TODO: Parse Nutritional Information + + # Import the original image from the zip file, if we cannot do that, attempt to download it again. + try: + if file['pictures'][0] !='': + image_file_name = file['pictures'][0].split('/')[-1] + for f in self.files: + if '.rtk' in f['name']: + import_zip = ZipFile(f['file']) + self.import_recipe_image(recipe, BytesIO(import_zip.read(image_file_name))) + else: + if file['originalPicture'] != '': + response=requests.get(file['originalPicture']) + if imghdr.what(BytesIO(response.content)) != None: + self.import_recipe_image(recipe, BytesIO(response.content)) + else: + raise Exception("Original image failed to download.") + except Exception as e: + print(recipe.name, ': failed to import image ', str(e)) + + return recipe + + def get_file_from_recipe(self, recipe): + raise NotImplementedError('Method not implemented in storage integration') diff --git a/cookbook/locale/de/LC_MESSAGES/django.po b/cookbook/locale/de/LC_MESSAGES/django.po index e7138d43..9393eca9 100644 --- a/cookbook/locale/de/LC_MESSAGES/django.po +++ b/cookbook/locale/de/LC_MESSAGES/django.po @@ -15,8 +15,8 @@ msgstr "" "Project-Id-Version: PACKAGE VERSION\n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-04-11 15:09+0200\n" -"PO-Revision-Date: 2021-04-11 15:23+0000\n" -"Last-Translator: vabene1111 \n" +"PO-Revision-Date: 2021-05-01 13:01+0000\n" +"Last-Translator: Marcel Paluch \n" "Language-Team: German \n" "Language: de\n" @@ -43,7 +43,9 @@ msgstr "" #: .\cookbook\forms.py:46 msgid "Default Unit to be used when inserting a new ingredient into a recipe." -msgstr "Standardeinheit für neue Zutaten." +msgstr "" +"Standardeinheit, die beim Einfügen einer neuen Zutat in ein Rezept zu " +"verwenden ist." #: .\cookbook\forms.py:47 msgid "" @@ -71,7 +73,9 @@ msgstr "Anzahl an Dezimalstellen, auf die gerundet werden soll." #: .\cookbook\forms.py:51 msgid "If you want to be able to create and see comments underneath recipes." -msgstr "Ob Kommentare unter Rezepten erstellt und angesehen werden können." +msgstr "" +"Wenn du in der Lage sein willst, Kommentare unter Rezepten zu erstellen und " +"zu sehen." #: .\cookbook\forms.py:53 msgid "" @@ -94,7 +98,7 @@ msgid "" "Both fields are optional. If none are given the username will be displayed " "instead" msgstr "" -"Beide Felder sind optional, wenn keins von beiden gegeben ist, wird der " +"Beide Felder sind optional. Wenn keins von beiden gegeben ist, wird der " "Nutzername angezeigt" #: .\cookbook\forms.py:93 .\cookbook\forms.py:315 @@ -123,7 +127,7 @@ msgstr "Pfad" #: .\cookbook\forms.py:98 msgid "Storage UID" -msgstr "Speicher-ID" +msgstr "Speicher-UID" #: .\cookbook\forms.py:121 msgid "Default" @@ -135,7 +139,7 @@ msgid "" "ignored. Check this box to import everything." msgstr "" "Um Duplikate zu vermeiden werden Rezepte mit dem gleichen Namen ignoriert. " -"Checken sie diese Box um alle Rezepte zu importieren." +"Aktivieren Sie dieses Kontrollkästchen, um alles zu importieren." #: .\cookbook\forms.py:149 msgid "New Unit" @@ -204,8 +208,8 @@ msgstr "Mindestens ein Rezept oder ein Titel müssen angegeben werden." #: .\cookbook\forms.py:367 msgid "You can list default users to share recipes with in the settings." msgstr "" -"Benutzer, mit denen neue Rezepte standardmäßig geteilt werden sollen, können " -"in den Einstellungen angegeben werden." +"Sie können in den Einstellungen Standardbenutzer auflisten, für die Sie " +"Rezepte freigeben möchten." #: .\cookbook\forms.py:368 #: .\cookbook\templates\forms\edit_internal_recipe.html:377 @@ -213,8 +217,8 @@ msgid "" "You can use markdown to format this field. See the docs here" msgstr "" -"Markdown kann genutzt werden, um dieses Feld zu formatieren. Siehe hier für weitere Information." +"Markdown kann genutzt werden, um dieses Feld zu formatieren. Siehe hier für weitere Information" #: .\cookbook\forms.py:393 msgid "A username is not required, if left blank the new user can choose one." @@ -309,8 +313,9 @@ msgstr "Quelle" #: .\cookbook\templates\forms\edit_internal_recipe.html:75 #: .\cookbook\templates\include\log_cooking.html:16 #: .\cookbook\templates\url_import.html:84 +#, fuzzy msgid "Servings" -msgstr "Portion(en)" +msgstr "Portionen" #: .\cookbook\integration\safron.py:25 msgid "Waiting time" diff --git a/cookbook/locale/nl/LC_MESSAGES/django.po b/cookbook/locale/nl/LC_MESSAGES/django.po index 34cc330b..968a11d1 100644 --- a/cookbook/locale/nl/LC_MESSAGES/django.po +++ b/cookbook/locale/nl/LC_MESSAGES/django.po @@ -13,7 +13,7 @@ msgstr "" "Project-Id-Version: PACKAGE VERSION\n" "Report-Msgid-Bugs-To: \n" "POT-Creation-Date: 2021-04-11 15:09+0200\n" -"PO-Revision-Date: 2021-04-22 18:29+0000\n" +"PO-Revision-Date: 2021-05-04 09:02+0000\n" "Last-Translator: Jesse \n" "Language-Team: Dutch \n" @@ -1441,7 +1441,7 @@ msgstr "Geschatte wachttijd" #: .\cookbook\templates\recipes_table.html:60 msgid "External" -msgstr "Extern" +msgstr "Externe" #: .\cookbook\templates\recipes_table.html:86 msgid "Log Cooking" diff --git a/cookbook/migrations/0120_bookmarklet.py b/cookbook/migrations/0120_bookmarklet.py new file mode 100644 index 00000000..24c40261 --- /dev/null +++ b/cookbook/migrations/0120_bookmarklet.py @@ -0,0 +1,34 @@ +# Generated by Django 3.1.7 on 2021-03-29 11:05 + +import cookbook.models +from django.conf import settings +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + + dependencies = [ + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ('cookbook', '0119_auto_20210411_2101'), + ] + + operations = [ + migrations.AlterField( + model_name='userpreference', + name='use_fractions', + field=models.BooleanField(default=True), + ), + migrations.CreateModel( + name='BookmarkletImport', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('html', models.TextField()), + ('url', models.CharField(blank=True, max_length=256, null=True)), + ('created_at', models.DateTimeField(auto_now_add=True)), + ('created_by', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), + ('space', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='cookbook.space')), + ], + bases=(models.Model, cookbook.models.PermissionModelMixin), + ), + ] diff --git a/cookbook/migrations/0121_auto_20210518_1638.py b/cookbook/migrations/0121_auto_20210518_1638.py new file mode 100644 index 00000000..678bfd36 --- /dev/null +++ b/cookbook/migrations/0121_auto_20210518_1638.py @@ -0,0 +1,23 @@ +# Generated by Django 3.2.3 on 2021-05-18 14:38 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0120_bookmarklet'), + ] + + operations = [ + migrations.AlterField( + model_name='userpreference', + name='search_style', + field=models.CharField(choices=[('SMALL', 'Small'), ('LARGE', 'Large'), ('NEW', 'New')], default='LARGE', max_length=64), + ), + migrations.AlterField( + model_name='userpreference', + name='use_fractions', + field=models.BooleanField(default=False), + ), + ] diff --git a/cookbook/migrations/0122_auto_20210527_1712.py b/cookbook/migrations/0122_auto_20210527_1712.py new file mode 100644 index 00000000..796225ca --- /dev/null +++ b/cookbook/migrations/0122_auto_20210527_1712.py @@ -0,0 +1,23 @@ +# Generated by Django 3.2.3 on 2021-05-27 15:12 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0121_auto_20210518_1638'), + ] + + operations = [ + migrations.AddField( + model_name='space', + name='allow_files', + field=models.BooleanField(default=True), + ), + migrations.AddField( + model_name='space', + name='max_users', + field=models.IntegerField(default=0), + ), + ] diff --git a/cookbook/migrations/0123_invitelink_email.py b/cookbook/migrations/0123_invitelink_email.py new file mode 100644 index 00000000..96d59b4e --- /dev/null +++ b/cookbook/migrations/0123_invitelink_email.py @@ -0,0 +1,18 @@ +# Generated by Django 3.2.3 on 2021-05-28 12:34 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0122_auto_20210527_1712'), + ] + + operations = [ + migrations.AddField( + model_name='invitelink', + name='email', + field=models.EmailField(blank=True, max_length=254), + ), + ] diff --git a/cookbook/migrations/0124_alter_userpreference_theme.py b/cookbook/migrations/0124_alter_userpreference_theme.py new file mode 100644 index 00000000..b7d7ebb3 --- /dev/null +++ b/cookbook/migrations/0124_alter_userpreference_theme.py @@ -0,0 +1,18 @@ +# Generated by Django 3.2.3 on 2021-05-30 15:53 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0123_invitelink_email'), + ] + + operations = [ + migrations.AlterField( + model_name='userpreference', + name='theme', + field=models.CharField(choices=[('BOOTSTRAP', 'Bootstrap'), ('DARKLY', 'Darkly'), ('FLATLY', 'Flatly'), ('SUPERHERO', 'Superhero'), ('TANDOOR', 'Tandoor')], default='FLATLY', max_length=128), + ), + ] diff --git a/cookbook/migrations/0125_space_demo.py b/cookbook/migrations/0125_space_demo.py new file mode 100644 index 00000000..ea252879 --- /dev/null +++ b/cookbook/migrations/0125_space_demo.py @@ -0,0 +1,18 @@ +# Generated by Django 3.2.3 on 2021-06-04 14:52 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0124_alter_userpreference_theme'), + ] + + operations = [ + migrations.AddField( + model_name='space', + name='demo', + field=models.BooleanField(default=False), + ), + ] diff --git a/cookbook/migrations/0126_alter_userpreference_theme.py b/cookbook/migrations/0126_alter_userpreference_theme.py new file mode 100644 index 00000000..08580d4b --- /dev/null +++ b/cookbook/migrations/0126_alter_userpreference_theme.py @@ -0,0 +1,18 @@ +# Generated by Django 3.2.3 on 2021-06-05 15:11 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0125_space_demo'), + ] + + operations = [ + migrations.AlterField( + model_name='userpreference', + name='theme', + field=models.CharField(choices=[('TANDOOR', 'Tandoor'), ('BOOTSTRAP', 'Bootstrap'), ('DARKLY', 'Darkly'), ('FLATLY', 'Flatly'), ('SUPERHERO', 'Superhero')], default='TANDOOR', max_length=128), + ), + ] diff --git a/cookbook/migrations/0127_remove_invitelink_username.py b/cookbook/migrations/0127_remove_invitelink_username.py new file mode 100644 index 00000000..ffb25260 --- /dev/null +++ b/cookbook/migrations/0127_remove_invitelink_username.py @@ -0,0 +1,17 @@ +# Generated by Django 3.2.3 on 2021-06-07 14:21 + +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0126_alter_userpreference_theme'), + ] + + operations = [ + migrations.RemoveField( + model_name='invitelink', + name='username', + ), + ] diff --git a/cookbook/migrations/0128_userfile.py b/cookbook/migrations/0128_userfile.py new file mode 100644 index 00000000..5f83a548 --- /dev/null +++ b/cookbook/migrations/0128_userfile.py @@ -0,0 +1,30 @@ +# Generated by Django 3.2.3 on 2021-06-08 10:23 + +import cookbook.models +from django.conf import settings +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + + dependencies = [ + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ('cookbook', '0127_remove_invitelink_username'), + ] + + operations = [ + migrations.CreateModel( + name='UserFile', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('name', models.CharField(max_length=128)), + ('file', models.FileField(upload_to='files/')), + ('file_size_kb', models.IntegerField()), + ('created_at', models.DateTimeField(auto_now_add=True)), + ('created_by', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), + ('space', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='cookbook.space')), + ], + bases=(models.Model, cookbook.models.PermissionModelMixin), + ), + ] diff --git a/cookbook/migrations/0129_auto_20210608_1233.py b/cookbook/migrations/0129_auto_20210608_1233.py new file mode 100644 index 00000000..23320d81 --- /dev/null +++ b/cookbook/migrations/0129_auto_20210608_1233.py @@ -0,0 +1,22 @@ +# Generated by Django 3.2.3 on 2021-06-08 10:33 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0128_userfile'), + ] + + operations = [ + migrations.RemoveField( + model_name='space', + name='allow_files', + ), + migrations.AddField( + model_name='space', + name='max_file_storage_mb', + field=models.IntegerField(default=0, help_text='Maximum file storage for space in MB. 0 for unlimited, -1 to disable file upload.'), + ), + ] diff --git a/cookbook/migrations/0130_alter_userfile_file_size_kb.py b/cookbook/migrations/0130_alter_userfile_file_size_kb.py new file mode 100644 index 00000000..916e29e5 --- /dev/null +++ b/cookbook/migrations/0130_alter_userfile_file_size_kb.py @@ -0,0 +1,18 @@ +# Generated by Django 3.2.3 on 2021-06-08 10:42 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0129_auto_20210608_1233'), + ] + + operations = [ + migrations.AlterField( + model_name='userfile', + name='file_size_kb', + field=models.IntegerField(blank=True, default=0), + ), + ] diff --git a/cookbook/migrations/0131_auto_20210608_1929.py b/cookbook/migrations/0131_auto_20210608_1929.py new file mode 100644 index 00000000..ce8126e5 --- /dev/null +++ b/cookbook/migrations/0131_auto_20210608_1929.py @@ -0,0 +1,24 @@ +# Generated by Django 3.2.4 on 2021-06-08 17:29 + +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + + dependencies = [ + ('cookbook', '0130_alter_userfile_file_size_kb'), + ] + + operations = [ + migrations.AddField( + model_name='step', + name='file', + field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, to='cookbook.userfile'), + ), + migrations.AlterField( + model_name='step', + name='type', + field=models.CharField(choices=[('TEXT', 'Text'), ('TIME', 'Time'), ('FILE', 'File')], default='TEXT', max_length=16), + ), + ] diff --git a/cookbook/models.py b/cookbook/models.py index 37a77c72..cf8b2103 100644 --- a/cookbook/models.py +++ b/cookbook/models.py @@ -1,3 +1,5 @@ +import operator +import pathlib import re import uuid from datetime import date, timedelta @@ -5,10 +7,12 @@ from datetime import date, timedelta from annoying.fields import AutoOneToOneField from django.contrib import auth from django.contrib.auth.models import Group, User +from django.core.files.uploadedfile import UploadedFile, InMemoryUploadedFile from django.core.validators import MinLengthValidator from django.db import models from django.utils import timezone from django.utils.translation import gettext as _ +from django_prometheus.models import ExportModelOperationsMixin from django_scopes import ScopedManager from recipes.settings import (COMMENT_PREF_DEFAULT, FRACTION_PREF_DEFAULT, @@ -52,16 +56,21 @@ class PermissionModelMixin: def get_space(self): p = '.'.join(self.get_space_key()) - if getattr(self, p, None): - return getattr(self, p) - raise NotImplementedError('get space for method not implemented and standard fields not available') + try: + if space := operator.attrgetter(p)(self): + return space + except AttributeError: + raise NotImplementedError('get space for method not implemented and standard fields not available') -class Space(models.Model): +class Space(ExportModelOperationsMixin('space'), models.Model): name = models.CharField(max_length=128, default='Default') created_by = models.ForeignKey(User, on_delete=models.PROTECT, null=True) message = models.CharField(max_length=512, default='', blank=True) max_recipes = models.IntegerField(default=0) + max_file_storage_mb = models.IntegerField(default=0, help_text=_('Maximum file storage for space in MB. 0 for unlimited, -1 to disable file upload.')) + max_users = models.IntegerField(default=0) + demo = models.BooleanField(default=False) def __str__(self): return self.name @@ -73,12 +82,14 @@ class UserPreference(models.Model, PermissionModelMixin): DARKLY = 'DARKLY' FLATLY = 'FLATLY' SUPERHERO = 'SUPERHERO' + TANDOOR = 'TANDOOR' THEMES = ( + (TANDOOR, 'Tandoor'), (BOOTSTRAP, 'Bootstrap'), (DARKLY, 'Darkly'), (FLATLY, 'Flatly'), - (SUPERHERO, 'Superhero') + (SUPERHERO, 'Superhero'), ) # Nav colors @@ -120,7 +131,7 @@ class UserPreference(models.Model, PermissionModelMixin): SEARCH_STYLE = ((SMALL, _('Small')), (LARGE, _('Large')), (NEW, _('New'))) user = AutoOneToOneField(User, on_delete=models.CASCADE, primary_key=True) - theme = models.CharField(choices=THEMES, max_length=128, default=FLATLY) + theme = models.CharField(choices=THEMES, max_length=128, default=TANDOOR) nav_color = models.CharField( choices=COLORS, max_length=128, default=PRIMARY ) @@ -243,7 +254,7 @@ class SyncLog(models.Model, PermissionModelMixin): return f"{self.created_at}:{self.sync} - {self.status}" -class Keyword(models.Model, PermissionModelMixin): +class Keyword(ExportModelOperationsMixin('keyword'), models.Model, PermissionModelMixin): name = models.CharField(max_length=64) icon = models.CharField(max_length=16, blank=True, null=True) description = models.TextField(default="", blank=True) @@ -263,7 +274,7 @@ class Keyword(models.Model, PermissionModelMixin): unique_together = (('space', 'name'),) -class Unit(models.Model, PermissionModelMixin): +class Unit(ExportModelOperationsMixin('unit'), models.Model, PermissionModelMixin): name = models.CharField(max_length=128, validators=[MinLengthValidator(1)]) description = models.TextField(blank=True, null=True) @@ -277,7 +288,7 @@ class Unit(models.Model, PermissionModelMixin): unique_together = (('space', 'name'),) -class Food(models.Model, PermissionModelMixin): +class Food(ExportModelOperationsMixin('food'), models.Model, PermissionModelMixin): name = models.CharField(max_length=128, validators=[MinLengthValidator(1)]) recipe = models.ForeignKey('Recipe', null=True, blank=True, on_delete=models.SET_NULL) supermarket_category = models.ForeignKey(SupermarketCategory, null=True, blank=True, on_delete=models.SET_NULL) @@ -294,7 +305,7 @@ class Food(models.Model, PermissionModelMixin): unique_together = (('space', 'name'),) -class Ingredient(models.Model, PermissionModelMixin): +class Ingredient(ExportModelOperationsMixin('ingredient'), models.Model, PermissionModelMixin): food = models.ForeignKey(Food, on_delete=models.PROTECT, null=True, blank=True) unit = models.ForeignKey(Unit, on_delete=models.PROTECT, null=True, blank=True) amount = models.DecimalField(default=0, decimal_places=16, max_digits=32) @@ -319,13 +330,14 @@ class Ingredient(models.Model, PermissionModelMixin): ordering = ['order', 'pk'] -class Step(models.Model, PermissionModelMixin): +class Step(ExportModelOperationsMixin('step'), models.Model, PermissionModelMixin): TEXT = 'TEXT' TIME = 'TIME' + FILE = 'FILE' name = models.CharField(max_length=128, default='', blank=True) type = models.CharField( - choices=((TEXT, _('Text')), (TIME, _('Time')),), + choices=((TEXT, _('Text')), (TIME, _('Time')), (FILE, _('File')),), default=TEXT, max_length=16 ) @@ -333,6 +345,7 @@ class Step(models.Model, PermissionModelMixin): ingredients = models.ManyToManyField(Ingredient, blank=True) time = models.IntegerField(default=0, blank=True) order = models.IntegerField(default=0) + file = models.ForeignKey('UserFile', on_delete=models.PROTECT, null=True, blank=True) show_as_header = models.BooleanField(default=True) objects = ScopedManager(space='recipe__space') @@ -376,7 +389,7 @@ class NutritionInformation(models.Model, PermissionModelMixin): return 'Nutrition' -class Recipe(models.Model, PermissionModelMixin): +class Recipe(ExportModelOperationsMixin('recipe'), models.Model, PermissionModelMixin): name = models.CharField(max_length=128) description = models.CharField(max_length=512, blank=True, null=True) servings = models.IntegerField(default=1) @@ -408,7 +421,7 @@ class Recipe(models.Model, PermissionModelMixin): return self.name -class Comment(models.Model, PermissionModelMixin): +class Comment(ExportModelOperationsMixin('comment'), models.Model, PermissionModelMixin): recipe = models.ForeignKey(Recipe, on_delete=models.CASCADE) text = models.TextField() created_by = models.ForeignKey(User, on_delete=models.CASCADE) @@ -421,6 +434,9 @@ class Comment(models.Model, PermissionModelMixin): def get_space_key(): return 'recipe', 'space' + def get_space(self): + return self.recipe.space + def __str__(self): return self.text @@ -439,7 +455,7 @@ class RecipeImport(models.Model, PermissionModelMixin): return self.name -class RecipeBook(models.Model, PermissionModelMixin): +class RecipeBook(ExportModelOperationsMixin('book'), models.Model, PermissionModelMixin): name = models.CharField(max_length=128) description = models.TextField(blank=True) icon = models.CharField(max_length=16, blank=True, null=True) @@ -453,7 +469,7 @@ class RecipeBook(models.Model, PermissionModelMixin): return self.name -class RecipeBookEntry(models.Model, PermissionModelMixin): +class RecipeBookEntry(ExportModelOperationsMixin('book_entry'), models.Model, PermissionModelMixin): recipe = models.ForeignKey(Recipe, on_delete=models.CASCADE) book = models.ForeignKey(RecipeBook, on_delete=models.CASCADE) @@ -488,7 +504,7 @@ class MealType(models.Model, PermissionModelMixin): return self.name -class MealPlan(models.Model, PermissionModelMixin): +class MealPlan(ExportModelOperationsMixin('meal_plan'), models.Model, PermissionModelMixin): recipe = models.ForeignKey(Recipe, on_delete=models.CASCADE, blank=True, null=True) servings = models.DecimalField(default=1, max_digits=8, decimal_places=4) title = models.CharField(max_length=64, blank=True, default='') @@ -513,7 +529,7 @@ class MealPlan(models.Model, PermissionModelMixin): return f'{self.get_label()} - {self.date} - {self.meal_type.name}' -class ShoppingListRecipe(models.Model, PermissionModelMixin): +class ShoppingListRecipe(ExportModelOperationsMixin('shopping_list_recipe'), models.Model, PermissionModelMixin): recipe = models.ForeignKey(Recipe, on_delete=models.CASCADE, null=True, blank=True) servings = models.DecimalField(default=1, max_digits=8, decimal_places=4) @@ -536,7 +552,7 @@ class ShoppingListRecipe(models.Model, PermissionModelMixin): return None -class ShoppingListEntry(models.Model, PermissionModelMixin): +class ShoppingListEntry(ExportModelOperationsMixin('shopping_list_entry'), models.Model, PermissionModelMixin): list_recipe = models.ForeignKey(ShoppingListRecipe, on_delete=models.CASCADE, null=True, blank=True) food = models.ForeignKey(Food, on_delete=models.CASCADE) unit = models.ForeignKey(Unit, on_delete=models.CASCADE, null=True, blank=True) @@ -566,7 +582,7 @@ class ShoppingListEntry(models.Model, PermissionModelMixin): return None -class ShoppingList(models.Model, PermissionModelMixin): +class ShoppingList(ExportModelOperationsMixin('shopping_list'), models.Model, PermissionModelMixin): uuid = models.UUIDField(default=uuid.uuid4) note = models.TextField(blank=True, null=True) recipes = models.ManyToManyField(ShoppingListRecipe, blank=True) @@ -584,7 +600,7 @@ class ShoppingList(models.Model, PermissionModelMixin): return f'Shopping list {self.id}' -class ShareLink(models.Model, PermissionModelMixin): +class ShareLink(ExportModelOperationsMixin('share_link'), models.Model, PermissionModelMixin): recipe = models.ForeignKey(Recipe, on_delete=models.CASCADE) uuid = models.UUIDField(default=uuid.uuid4) created_by = models.ForeignKey(User, on_delete=models.CASCADE) @@ -601,9 +617,9 @@ def default_valid_until(): return date.today() + timedelta(days=14) -class InviteLink(models.Model, PermissionModelMixin): +class InviteLink(ExportModelOperationsMixin('invite_link'), models.Model, PermissionModelMixin): uuid = models.UUIDField(default=uuid.uuid4) - username = models.CharField(blank=True, max_length=64) + email = models.EmailField(blank=True) group = models.ForeignKey(Group, on_delete=models.CASCADE) valid_until = models.DateField(default=default_valid_until) used_by = models.ForeignKey( @@ -633,7 +649,7 @@ class TelegramBot(models.Model, PermissionModelMixin): return f"{self.name}" -class CookLog(models.Model, PermissionModelMixin): +class CookLog(ExportModelOperationsMixin('cook_log'), models.Model, PermissionModelMixin): recipe = models.ForeignKey(Recipe, on_delete=models.CASCADE) created_by = models.ForeignKey(User, on_delete=models.CASCADE) created_at = models.DateTimeField(default=timezone.now) @@ -647,7 +663,7 @@ class CookLog(models.Model, PermissionModelMixin): return self.recipe.name -class ViewLog(models.Model, PermissionModelMixin): +class ViewLog(ExportModelOperationsMixin('view_log'), models.Model, PermissionModelMixin): recipe = models.ForeignKey(Recipe, on_delete=models.CASCADE) created_by = models.ForeignKey(User, on_delete=models.CASCADE) created_at = models.DateTimeField(auto_now_add=True) @@ -672,3 +688,30 @@ class ImportLog(models.Model, PermissionModelMixin): def __str__(self): return f"{self.created_at}:{self.type}" + + +class BookmarkletImport(ExportModelOperationsMixin('bookmarklet_import'), models.Model, PermissionModelMixin): + html = models.TextField() + url = models.CharField(max_length=256, null=True, blank=True) + created_at = models.DateTimeField(auto_now_add=True) + created_by = models.ForeignKey(User, on_delete=models.CASCADE) + + objects = ScopedManager(space='space') + space = models.ForeignKey(Space, on_delete=models.CASCADE) + + +class UserFile(ExportModelOperationsMixin('user_files'), models.Model, PermissionModelMixin): + name = models.CharField(max_length=128) + file = models.FileField(upload_to='files/') + file_size_kb = models.IntegerField(default=0, blank=True) + created_at = models.DateTimeField(auto_now_add=True) + created_by = models.ForeignKey(User, on_delete=models.CASCADE) + + objects = ScopedManager(space='space') + space = models.ForeignKey(Space, on_delete=models.CASCADE) + + def save(self, *args, **kwargs): + if hasattr(self.file, 'file') and isinstance(self.file.file, UploadedFile) or isinstance(self.file.file, InMemoryUploadedFile): + self.file.name = f'{uuid.uuid4()}' + pathlib.Path(self.file.name).suffix + self.file_size_kb = round(self.file.size / 1000) + super(UserFile, self).save(*args, **kwargs) diff --git a/cookbook/serializer.py b/cookbook/serializer.py index 1da30c67..f64e0a9f 100644 --- a/cookbook/serializer.py +++ b/cookbook/serializer.py @@ -1,20 +1,20 @@ from decimal import Decimal +from gettext import gettext as _ from django.contrib.auth.models import User -from django.db.models import QuerySet +from django.db.models import QuerySet, Sum from drf_writable_nested import (UniqueFieldsMixin, WritableNestedModelSerializer) from rest_framework import serializers -from rest_framework.exceptions import ValidationError, NotAuthenticated, NotFound, ParseError -from rest_framework.fields import ModelField -from rest_framework.serializers import BaseSerializer, Serializer +from rest_framework.exceptions import ValidationError, NotFound from cookbook.models import (Comment, CookLog, Food, Ingredient, Keyword, MealPlan, MealType, NutritionInformation, Recipe, RecipeBook, RecipeBookEntry, RecipeImport, ShareLink, ShoppingList, ShoppingListEntry, ShoppingListRecipe, Step, Storage, Sync, SyncLog, - Unit, UserPreference, ViewLog, SupermarketCategory, Supermarket, SupermarketCategoryRelation, ImportLog) + Unit, UserPreference, ViewLog, SupermarketCategory, Supermarket, + SupermarketCategoryRelation, ImportLog, BookmarkletImport, UserFile) from cookbook.templatetags.custom_tags import markdown @@ -102,6 +102,51 @@ class UserPreferenceSerializer(serializers.ModelSerializer): ) +class UserFileSerializer(serializers.ModelSerializer): + + def check_file_limit(self, validated_data): + if self.context['request'].space.max_file_storage_mb == -1: + raise ValidationError(_('File uploads are not enabled for this Space.')) + + try: + current_file_size_mb = UserFile.objects.filter(space=self.context['request'].space).aggregate(Sum('file_size_kb'))['file_size_kb__sum'] / 1000 + except TypeError: + current_file_size_mb = 0 + + if (validated_data['file'].size / 1000 / 1000 + current_file_size_mb - 5) > self.context['request'].space.max_file_storage_mb != 0: + raise ValidationError(_('You have reached your file upload limit.')) + + def create(self, validated_data): + self.check_file_limit(validated_data) + validated_data['created_by'] = self.context['request'].user + validated_data['space'] = self.context['request'].space + return super().create(validated_data) + + def update(self, instance, validated_data): + self.check_file_limit(validated_data) + return super().update(instance, validated_data) + + class Meta: + model = UserFile + fields = ('name', 'file', 'file_size_kb', 'id',) + read_only_fields = ('id', 'file_size_kb') + extra_kwargs = {"file": {"required": False, }} + + +class UserFileViewSerializer(serializers.ModelSerializer): + + def create(self, validated_data): + raise ValidationError('Cannot create File over this view') + + def update(self, instance, validated_data): + return instance + + class Meta: + model = UserFile + fields = ('name', 'file', 'id',) + read_only_fields = ('id', 'file') + + class StorageSerializer(SpacedModelSerializer): def create(self, validated_data): @@ -160,7 +205,7 @@ class KeywordSerializer(UniqueFieldsMixin, serializers.ModelSerializer): return str(obj) def create(self, validated_data): - obj, created = Keyword.objects.get_or_create(name=validated_data['name'], space=self.context['request'].space) + obj, created = Keyword.objects.get_or_create(name=validated_data['name'].strip(), space=self.context['request'].space) return obj class Meta: @@ -174,9 +219,13 @@ class KeywordSerializer(UniqueFieldsMixin, serializers.ModelSerializer): class UnitSerializer(UniqueFieldsMixin, serializers.ModelSerializer): def create(self, validated_data): - obj, created = Unit.objects.get_or_create(name=validated_data['name'], space=self.context['request'].space) + obj, created = Unit.objects.get_or_create(name=validated_data['name'].strip(), space=self.context['request'].space) return obj + def update(self, instance, validated_data): + validated_data['name'] = validated_data['name'].strip() + return super(UnitSerializer, self).update(instance, validated_data) + class Meta: model = Unit fields = ('id', 'name', 'description') @@ -217,10 +266,11 @@ class FoodSerializer(UniqueFieldsMixin, WritableNestedModelSerializer): supermarket_category = SupermarketCategorySerializer(allow_null=True, required=False) def create(self, validated_data): - obj, created = Food.objects.get_or_create(name=validated_data['name'], space=self.context['request'].space) + obj, created = Food.objects.get_or_create(name=validated_data['name'].strip(), space=self.context['request'].space) return obj def update(self, instance, validated_data): + validated_data['name'] = validated_data['name'].strip() return super(FoodSerializer, self).update(instance, validated_data) class Meta: @@ -245,6 +295,7 @@ class StepSerializer(WritableNestedModelSerializer): ingredients = IngredientSerializer(many=True) ingredients_markdown = serializers.SerializerMethodField('get_ingredients_markdown') ingredients_vue = serializers.SerializerMethodField('get_ingredients_vue') + file = UserFileViewSerializer(allow_null=True) def get_ingredients_vue(self, obj): return obj.get_instruction_render() @@ -256,7 +307,7 @@ class StepSerializer(WritableNestedModelSerializer): model = Step fields = ( 'id', 'name', 'type', 'instruction', 'ingredients', 'ingredients_markdown', - 'ingredients_vue', 'time', 'order', 'show_as_header' + 'ingredients_vue', 'time', 'order', 'show_as_header', 'file', ) @@ -349,7 +400,7 @@ class RecipeBookEntrySerializer(serializers.ModelSerializer): class MealPlanSerializer(SpacedModelSerializer, WritableNestedModelSerializer): - recipe = RecipeOverviewSerializer() + recipe = RecipeOverviewSerializer(required=False, allow_null=True) recipe_name = serializers.ReadOnlyField(source='recipe.name') meal_type_name = serializers.ReadOnlyField(source='meal_type.name') note_markdown = serializers.SerializerMethodField('get_note_markdown') @@ -473,6 +524,21 @@ class ImportLogSerializer(serializers.ModelSerializer): read_only_fields = ('created_by',) +# CORS, REST and Scopes aren't currently working +# Scopes are evaluating before REST has authenticated the user assiging a None space +# I've made the change below to fix the bookmarklet, other serializers likely need a similar/better fix +class BookmarkletImportSerializer(serializers.ModelSerializer): + def create(self, validated_data): + validated_data['created_by'] = self.context['request'].user + validated_data['space'] = self.context['request'].user.userpreference.space + return super().create(validated_data) + + class Meta: + model = BookmarkletImport + fields = ('id', 'url', 'html', 'created_by', 'created_at') + read_only_fields = ('created_by', 'space') + + # Export/Import Serializers class KeywordExportSerializer(KeywordSerializer): diff --git a/cookbook/static/assets/brand_logo.png b/cookbook/static/assets/brand_logo.png new file mode 100644 index 00000000..37cc7fbe Binary files /dev/null and b/cookbook/static/assets/brand_logo.png differ diff --git a/cookbook/static/assets/brand_logo.svg b/cookbook/static/assets/brand_logo.svg new file mode 100644 index 00000000..6e80af2f --- /dev/null +++ b/cookbook/static/assets/brand_logo.svg @@ -0,0 +1,165 @@ + +image/svg+xml + + + + +Tandoor diff --git a/cookbook/static/assets/brand_logo_white.svg b/cookbook/static/assets/brand_logo_white.svg new file mode 100644 index 00000000..cf20950e --- /dev/null +++ b/cookbook/static/assets/brand_logo_white.svg @@ -0,0 +1,167 @@ + +image/svg+xml + + + + +Tandoor diff --git a/cookbook/static/assets/header.svg b/cookbook/static/assets/header.svg new file mode 100644 index 00000000..41341cd6 --- /dev/null +++ b/cookbook/static/assets/header.svg @@ -0,0 +1,544 @@ + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/cookbook/static/assets/spinner.svg b/cookbook/static/assets/spinner.svg new file mode 100644 index 00000000..4c8ab12d --- /dev/null +++ b/cookbook/static/assets/spinner.svg @@ -0,0 +1,115 @@ + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + + + + + diff --git a/cookbook/static/css/app.min.css b/cookbook/static/css/app.min.css new file mode 100644 index 00000000..f417a5d0 --- /dev/null +++ b/cookbook/static/css/app.min.css @@ -0,0 +1,17 @@ +.spinner-tandoor { + animation: rotation 3s infinite linear; + content: url("../assets/spinner.svg"); + width: auto; + height: 20vh; + margin: 0; + padding: 0; +} + +@keyframes rotation { + from { + transform: rotate(0deg); + } + to { + transform: rotate(359deg); + } +} \ No newline at end of file diff --git a/cookbook/static/js/bookmarklet.js b/cookbook/static/js/bookmarklet.js new file mode 100644 index 00000000..434b2632 --- /dev/null +++ b/cookbook/static/js/bookmarklet.js @@ -0,0 +1,47 @@ +(function(){ + + var v = "1.3.2"; + + if (window.jQuery === undefined || window.jQuery.fn.jquery < v) { + var done = false; + var script = document.createElement("script"); + script.src = "https://ajax.googleapis.com/ajax/libs/jquery/" + v + "/jquery.min.js"; + script.onload = script.onreadystatechange = function(){ + if (!done && (!this.readyState || this.readyState == "loaded" || this.readyState == "complete")) { + done = true; + initBookmarklet(); + } + }; + document.getElementsByTagName("head")[0].appendChild(script); + } else { + initBookmarklet(); + } + function initBookmarklet() { + (window.bookmarkletTandoor = function() { + let recipe = document.documentElement.innerHTML + let windowName = "ImportRecipe" + let url = localStorage.getItem('importURL') + let redirect = localStorage.getItem('redirectURL') + let token = localStorage.getItem('token') + let params = { 'url': window.location.protocol + '//' + window.location.host + window.location.pathname, 'html' : recipe}; + + const xhr = new XMLHttpRequest(); + xhr.open('POST', url, true); + xhr.setRequestHeader('Content-Type', 'application/json'); + xhr.setRequestHeader('Authorization', 'Token ' + token); + + // listen for `onload` event + xhr.onload = () => { + // process response + if (xhr.readyState == 4 && xhr.status == 201) { + // parse JSON data + window.open(redirect.concat('?id=', JSON.parse(xhr.response).id) ) + } else { + console.error('Error!'); + } + }; + xhr.send(JSON.stringify(params)); + } + )(); + } +})(); \ No newline at end of file diff --git a/cookbook/static/js/vue-jstree.js b/cookbook/static/js/vue-jstree.js new file mode 100644 index 00000000..50d5ade5 --- /dev/null +++ b/cookbook/static/js/vue-jstree.js @@ -0,0 +1,2 @@ +!function(e,A){"object"==typeof exports&&"object"==typeof module?module.exports=A():"function"==typeof define&&define.amd?define("vue-jstree",[],A):"object"==typeof exports?exports["vue-jstree"]=A():e["vue-jstree"]=A()}(this,function(){return function(e){function A(r){if(t[r])return t[r].exports;var n=t[r]={i:r,l:!1,exports:{}};return e[r].call(n.exports,n,n.exports,A),n.l=!0,n.exports}var t={};return A.m=e,A.c=t,A.i=function(e){return e},A.d=function(e,t,r){A.o(e,t)||Object.defineProperty(e,t,{configurable:!1,enumerable:!0,get:r})},A.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return A.d(t,"a",t),t},A.o=function(e,A){return Object.prototype.hasOwnProperty.call(e,A)},A.p="dist/",A(A.s=4)}([function(e,A){e.exports=function(e,A,t,r,n){var o,a=e=e||{},l=typeof e.default;"object"!==l&&"function"!==l||(o=e,a=e.default);var i="function"==typeof a?a.options:a;A&&(i.render=A.render,i.staticRenderFns=A.staticRenderFns),r&&(i._scopeId=r);var d;if(n?(d=function(e){e=e||this.$vnode&&this.$vnode.ssrContext||this.parent&&this.parent.$vnode&&this.parent.$vnode.ssrContext,e||"undefined"==typeof __VUE_SSR_CONTEXT__||(e=__VUE_SSR_CONTEXT__),t&&t.call(this,e),e&&e._registeredComponents&&e._registeredComponents.add(n)},i._ssrRegister=d):t&&(d=t),d){var g=i.functional,s=g?i.render:i.beforeCreate;g?i.render=function(e,A){return d.call(A),s(e,A)}:i.beforeCreate=s?[].concat(s,d):[d]}return{esModule:o,exports:a,options:i}}},function(e,A,t){function r(e){t(10)}var n=t(0)(t(3),t(9),r,null,null);e.exports=n.exports},function(e,A,t){"use strict";function r(e,A,t){return A in e?Object.defineProperty(e,A,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[A]=t,e}Object.defineProperty(A,"__esModule",{value:!0}),A.default={name:"TreeItem",props:{data:{type:Object,required:!0},textFieldName:{type:String},valueFieldName:{type:String},childrenFieldName:{type:String},itemEvents:{type:Object},wholeRow:{type:Boolean,default:!1},showCheckbox:{type:Boolean,default:!1},allowTransition:{type:Boolean,default:!0},height:{type:Number,default:24},parentItem:{type:Array},draggable:{type:Boolean,default:!1},dragOverBackgroundColor:{type:String},onItemClick:{type:Function,default:function(){return!1}},onItemToggle:{type:Function,default:function(){return!1}},onItemDragStart:{type:Function,default:function(){return!1}},onItemDragEnd:{type:Function,default:function(){return!1}},onItemDrop:{type:Function,default:function(){return!1}},klass:String},data:function(){return{isHover:!1,isDragEnter:!1,model:this.data,maxHeight:0,events:{}}},watch:{isDragEnter:function(e){this.$el.style.backgroundColor=e?this.dragOverBackgroundColor:"inherit"},data:function(e){this.model=e},"model.opened":{handler:function(e,A){this.onItemToggle(this,this.model),this.handleGroupMaxHeight()},deep:!0}},computed:{isFolder:function(){return this.model[this.childrenFieldName]&&this.model[this.childrenFieldName].length},classes:function(){return[{"tree-node":!0},{"tree-open":this.model.opened},{"tree-closed":!this.model.opened},{"tree-leaf":!this.isFolder},{"tree-loading":!!this.model.loading},{"tree-drag-enter":this.isDragEnter},r({},this.klass,!!this.klass)]},anchorClasses:function(){return[{"tree-anchor":!0},{"tree-disabled":this.model.disabled},{"tree-selected":this.model.selected},{"tree-hovered":this.isHover}]},wholeRowClasses:function(){return[{"tree-wholerow":!0},{"tree-wholerow-clicked":this.model.selected},{"tree-wholerow-hovered":this.isHover}]},themeIconClasses:function(){return[{"tree-icon":!0},{"tree-themeicon":!0},r({},this.model.icon,!!this.model.icon),{"tree-themeicon-custom":!!this.model.icon}]},isWholeRow:function(){if(this.wholeRow)return void 0===this.$parent.model||!0===this.$parent.model.opened},groupStyle:function(){return{position:this.model.opened?"":"relative","max-height":this.allowTransition?this.maxHeight+"px":"","transition-duration":this.allowTransition?300*Math.ceil(this.model[this.childrenFieldName].length/100)+"ms":"","transition-property":this.allowTransition?"max-height":"",display:this.allowTransition?"block":this.model.opened?"block":"none"}}},methods:{handleItemToggle:function(e){this.isFolder&&(this.model.opened=!this.model.opened,this.onItemToggle(this,this.model))},handleGroupMaxHeight:function(){if(this.allowTransition){var e=0,A=0;if(this.model.opened){e=this.$children.length;var t=!0,r=!1,n=void 0;try{for(var o,a=this.$children[Symbol.iterator]();!(t=(o=a.next()).done);t=!0){A+=o.value.maxHeight}}catch(e){r=!0,n=e}finally{try{!t&&a.return&&a.return()}finally{if(r)throw n}}}this.maxHeight=e*this.height+A,"tree-item"===this.$parent.$options._componentTag&&this.$parent.handleGroupMaxHeight()}},handleItemClick:function(e){this.model.disabled||(this.model.selected=!this.model.selected,this.onItemClick(this,this.model,e))},handleItemMouseOver:function(){this.isHover=!0},handleItemMouseOut:function(){this.isHover=!1},handleItemDrop:function(e,A,t){this.$el.style.backgroundColor="inherit",this.onItemDrop(e,A,t)}},created:function(){var e=this,A=this,t={click:this.handleItemClick,mouseover:this.handleItemMouseOver,mouseout:this.handleItemMouseOut};for(var r in this.itemEvents)!function(r){var n=e.itemEvents[r];if(t.hasOwnProperty(r)){var o=t[r];t[r]=function(e){o(A,A.model,e),n(A,A.model,e)}}else t[r]=function(e){n(A,A.model,e)}}(r);this.events=t},mounted:function(){this.handleGroupMaxHeight()}}},function(e,A,t){"use strict";function r(e,A,t){return A in e?Object.defineProperty(e,A,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[A]=t,e}Object.defineProperty(A,"__esModule",{value:!0});var n=t(7),o=t.n(n),a="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},l=0;A.default={name:"VJstree",props:{data:{type:Array},size:{type:String,validator:function(e){return["large","small"].indexOf(e)>-1}},showCheckbox:{type:Boolean,default:!1},wholeRow:{type:Boolean,default:!1},noDots:{type:Boolean,default:!1},collapse:{type:Boolean,default:!1},multiple:{type:Boolean,default:!1},allowBatch:{type:Boolean,default:!1},allowTransition:{type:Boolean,default:!0},textFieldName:{type:String,default:"text"},valueFieldName:{type:String,default:"value"},childrenFieldName:{type:String,default:"children"},itemEvents:{type:Object,default:function(){return{}}},async:{type:Function},loadingText:{type:String,default:"Loading..."},draggable:{type:Boolean,default:!1},dragOverBackgroundColor:{type:String,default:"#C9FDC9"},klass:String},data:function(){return{draggedItem:void 0,draggedElm:void 0}},computed:{classes:function(){return[{tree:!0},{"tree-default":!this.size},r({},"tree-default-"+this.size,!!this.size),{"tree-checkbox-selection":!!this.showCheckbox},r({},this.klass,!!this.klass)]},containerClasses:function(){return[{"tree-container-ul":!0},{"tree-children":!0},{"tree-wholerow-ul":!!this.wholeRow},{"tree-no-dots":!!this.noDots}]},sizeHeight:function(){switch(this.size){case"large":return 32;case"small":return 18;default:return 24}}},methods:{initializeData:function(e){if(e&&e.length>0)for(var A in e){var t=this.initializeDataItem(e[A]);e[A]=t,this.initializeData(e[A][this.childrenFieldName])}},initializeDataItem:function(e){function A(e,A,t,r,n){this.id=e.id||l++,this[A]=e[A]||"",this[t]=e[t]||e[A],this.icon=e.icon||"",this.opened=e.opened||n,this.selected=e.selected||!1,this.disabled=e.disabled||!1,this.loading=e.loading||!1,this[r]=e[r]||[]}var t=Object.assign(new A(e,this.textFieldName,this.valueFieldName,this.childrenFieldName,this.collapse),e),r=this;return t.addBefore=function(e,A){var n=r.initializeDataItem(e),o=A.parentItem.findIndex(function(e){return e.id===t.id});A.parentItem.splice(o,0,n)},t.addAfter=function(e,A){var n=r.initializeDataItem(e),o=A.parentItem.findIndex(function(e){return e.id===t.id})+1;A.parentItem.splice(o,0,n)},t.addChild=function(e){var A=r.initializeDataItem(e);t.opened=!0,t[r.childrenFieldName].push(A)},t.openChildren=function(){t.opened=!0,r.handleRecursionNodeChildren(t,function(e){e.opened=!0})},t.closeChildren=function(){t.opened=!1,r.handleRecursionNodeChildren(t,function(e){e.opened=!1})},t},initializeLoading:function(){var e={};return e[this.textFieldName]=this.loadingText,e.disabled=!0,e.loading=!0,this.initializeDataItem(e)},handleRecursionNodeChilds:function(e,A){if(!1!==A(e)&&e.$children&&e.$children.length>0){var t=!0,r=!1,n=void 0;try{for(var o,a=e.$children[Symbol.iterator]();!(t=(o=a.next()).done);t=!0){var l=o.value;l.disabled||this.handleRecursionNodeChilds(l,A)}}catch(e){r=!0,n=e}finally{try{!t&&a.return&&a.return()}finally{if(r)throw n}}}},handleRecursionNodeChildren:function(e,A){if(!1!==A(e)&&e[this.childrenFieldName]&&e[this.childrenFieldName].length>0){var t=!0,r=!1,n=void 0;try{for(var o,a=e[this.childrenFieldName][Symbol.iterator]();!(t=(o=a.next()).done);t=!0){var l=o.value;this.handleRecursionNodeChildren(l,A)}}catch(e){r=!0,n=e}finally{try{!t&&a.return&&a.return()}finally{if(r)throw n}}}},onItemClick:function(e,A,t){this.multiple?this.allowBatch&&this.handleBatchSelectItems(e,A):this.handleSingleSelectItems(e,A),this.$emit("item-click",e,A,t)},handleSingleSelectItems:function(e,A){this.handleRecursionNodeChilds(this,function(e){e.model&&(e.model.selected=!1)}),e.model.selected=!0},handleBatchSelectItems:function(e,A){this.handleRecursionNodeChilds(e,function(A){A.model.disabled||(A.model.selected=e.model.selected)})},onItemToggle:function(e,A,t){e.model.opened&&this.handleAsyncLoad(e.model[this.childrenFieldName],e,A),this.$emit("item-toggle",e,A,t)},handleAsyncLoad:function(e,A,t){var r=this;this.async&&e[0].loading&&this.async(A,function(t){if(t.length>0)for(var n in t){t[n].isLeaf||"object"!==a(t[n][r.childrenFieldName])&&(t[n][r.childrenFieldName]=[r.initializeLoading()]);var o=r.initializeDataItem(t[n]);r.$set(e,n,o)}else A.model[r.childrenFieldName]=[]})},onItemDragStart:function(e,A,t){if(!this.draggable||t.dragDisabled)return!1;e.dataTransfer.effectAllowed="move",e.dataTransfer.setData("text",null),this.draggedElm=e.target,this.draggedItem={item:t,parentItem:A.parentItem,index:A.parentItem.findIndex(function(e){return e.id===t.id})},this.$emit("item-drag-start",A,t,e)},onItemDragEnd:function(e,A,t){this.draggedItem=void 0,this.draggedElm=void 0,this.$emit("item-drag-end",A,t,e)},onItemDrop:function(e,A,t){var r=this;if(!this.draggable||t.dropDisabled)return!1;if(this.$emit("item-drop-before",A,t,this.draggedItem?this.draggedItem.item:void 0,e),this.draggedElm&&this.draggedElm!==e.target&&!this.draggedElm.contains(e.target)&&this.draggedItem){if(this.draggedItem.parentItem===t[this.childrenFieldName]||this.draggedItem.item===t||t[this.childrenFieldName]&&-1!==t[this.childrenFieldName].findIndex(function(e){return e.id===r.draggedItem.item.id}))return;t[this.childrenFieldName]?t[this.childrenFieldName].push(this.draggedItem.item):t[this.childrenFieldName]=[this.draggedItem.item],t.opened=!0;var n=this.draggedItem;this.$nextTick(function(){n.parentItem.splice(n.index,1)}),this.$emit("item-drop",A,t,n.item,e)}}},created:function(){this.initializeData(this.data)},mounted:function(){this.async&&(this.$set(this.data,0,this.initializeLoading()),this.handleAsyncLoad(this.data,this))},components:{TreeItem:o.a}}},function(e,A,t){"use strict";Object.defineProperty(A,"__esModule",{value:!0});var r=t(1),n=t.n(r);n.a.install=function(e){e.component(n.a.name,n.a)},"undefined"!=typeof window&&window.Vue&&window.Vue.use(n.a),A.default=n.a},function(e,A,t){A=e.exports=t(6)(),A.push([e.i,'.tree-children,.tree-container-ul,.tree-node{display:block;margin:0;padding:0;list-style-type:none;list-style-image:none}.tree-children{overflow:hidden}.tree-anchor,.tree-node{white-space:nowrap}.tree-anchor{display:inline-block;color:#000;padding:0 4px 0 1px;margin:0;vertical-align:top;font-size:14px;cursor:pointer}.tree-anchor:focus{outline:0}.tree-anchor,.tree-anchor:active,.tree-anchor:hover,.tree-anchor:link,.tree-anchor:visited{text-decoration:none;color:inherit}.tree-icon,.tree-icon:empty{display:inline-block;text-decoration:none;margin:0;padding:0;vertical-align:top;text-align:center}.tree-ocl{cursor:pointer}.tree-leaf>.tree-ocl{cursor:default}.tree-anchor>.tree-themeicon{margin-right:2px}.tree-anchor>.tree-themeicon-hidden,.tree-hidden,.tree-no-icons .tree-themeicon,.tree-node.tree-hidden{display:none}.tree-rtl .tree-anchor{padding:0 1px 0 4px}.tree-rtl .tree-anchor>.tree-themeicon{margin-left:2px;margin-right:0}.tree-rtl .tree-node{margin-left:0}.tree-rtl .tree-container-ul>.tree-node{margin-right:0}.tree-wholerow-ul{position:relative;display:inline-block;min-width:100%}.tree-wholerow-ul .tree-leaf>.tree-ocl{cursor:pointer}.tree-wholerow-ul .tree-anchor,.tree-wholerow-ul .tree-icon{position:relative}.tree-wholerow-ul .tree-wholerow{width:100%;cursor:pointer;z-index:-1;position:absolute;left:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.tree{text-align:left}.tree-default .tree-icon,.tree-default .tree-node{background-repeat:no-repeat;background-color:transparent}.tree-default .tree-anchor,.tree-default .tree-animated,.tree-default .tree-wholerow{transition:background-color .15s,box-shadow .15s}.tree-default .tree-context,.tree-default .tree-hovered{background:#eee;border:0;box-shadow:none}.tree-default .tree-selected{background:#e1e1e1;border:0;box-shadow:none}.tree-default .tree-no-icons .tree-anchor>.tree-themeicon{display:none}.tree-default .tree-disabled{color:#666}.tree-default .tree-disabled.tree-hovered{box-shadow:none}.tree-default .tree-disabled>.tree-icon{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default .tree-search{font-style:italic;color:#8b0000;font-weight:700}.tree-default .tree-no-checkboxes .tree-checkbox{display:none!important}.tree-default.tree-checkbox-no-clicked .tree-selected{background:transparent;box-shadow:none}.tree-default.tree-checkbox-no-clicked .tree-selected.tree-hovered{background:#eee}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked{background:transparent}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked.tree-wholerow-hovered{background:#eee}.tree-default>.tree-striped{min-width:100%;display:inline-block;background:url("") 0 0 repeat}.tree-default>.tree-wholerow-ul .tree-hovered,.tree-default>.tree-wholerow-ul .tree-selected{background:transparent;box-shadow:none;border-radius:0}.tree-default .tree-wholerow{box-sizing:border-box}.tree-default .tree-wholerow-hovered{background:#eee}.tree-default .tree-wholerow-clicked{background:#e1e1e1}.tree-default .tree-node{min-height:24px;line-height:24px;margin-left:30px;min-width:24px}.tree-default .tree-anchor,.tree-default .tree-icon{line-height:24px;height:24px}.tree-default .tree-icon{width:24px}.tree-default .tree-icon:empty{width:24px;height:24px;line-height:24px}.tree-default.tree-rtl .tree-node{margin-right:24px}.tree-default .tree-wholerow{height:24px}.tree-default .tree-icon,.tree-default .tree-node{background-image:url("")}.tree-default .tree-node{background-position:-292px -4px;background-repeat:repeat-y}.tree-default .tree-last{background:transparent}.tree-default .tree-open>.tree-ocl{background-position:-132px -4px}.tree-default .tree-closed>.tree-ocl{background-position:-100px -4px}.tree-default .tree-leaf>.tree-ocl{background-position:-68px -4px}.tree-default .tree-themeicon{background-position:-260px -4px}.tree-default>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default>.tree-no-dots .tree-node{background:transparent}.tree-default>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -4px}.tree-default>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -4px}.tree-default .tree-disabled,.tree-default .tree-disabled.tree-hovered{background:transparent}.tree-default .tree-disabled.tree-selected{background:#efefef}.tree-default .tree-checkbox{background-position:-164px -4px}.tree-default .tree-checkbox:hover{background-position:-164px -36px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default .tree-checked>.tree-checkbox{background-position:-228px -4px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default .tree-checked>.tree-checkbox:hover{background-position:-228px -36px}.tree-default .tree-anchor>.tree-undetermined{background-position:-196px -4px}.tree-default .tree-anchor>.tree-undetermined:hover{background-position:-196px -36px}.tree-default .tree-checkbox-disabled{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default>.tree-striped{background-size:auto 48px}.tree-default.tree-rtl .tree-node{background-position:100% 1px;background-repeat:repeat-y}.tree-default.tree-rtl .tree-open>.tree-ocl{background-position:-132px -36px}.tree-default.tree-rtl .tree-closed>.tree-ocl{background-position:-100px -36px}.tree-default.tree-rtl .tree-leaf>.tree-ocl{background-position:-68px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -36px}.tree-default .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default .tree-node.tree-loading{background:none}.tree-default>.tree-container-ul .tree-loading>.tree-ocl{background:url("") 50% no-repeat}.tree-default .tree-file{background:url("") -100px -68px no-repeat}.tree-default .tree-folder{background:url("") -260px -4px no-repeat}.tree-default>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default .tree-ellipsis{overflow:hidden}.tree-default .tree-ellipsis .tree-anchor{width:calc(100% - 29px);text-overflow:ellipsis;overflow:hidden}.tree-default .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default.tree-rtl .tree-node{background-image:url("")}.tree-default.tree-rtl .tree-last{background:transparent}.tree-default-small .tree-node{min-height:18px;line-height:18px;margin-left:24px;min-width:18px}.tree-default-small .tree-anchor{line-height:18px;height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-icon:empty{width:18px;height:18px;line-height:18px}.tree-default-small.tree-rtl .tree-node{margin-right:18px}.tree-default-small .tree-wholerow{height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-node{background-image:url("")}.tree-default-small .tree-node{background-position:-295px -7px;background-repeat:repeat-y}.tree-default-small .tree-last{background:transparent}.tree-default-small .tree-open>.tree-ocl{background-position:-135px -7px}.tree-default-small .tree-closed>.tree-ocl{background-position:-103px -7px}.tree-default-small .tree-leaf>.tree-ocl{background-position:-71px -7px}.tree-default-small .tree-themeicon{background-position:-263px -7px}.tree-default-small>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small>.tree-no-dots .tree-node{background:transparent}.tree-default-small>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -7px}.tree-default-small>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -7px}.tree-default-small .tree-disabled,.tree-default-small .tree-disabled.tree-hovered{background:transparent}.tree-default-small .tree-disabled.tree-selected{background:#efefef}.tree-default-small .tree-checkbox{background-position:-167px -7px}.tree-default-small .tree-checkbox:hover{background-position:-167px -39px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-small .tree-checked>.tree-checkbox{background-position:-231px -7px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-small .tree-checked>.tree-checkbox:hover{background-position:-231px -39px}.tree-default-small .tree-anchor>.tree-undetermined{background-position:-199px -7px}.tree-default-small .tree-anchor>.tree-undetermined:hover{background-position:-199px -39px}.tree-default-small .tree-checkbox-disabled{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-small>.tree-striped{background-size:auto 36px}.tree-default-small.tree-rtl .tree-node{background-image:url("");background-position:100% 1px;background-repeat:repeat-y}.tree-default-small.tree-rtl .tree-open>.tree-ocl{background-position:-135px -39px}.tree-default-small.tree-rtl .tree-closed>.tree-ocl{background-position:-103px -39px}.tree-default-small.tree-rtl .tree-leaf>.tree-ocl{background-position:-71px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-small.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -39px}.tree-default-small .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-small .tree-node.tree-loading{background:none}.tree-default-small>.tree-container-ul .tree-loading>.tree-ocl{background:url("") 50% no-repeat}.tree-default-small .tree-file{background:url("") -103px -71px no-repeat}.tree-default-small .tree-folder{background:url("") -263px -7px no-repeat}.tree-default-small>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-small .tree-ellipsis{overflow:hidden}.tree-default-small .tree-ellipsis .tree-anchor{width:calc(100% - 23px);text-overflow:ellipsis;overflow:hidden}.tree-default-small .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-small.tree-rtl .tree-node{background-image:url("")}.tree-default-small.tree-rtl .tree-last{background:transparent}.tree-default-large .tree-node{min-height:32px;line-height:32px;margin-left:38px;min-width:32px}.tree-default-large .tree-anchor{line-height:32px;height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-icon:empty{width:32px;height:32px;line-height:32px}.tree-default-large.tree-rtl .tree-node{margin-right:32px}.tree-default-large .tree-wholerow{height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-node{background-image:url("")}.tree-default-large .tree-node{background-position:-288px 0;background-repeat:repeat-y}.tree-default-large .tree-last{background:transparent}.tree-default-large .tree-open>.tree-ocl{background-position:-128px 0}.tree-default-large .tree-closed>.tree-ocl{background-position:-96px 0}.tree-default-large .tree-leaf>.tree-ocl{background-position:-64px 0}.tree-default-large .tree-themeicon{background-position:-256px 0}.tree-default-large>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large>.tree-no-dots .tree-node{background:transparent}.tree-default-large>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px 0}.tree-default-large>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 0}.tree-default-large .tree-disabled,.tree-default-large .tree-disabled.tree-hovered{background:transparent}.tree-default-large .tree-disabled.tree-selected{background:#efefef}.tree-default-large .tree-checkbox{background-position:-160px 0}.tree-default-large .tree-checkbox:hover{background-position:-160px -32px}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-large .tree-checked>.tree-checkbox{background-position:-224px 0}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-large .tree-checked>.tree-checkbox:hover{background-position:-224px -32px}.tree-default-large .tree-anchor>.tree-undetermined{background-position:-192px 0}.tree-default-large .tree-anchor>.tree-undetermined:hover{background-position:-192px -32px}.tree-default-large .tree-checkbox-disabled{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-large>.tree-striped{background-size:auto 64px}.tree-default-large.tree-rtl .tree-node{background-image:url("");background-position:100% 1px;background-repeat:repeat-y}.tree-default-large.tree-rtl .tree-open>.tree-ocl{background-position:-128px -32px}.tree-default-large.tree-rtl .tree-closed>.tree-ocl{background-position:-96px -32px}.tree-default-large.tree-rtl .tree-leaf>.tree-ocl{background-position:-64px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-large.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 -32px}.tree-default-large .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-large .tree-node.tree-loading{background:none}.tree-default-large>.tree-container-ul .tree-loading>.tree-ocl{background:url("") 50% no-repeat}.tree-default-large .tree-file{background:url("") -96px -64px no-repeat}.tree-default-large .tree-folder{background:url("") -256px 0 no-repeat}.tree-default-large>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-large .tree-ellipsis{overflow:hidden}.tree-default-large .tree-ellipsis .tree-anchor{width:calc(100% - 37px);text-overflow:ellipsis;overflow:hidden}.tree-default-large .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-large.tree-rtl .tree-node{background-image:url("")}.tree-default-large.tree-rtl .tree-last{background:transparent}',""])},function(e,A){e.exports=function(){var e=[];return e.toString=function(){for(var e=[],A=0;At.parts.length&&(r.parts.length=t.parts.length)}else{for(var a=[],n=0;n -1;\n } },\n showCheckbox: { type: Boolean, default: false },\n wholeRow: { type: Boolean, default: false },\n noDots: { type: Boolean, default: false },\n collapse: { type: Boolean, default: false },\n multiple: { type: Boolean, default: false },\n allowBatch: { type: Boolean, default: false },\n allowTransition: { type: Boolean, default: true },\n textFieldName: { type: String, default: 'text' },\n valueFieldName: { type: String, default: 'value' },\n childrenFieldName: { type: String, default: 'children' },\n itemEvents: {\n type: Object, default: function _default() {\n return {};\n }\n },\n async: { type: Function },\n loadingText: { type: String, default: 'Loading...' },\n draggable: { type: Boolean, default: false },\n dragOverBackgroundColor: { type: String, default: \"#C9FDC9\" },\n klass: String\n },\n data: function data() {\n return {\n draggedItem: undefined,\n draggedElm: undefined\n };\n },\n\n computed: {\n classes: function classes() {\n return [{ 'tree': true }, { 'tree-default': !this.size }, _defineProperty({}, 'tree-default-' + this.size, !!this.size), { 'tree-checkbox-selection': !!this.showCheckbox }, _defineProperty({}, this.klass, !!this.klass)];\n },\n containerClasses: function containerClasses() {\n return [{ 'tree-container-ul': true }, { 'tree-children': true }, { 'tree-wholerow-ul': !!this.wholeRow }, { 'tree-no-dots': !!this.noDots }];\n },\n sizeHeight: function sizeHeight() {\n switch (this.size) {\n case 'large':\n return ITEM_HEIGHT_LARGE;\n case 'small':\n return ITEM_HEIGHT_SMALL;\n default:\n return ITEM_HEIGHT_DEFAULT;\n }\n }\n },\n methods: {\n initializeData: function initializeData(items) {\n if (items && items.length > 0) {\n for (var i in items) {\n var dataItem = this.initializeDataItem(items[i]);\n items[i] = dataItem;\n this.initializeData(items[i][this.childrenFieldName]);\n }\n }\n },\n initializeDataItem: function initializeDataItem(item) {\n function Model(item, textFieldName, valueFieldName, childrenFieldName, collapse) {\n this.id = item.id || ITEM_ID++;\n this[textFieldName] = item[textFieldName] || '';\n this[valueFieldName] = item[valueFieldName] || item[textFieldName];\n this.icon = item.icon || '';\n this.opened = item.opened || collapse;\n this.selected = item.selected || false;\n this.disabled = item.disabled || false;\n this.loading = item.loading || false;\n this[childrenFieldName] = item[childrenFieldName] || [];\n }\n\n var node = Object.assign(new Model(item, this.textFieldName, this.valueFieldName, this.childrenFieldName, this.collapse), item);\n var self = this;\n node.addBefore = function (data, selectedNode) {\n var newItem = self.initializeDataItem(data);\n var index = selectedNode.parentItem.findIndex(function (t) {\n return t.id === node.id;\n });\n selectedNode.parentItem.splice(index, 0, newItem);\n };\n node.addAfter = function (data, selectedNode) {\n var newItem = self.initializeDataItem(data);\n var index = selectedNode.parentItem.findIndex(function (t) {\n return t.id === node.id;\n }) + 1;\n selectedNode.parentItem.splice(index, 0, newItem);\n };\n node.addChild = function (data) {\n var newItem = self.initializeDataItem(data);\n node.opened = true;\n node[self.childrenFieldName].push(newItem);\n };\n node.openChildren = function () {\n node.opened = true;\n self.handleRecursionNodeChildren(node, function (node) {\n node.opened = true;\n });\n };\n node.closeChildren = function () {\n node.opened = false;\n self.handleRecursionNodeChildren(node, function (node) {\n node.opened = false;\n });\n };\n return node;\n },\n initializeLoading: function initializeLoading() {\n var item = {};\n item[this.textFieldName] = this.loadingText;\n item.disabled = true;\n item.loading = true;\n return this.initializeDataItem(item);\n },\n handleRecursionNodeChilds: function handleRecursionNodeChilds(node, func) {\n if (func(node) !== false) {\n if (node.$children && node.$children.length > 0) {\n var _iteratorNormalCompletion = true;\n var _didIteratorError = false;\n var _iteratorError = undefined;\n\n try {\n for (var _iterator = node.$children[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {\n var childNode = _step.value;\n\n if (!childNode.disabled) {\n this.handleRecursionNodeChilds(childNode, func);\n }\n }\n } catch (err) {\n _didIteratorError = true;\n _iteratorError = err;\n } finally {\n try {\n if (!_iteratorNormalCompletion && _iterator.return) {\n _iterator.return();\n }\n } finally {\n if (_didIteratorError) {\n throw _iteratorError;\n }\n }\n }\n }\n }\n },\n handleRecursionNodeChildren: function handleRecursionNodeChildren(node, func) {\n if (func(node) !== false) {\n if (node[this.childrenFieldName] && node[this.childrenFieldName].length > 0) {\n var _iteratorNormalCompletion2 = true;\n var _didIteratorError2 = false;\n var _iteratorError2 = undefined;\n\n try {\n for (var _iterator2 = node[this.childrenFieldName][Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {\n var childNode = _step2.value;\n\n this.handleRecursionNodeChildren(childNode, func);\n }\n } catch (err) {\n _didIteratorError2 = true;\n _iteratorError2 = err;\n } finally {\n try {\n if (!_iteratorNormalCompletion2 && _iterator2.return) {\n _iterator2.return();\n }\n } finally {\n if (_didIteratorError2) {\n throw _iteratorError2;\n }\n }\n }\n }\n }\n },\n onItemClick: function onItemClick(oriNode, oriItem, e) {\n if (this.multiple) {\n if (this.allowBatch) {\n this.handleBatchSelectItems(oriNode, oriItem);\n }\n } else {\n this.handleSingleSelectItems(oriNode, oriItem);\n }\n this.$emit('item-click', oriNode, oriItem, e);\n },\n handleSingleSelectItems: function handleSingleSelectItems(oriNode, oriItem) {\n this.handleRecursionNodeChilds(this, function (node) {\n if (node.model) node.model.selected = false;\n });\n oriNode.model.selected = true;\n },\n handleBatchSelectItems: function handleBatchSelectItems(oriNode, oriItem) {\n this.handleRecursionNodeChilds(oriNode, function (node) {\n if (node.model.disabled) return;\n node.model.selected = oriNode.model.selected;\n });\n },\n onItemToggle: function onItemToggle(oriNode, oriItem, e) {\n if (oriNode.model.opened) {\n this.handleAsyncLoad(oriNode.model[this.childrenFieldName], oriNode, oriItem);\n }\n this.$emit('item-toggle', oriNode, oriItem, e);\n },\n handleAsyncLoad: function handleAsyncLoad(oriParent, oriNode, oriItem) {\n var self = this;\n if (this.async) {\n if (oriParent[0].loading) {\n this.async(oriNode, function (data) {\n if (data.length > 0) {\n for (var i in data) {\n if (!data[i].isLeaf) {\n if (_typeof(data[i][self.childrenFieldName]) !== \"object\") {\n data[i][self.childrenFieldName] = [self.initializeLoading()];\n }\n }\n var dataItem = self.initializeDataItem(data[i]);\n self.$set(oriParent, i, dataItem);\n }\n } else {\n oriNode.model[self.childrenFieldName] = [];\n }\n });\n }\n }\n },\n onItemDragStart: function onItemDragStart(e, oriNode, oriItem) {\n if (!this.draggable || oriItem.dragDisabled) return false;\n e.dataTransfer.effectAllowed = \"move\";\n e.dataTransfer.setData('text', null);\n this.draggedElm = e.target;\n this.draggedItem = {\n item: oriItem,\n parentItem: oriNode.parentItem,\n index: oriNode.parentItem.findIndex(function (t) {\n return t.id === oriItem.id;\n })\n };\n\n this.$emit(\"item-drag-start\", oriNode, oriItem, e);\n },\n onItemDragEnd: function onItemDragEnd(e, oriNode, oriItem) {\n this.draggedItem = undefined;\n this.draggedElm = undefined;\n this.$emit(\"item-drag-end\", oriNode, oriItem, e);\n },\n onItemDrop: function onItemDrop(e, oriNode, oriItem) {\n var _this = this;\n\n if (!this.draggable || !!oriItem.dropDisabled) return false;\n this.$emit(\"item-drop-before\", oriNode, oriItem, !this.draggedItem ? undefined : this.draggedItem.item, e);\n if (!this.draggedElm || this.draggedElm === e.target || this.draggedElm.contains(e.target)) {\n return;\n }\n if (this.draggedItem) {\n if (this.draggedItem.parentItem === oriItem[this.childrenFieldName] || this.draggedItem.item === oriItem || oriItem[this.childrenFieldName] && oriItem[this.childrenFieldName].findIndex(function (t) {\n return t.id === _this.draggedItem.item.id;\n }) !== -1) {\n return;\n }\n if (!!oriItem[this.childrenFieldName]) {\n oriItem[this.childrenFieldName].push(this.draggedItem.item);\n } else {\n oriItem[this.childrenFieldName] = [this.draggedItem.item];\n }\n oriItem.opened = true;\n var draggedItem = this.draggedItem;\n this.$nextTick(function () {\n draggedItem.parentItem.splice(draggedItem.index, 1);\n });\n this.$emit(\"item-drop\", oriNode, oriItem, draggedItem.item, e);\n }\n }\n },\n created: function created() {\n this.initializeData(this.data);\n },\n mounted: function mounted() {\n if (this.async) {\n this.$set(this.data, 0, this.initializeLoading());\n this.handleAsyncLoad(this.data, this);\n }\n },\n\n components: {\n TreeItem: __WEBPACK_IMPORTED_MODULE_0__tree_item_vue___default.a\n }\n});\n\n/***/ }),\n/* 4 */\n/***/ (function(module, __webpack_exports__, __webpack_require__) {\n\n\"use strict\";\nObject.defineProperty(__webpack_exports__, \"__esModule\", { value: true });\n/* harmony import */ var __WEBPACK_IMPORTED_MODULE_0__tree_vue__ = __webpack_require__(1);\n/* harmony import */ var __WEBPACK_IMPORTED_MODULE_0__tree_vue___default = __webpack_require__.n(__WEBPACK_IMPORTED_MODULE_0__tree_vue__);\n/**\r\n * Created by virus_zhh on 2017/9/29.\r\n */\n\n\n__WEBPACK_IMPORTED_MODULE_0__tree_vue___default.a.install = function (Vue) {\n Vue.component(__WEBPACK_IMPORTED_MODULE_0__tree_vue___default.a.name, __WEBPACK_IMPORTED_MODULE_0__tree_vue___default.a);\n};\n\nif (typeof window !== 'undefined' && window.Vue) {\n window.Vue.use(__WEBPACK_IMPORTED_MODULE_0__tree_vue___default.a);\n}\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (__WEBPACK_IMPORTED_MODULE_0__tree_vue___default.a);\n\n/***/ }),\n/* 5 */\n/***/ (function(module, exports, __webpack_require__) {\n\nexports = module.exports = __webpack_require__(6)();\n// imports\n\n\n// module\nexports.push([module.i, \".tree-children,.tree-container-ul,.tree-node{display:block;margin:0;padding:0;list-style-type:none;list-style-image:none}.tree-children{overflow:hidden}.tree-anchor,.tree-node{white-space:nowrap}.tree-anchor{display:inline-block;color:#000;padding:0 4px 0 1px;margin:0;vertical-align:top;font-size:14px;cursor:pointer}.tree-anchor:focus{outline:0}.tree-anchor,.tree-anchor:active,.tree-anchor:hover,.tree-anchor:link,.tree-anchor:visited{text-decoration:none;color:inherit}.tree-icon,.tree-icon:empty{display:inline-block;text-decoration:none;margin:0;padding:0;vertical-align:top;text-align:center}.tree-ocl{cursor:pointer}.tree-leaf>.tree-ocl{cursor:default}.tree-anchor>.tree-themeicon{margin-right:2px}.tree-anchor>.tree-themeicon-hidden,.tree-hidden,.tree-no-icons .tree-themeicon,.tree-node.tree-hidden{display:none}.tree-rtl .tree-anchor{padding:0 1px 0 4px}.tree-rtl .tree-anchor>.tree-themeicon{margin-left:2px;margin-right:0}.tree-rtl .tree-node{margin-left:0}.tree-rtl .tree-container-ul>.tree-node{margin-right:0}.tree-wholerow-ul{position:relative;display:inline-block;min-width:100%}.tree-wholerow-ul .tree-leaf>.tree-ocl{cursor:pointer}.tree-wholerow-ul .tree-anchor,.tree-wholerow-ul .tree-icon{position:relative}.tree-wholerow-ul .tree-wholerow{width:100%;cursor:pointer;z-index:-1;position:absolute;left:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.tree{text-align:left}.tree-default .tree-icon,.tree-default .tree-node{background-repeat:no-repeat;background-color:transparent}.tree-default .tree-anchor,.tree-default .tree-animated,.tree-default .tree-wholerow{transition:background-color .15s,box-shadow .15s}.tree-default .tree-context,.tree-default .tree-hovered{background:#eee;border:0;box-shadow:none}.tree-default .tree-selected{background:#e1e1e1;border:0;box-shadow:none}.tree-default .tree-no-icons .tree-anchor>.tree-themeicon{display:none}.tree-default .tree-disabled{color:#666}.tree-default .tree-disabled.tree-hovered{box-shadow:none}.tree-default .tree-disabled>.tree-icon{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default .tree-search{font-style:italic;color:#8b0000;font-weight:700}.tree-default .tree-no-checkboxes .tree-checkbox{display:none!important}.tree-default.tree-checkbox-no-clicked .tree-selected{background:transparent;box-shadow:none}.tree-default.tree-checkbox-no-clicked .tree-selected.tree-hovered{background:#eee}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked{background:transparent}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked.tree-wholerow-hovered{background:#eee}.tree-default>.tree-striped{min-width:100%;display:inline-block;background:url(\\\"\\\") 0 0 repeat}.tree-default>.tree-wholerow-ul .tree-hovered,.tree-default>.tree-wholerow-ul .tree-selected{background:transparent;box-shadow:none;border-radius:0}.tree-default .tree-wholerow{box-sizing:border-box}.tree-default .tree-wholerow-hovered{background:#eee}.tree-default .tree-wholerow-clicked{background:#e1e1e1}.tree-default .tree-node{min-height:24px;line-height:24px;margin-left:30px;min-width:24px}.tree-default .tree-anchor,.tree-default .tree-icon{line-height:24px;height:24px}.tree-default .tree-icon{width:24px}.tree-default .tree-icon:empty{width:24px;height:24px;line-height:24px}.tree-default.tree-rtl .tree-node{margin-right:24px}.tree-default .tree-wholerow{height:24px}.tree-default .tree-icon,.tree-default .tree-node{background-image:url(\\\"\\\")}.tree-default .tree-node{background-position:-292px -4px;background-repeat:repeat-y}.tree-default .tree-last{background:transparent}.tree-default .tree-open>.tree-ocl{background-position:-132px -4px}.tree-default .tree-closed>.tree-ocl{background-position:-100px -4px}.tree-default .tree-leaf>.tree-ocl{background-position:-68px -4px}.tree-default .tree-themeicon{background-position:-260px -4px}.tree-default>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default>.tree-no-dots .tree-node{background:transparent}.tree-default>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -4px}.tree-default>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -4px}.tree-default .tree-disabled,.tree-default .tree-disabled.tree-hovered{background:transparent}.tree-default .tree-disabled.tree-selected{background:#efefef}.tree-default .tree-checkbox{background-position:-164px -4px}.tree-default .tree-checkbox:hover{background-position:-164px -36px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default .tree-checked>.tree-checkbox{background-position:-228px -4px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default .tree-checked>.tree-checkbox:hover{background-position:-228px -36px}.tree-default .tree-anchor>.tree-undetermined{background-position:-196px -4px}.tree-default .tree-anchor>.tree-undetermined:hover{background-position:-196px -36px}.tree-default .tree-checkbox-disabled{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default>.tree-striped{background-size:auto 48px}.tree-default.tree-rtl .tree-node{background-position:100% 1px;background-repeat:repeat-y}.tree-default.tree-rtl .tree-open>.tree-ocl{background-position:-132px -36px}.tree-default.tree-rtl .tree-closed>.tree-ocl{background-position:-100px -36px}.tree-default.tree-rtl .tree-leaf>.tree-ocl{background-position:-68px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -36px}.tree-default .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default .tree-node.tree-loading{background:none}.tree-default>.tree-container-ul .tree-loading>.tree-ocl{background:url(\\\"\\\") 50% no-repeat}.tree-default .tree-file{background:url(\\\"\\\") -100px -68px no-repeat}.tree-default .tree-folder{background:url(\\\"\\\") -260px -4px no-repeat}.tree-default>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default .tree-ellipsis{overflow:hidden}.tree-default .tree-ellipsis .tree-anchor{width:calc(100% - 29px);text-overflow:ellipsis;overflow:hidden}.tree-default .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default.tree-rtl .tree-node{background-image:url(\\\"\\\")}.tree-default.tree-rtl .tree-last{background:transparent}.tree-default-small .tree-node{min-height:18px;line-height:18px;margin-left:24px;min-width:18px}.tree-default-small .tree-anchor{line-height:18px;height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-icon:empty{width:18px;height:18px;line-height:18px}.tree-default-small.tree-rtl .tree-node{margin-right:18px}.tree-default-small .tree-wholerow{height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-node{background-image:url(\\\"\\\")}.tree-default-small .tree-node{background-position:-295px -7px;background-repeat:repeat-y}.tree-default-small .tree-last{background:transparent}.tree-default-small .tree-open>.tree-ocl{background-position:-135px -7px}.tree-default-small .tree-closed>.tree-ocl{background-position:-103px -7px}.tree-default-small .tree-leaf>.tree-ocl{background-position:-71px -7px}.tree-default-small .tree-themeicon{background-position:-263px -7px}.tree-default-small>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small>.tree-no-dots .tree-node{background:transparent}.tree-default-small>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -7px}.tree-default-small>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -7px}.tree-default-small .tree-disabled,.tree-default-small .tree-disabled.tree-hovered{background:transparent}.tree-default-small .tree-disabled.tree-selected{background:#efefef}.tree-default-small .tree-checkbox{background-position:-167px -7px}.tree-default-small .tree-checkbox:hover{background-position:-167px -39px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-small .tree-checked>.tree-checkbox{background-position:-231px -7px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-small .tree-checked>.tree-checkbox:hover{background-position:-231px -39px}.tree-default-small .tree-anchor>.tree-undetermined{background-position:-199px -7px}.tree-default-small .tree-anchor>.tree-undetermined:hover{background-position:-199px -39px}.tree-default-small .tree-checkbox-disabled{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-small>.tree-striped{background-size:auto 36px}.tree-default-small.tree-rtl .tree-node{background-image:url(\\\"\\\");background-position:100% 1px;background-repeat:repeat-y}.tree-default-small.tree-rtl .tree-open>.tree-ocl{background-position:-135px -39px}.tree-default-small.tree-rtl .tree-closed>.tree-ocl{background-position:-103px -39px}.tree-default-small.tree-rtl .tree-leaf>.tree-ocl{background-position:-71px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-small.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -39px}.tree-default-small .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-small .tree-node.tree-loading{background:none}.tree-default-small>.tree-container-ul .tree-loading>.tree-ocl{background:url(\\\"\\\") 50% no-repeat}.tree-default-small .tree-file{background:url(\\\"\\\") -103px -71px no-repeat}.tree-default-small .tree-folder{background:url(\\\"\\\") -263px -7px no-repeat}.tree-default-small>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-small .tree-ellipsis{overflow:hidden}.tree-default-small .tree-ellipsis .tree-anchor{width:calc(100% - 23px);text-overflow:ellipsis;overflow:hidden}.tree-default-small .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-small.tree-rtl .tree-node{background-image:url(\\\"\\\")}.tree-default-small.tree-rtl .tree-last{background:transparent}.tree-default-large .tree-node{min-height:32px;line-height:32px;margin-left:38px;min-width:32px}.tree-default-large .tree-anchor{line-height:32px;height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-icon:empty{width:32px;height:32px;line-height:32px}.tree-default-large.tree-rtl .tree-node{margin-right:32px}.tree-default-large .tree-wholerow{height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-node{background-image:url(\\\"\\\")}.tree-default-large .tree-node{background-position:-288px 0;background-repeat:repeat-y}.tree-default-large .tree-last{background:transparent}.tree-default-large .tree-open>.tree-ocl{background-position:-128px 0}.tree-default-large .tree-closed>.tree-ocl{background-position:-96px 0}.tree-default-large .tree-leaf>.tree-ocl{background-position:-64px 0}.tree-default-large .tree-themeicon{background-position:-256px 0}.tree-default-large>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large>.tree-no-dots .tree-node{background:transparent}.tree-default-large>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px 0}.tree-default-large>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 0}.tree-default-large .tree-disabled,.tree-default-large .tree-disabled.tree-hovered{background:transparent}.tree-default-large .tree-disabled.tree-selected{background:#efefef}.tree-default-large .tree-checkbox{background-position:-160px 0}.tree-default-large .tree-checkbox:hover{background-position:-160px -32px}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-large .tree-checked>.tree-checkbox{background-position:-224px 0}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-large .tree-checked>.tree-checkbox:hover{background-position:-224px -32px}.tree-default-large .tree-anchor>.tree-undetermined{background-position:-192px 0}.tree-default-large .tree-anchor>.tree-undetermined:hover{background-position:-192px -32px}.tree-default-large .tree-checkbox-disabled{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-large>.tree-striped{background-size:auto 64px}.tree-default-large.tree-rtl .tree-node{background-image:url(\\\"\\\");background-position:100% 1px;background-repeat:repeat-y}.tree-default-large.tree-rtl .tree-open>.tree-ocl{background-position:-128px -32px}.tree-default-large.tree-rtl .tree-closed>.tree-ocl{background-position:-96px -32px}.tree-default-large.tree-rtl .tree-leaf>.tree-ocl{background-position:-64px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-large.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 -32px}.tree-default-large .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-large .tree-node.tree-loading{background:none}.tree-default-large>.tree-container-ul .tree-loading>.tree-ocl{background:url(\\\"\\\") 50% no-repeat}.tree-default-large .tree-file{background:url(\\\"\\\") -96px -64px no-repeat}.tree-default-large .tree-folder{background:url(\\\"\\\") -256px 0 no-repeat}.tree-default-large>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-large .tree-ellipsis{overflow:hidden}.tree-default-large .tree-ellipsis .tree-anchor{width:calc(100% - 37px);text-overflow:ellipsis;overflow:hidden}.tree-default-large .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-large.tree-rtl .tree-node{background-image:url(\\\"\\\")}.tree-default-large.tree-rtl .tree-last{background:transparent}\", \"\"]);\n\n// exports\n\n\n/***/ }),\n/* 6 */\n/***/ (function(module, exports) {\n\n/*\r\n\tMIT License http://www.opensource.org/licenses/mit-license.php\r\n\tAuthor Tobias Koppers @sokra\r\n*/\r\n// css base code, injected by the css-loader\r\nmodule.exports = function() {\r\n\tvar list = [];\r\n\r\n\t// return the list of modules as css string\r\n\tlist.toString = function toString() {\r\n\t\tvar result = [];\r\n\t\tfor(var i = 0; i < this.length; i++) {\r\n\t\t\tvar item = this[i];\r\n\t\t\tif(item[2]) {\r\n\t\t\t\tresult.push(\"@media \" + item[2] + \"{\" + item[1] + \"}\");\r\n\t\t\t} else {\r\n\t\t\t\tresult.push(item[1]);\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn result.join(\"\");\r\n\t};\r\n\r\n\t// import a list of modules into the list\r\n\tlist.i = function(modules, mediaQuery) {\r\n\t\tif(typeof modules === \"string\")\r\n\t\t\tmodules = [[null, modules, \"\"]];\r\n\t\tvar alreadyImportedModules = {};\r\n\t\tfor(var i = 0; i < this.length; i++) {\r\n\t\t\tvar id = this[i][0];\r\n\t\t\tif(typeof id === \"number\")\r\n\t\t\t\talreadyImportedModules[id] = true;\r\n\t\t}\r\n\t\tfor(i = 0; i < modules.length; i++) {\r\n\t\t\tvar item = modules[i];\r\n\t\t\t// skip already imported module\r\n\t\t\t// this implementation is not 100% perfect for weird media query combinations\r\n\t\t\t// when a module is imported multiple times with different media queries.\r\n\t\t\t// I hope this will never occur (Hey this way we have smaller bundles)\r\n\t\t\tif(typeof item[0] !== \"number\" || !alreadyImportedModules[item[0]]) {\r\n\t\t\t\tif(mediaQuery && !item[2]) {\r\n\t\t\t\t\titem[2] = mediaQuery;\r\n\t\t\t\t} else if(mediaQuery) {\r\n\t\t\t\t\titem[2] = \"(\" + item[2] + \") and (\" + mediaQuery + \")\";\r\n\t\t\t\t}\r\n\t\t\t\tlist.push(item);\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\treturn list;\r\n};\r\n\n\n/***/ }),\n/* 7 */\n/***/ (function(module, exports, __webpack_require__) {\n\nvar Component = __webpack_require__(0)(\n /* script */\n __webpack_require__(2),\n /* template */\n __webpack_require__(8),\n /* styles */\n null,\n /* scopeId */\n null,\n /* moduleIdentifier (server only) */\n null\n)\n\nmodule.exports = Component.exports\n\n\n/***/ }),\n/* 8 */\n/***/ (function(module, exports) {\n\nmodule.exports={render:function (){var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;\n return _c('li', {\n class: _vm.classes,\n attrs: {\n \"role\": \"treeitem\",\n \"draggable\": _vm.draggable\n },\n on: {\n \"dragstart\": function($event) {\n $event.stopPropagation();\n _vm.onItemDragStart($event, _vm._self, _vm._self.model)\n },\n \"dragend\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.onItemDragEnd($event, _vm._self, _vm._self.model)\n },\n \"dragover\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.isDragEnter = true\n },\n \"dragenter\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.isDragEnter = true\n },\n \"dragleave\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.isDragEnter = false\n },\n \"drop\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.handleItemDrop($event, _vm._self, _vm._self.model)\n }\n }\n }, [(_vm.isWholeRow) ? _c('div', {\n class: _vm.wholeRowClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }, [_vm._v(\" \")]) : _vm._e(), _vm._v(\" \"), _c('i', {\n staticClass: \"tree-icon tree-ocl\",\n attrs: {\n \"role\": \"presentation\"\n },\n on: {\n \"click\": _vm.handleItemToggle\n }\n }), _vm._v(\" \"), _c('div', _vm._g({\n class: _vm.anchorClasses\n }, _vm.events), [(_vm.showCheckbox && !_vm.model.loading) ? _c('i', {\n staticClass: \"tree-icon tree-checkbox\",\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _vm._t(\"default\", [(!_vm.model.loading) ? _c('i', {\n class: _vm.themeIconClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _c('span', {\n domProps: {\n \"innerHTML\": _vm._s(_vm.model[_vm.textFieldName])\n }\n })], {\n vm: this,\n model: _vm.model\n })], 2), _vm._v(\" \"), (_vm.isFolder) ? _c('ul', {\n ref: \"group\",\n staticClass: \"tree-children\",\n style: (_vm.groupStyle),\n attrs: {\n \"role\": \"group\"\n }\n }, _vm._l((_vm.model[_vm.childrenFieldName]), function(child, index) {\n return _c('tree-item', {\n key: index,\n attrs: {\n \"data\": child,\n \"text-field-name\": _vm.textFieldName,\n \"value-field-name\": _vm.valueFieldName,\n \"children-field-name\": _vm.childrenFieldName,\n \"item-events\": _vm.itemEvents,\n \"whole-row\": _vm.wholeRow,\n \"show-checkbox\": _vm.showCheckbox,\n \"allow-transition\": _vm.allowTransition,\n \"height\": _vm.height,\n \"parent-item\": _vm.model[_vm.childrenFieldName],\n \"draggable\": _vm.draggable,\n \"drag-over-background-color\": _vm.dragOverBackgroundColor,\n \"on-item-click\": _vm.onItemClick,\n \"on-item-toggle\": _vm.onItemToggle,\n \"on-item-drag-start\": _vm.onItemDragStart,\n \"on-item-drag-end\": _vm.onItemDragEnd,\n \"on-item-drop\": _vm.onItemDrop,\n \"klass\": index === _vm.model[_vm.childrenFieldName].length - 1 ? 'tree-last' : ''\n },\n scopedSlots: _vm._u([{\n key: \"default\",\n fn: function(_) {\n return [_vm._t(\"default\", [(!_vm.model.loading) ? _c('i', {\n class: _.vm.themeIconClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _c('span', {\n domProps: {\n \"innerHTML\": _vm._s(_.model[_vm.textFieldName])\n }\n })], {\n vm: _.vm,\n model: _.model\n })]\n }\n }])\n })\n })) : _vm._e()])\n},staticRenderFns: []}\n\n/***/ }),\n/* 9 */\n/***/ (function(module, exports) {\n\nmodule.exports={render:function (){var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;\n return _c('div', {\n class: _vm.classes,\n attrs: {\n \"role\": \"tree\",\n \"onselectstart\": \"return false\"\n }\n }, [_c('ul', {\n class: _vm.containerClasses,\n attrs: {\n \"role\": \"group\"\n }\n }, _vm._l((_vm.data), function(child, index) {\n return _c('tree-item', {\n key: index,\n attrs: {\n \"data\": child,\n \"text-field-name\": _vm.textFieldName,\n \"value-field-name\": _vm.valueFieldName,\n \"children-field-name\": _vm.childrenFieldName,\n \"item-events\": _vm.itemEvents,\n \"whole-row\": _vm.wholeRow,\n \"show-checkbox\": _vm.showCheckbox,\n \"allow-transition\": _vm.allowTransition,\n \"height\": _vm.sizeHeight,\n \"parent-item\": _vm.data,\n \"draggable\": _vm.draggable,\n \"drag-over-background-color\": _vm.dragOverBackgroundColor,\n \"on-item-click\": _vm.onItemClick,\n \"on-item-toggle\": _vm.onItemToggle,\n \"on-item-drag-start\": _vm.onItemDragStart,\n \"on-item-drag-end\": _vm.onItemDragEnd,\n \"on-item-drop\": _vm.onItemDrop,\n \"klass\": index === _vm.data.length - 1 ? 'tree-last' : ''\n },\n scopedSlots: _vm._u([{\n key: \"default\",\n fn: function(_) {\n return [_vm._t(\"default\", [(!_.model.loading) ? _c('i', {\n class: _.vm.themeIconClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _c('span', {\n domProps: {\n \"innerHTML\": _vm._s(_.model[_vm.textFieldName])\n }\n })], {\n vm: _.vm,\n model: _.model\n })]\n }\n }])\n })\n }))])\n},staticRenderFns: []}\n\n/***/ }),\n/* 10 */\n/***/ (function(module, exports, __webpack_require__) {\n\n// style-loader: Adds some css to the DOM by adding a \r\n\n\n\n// WEBPACK FOOTER //\n// tree.vue?0e418d98","/**\r\n * Created by virus_zhh on 2017/9/29.\r\n */\r\nimport VJstree from './tree.vue'\r\n\r\nVJstree.install = function(Vue){\r\n Vue.component(VJstree.name, VJstree)\r\n}\r\n\r\nif (typeof window !== 'undefined' && window.Vue) {\r\n window.Vue.use(VJstree);\r\n}\r\n\r\nexport default VJstree\n\n\n// WEBPACK FOOTER //\n// ./src/index.js","exports = module.exports = require(\"../node_modules/css-loader/lib/css-base.js\")();\n// imports\n\n\n// module\nexports.push([module.id, \".tree-children,.tree-container-ul,.tree-node{display:block;margin:0;padding:0;list-style-type:none;list-style-image:none}.tree-children{overflow:hidden}.tree-anchor,.tree-node{white-space:nowrap}.tree-anchor{display:inline-block;color:#000;padding:0 4px 0 1px;margin:0;vertical-align:top;font-size:14px;cursor:pointer}.tree-anchor:focus{outline:0}.tree-anchor,.tree-anchor:active,.tree-anchor:hover,.tree-anchor:link,.tree-anchor:visited{text-decoration:none;color:inherit}.tree-icon,.tree-icon:empty{display:inline-block;text-decoration:none;margin:0;padding:0;vertical-align:top;text-align:center}.tree-ocl{cursor:pointer}.tree-leaf>.tree-ocl{cursor:default}.tree-anchor>.tree-themeicon{margin-right:2px}.tree-anchor>.tree-themeicon-hidden,.tree-hidden,.tree-no-icons .tree-themeicon,.tree-node.tree-hidden{display:none}.tree-rtl .tree-anchor{padding:0 1px 0 4px}.tree-rtl .tree-anchor>.tree-themeicon{margin-left:2px;margin-right:0}.tree-rtl .tree-node{margin-left:0}.tree-rtl .tree-container-ul>.tree-node{margin-right:0}.tree-wholerow-ul{position:relative;display:inline-block;min-width:100%}.tree-wholerow-ul .tree-leaf>.tree-ocl{cursor:pointer}.tree-wholerow-ul .tree-anchor,.tree-wholerow-ul .tree-icon{position:relative}.tree-wholerow-ul .tree-wholerow{width:100%;cursor:pointer;z-index:-1;position:absolute;left:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.tree{text-align:left}.tree-default .tree-icon,.tree-default .tree-node{background-repeat:no-repeat;background-color:transparent}.tree-default .tree-anchor,.tree-default .tree-animated,.tree-default .tree-wholerow{transition:background-color .15s,box-shadow .15s}.tree-default .tree-context,.tree-default .tree-hovered{background:#eee;border:0;box-shadow:none}.tree-default .tree-selected{background:#e1e1e1;border:0;box-shadow:none}.tree-default .tree-no-icons .tree-anchor>.tree-themeicon{display:none}.tree-default .tree-disabled{color:#666}.tree-default .tree-disabled.tree-hovered{box-shadow:none}.tree-default .tree-disabled>.tree-icon{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default .tree-search{font-style:italic;color:#8b0000;font-weight:700}.tree-default .tree-no-checkboxes .tree-checkbox{display:none!important}.tree-default.tree-checkbox-no-clicked .tree-selected{background:transparent;box-shadow:none}.tree-default.tree-checkbox-no-clicked .tree-selected.tree-hovered{background:#eee}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked{background:transparent}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked.tree-wholerow-hovered{background:#eee}.tree-default>.tree-striped{min-width:100%;display:inline-block;background:url(\\\"\\\") 0 0 repeat}.tree-default>.tree-wholerow-ul .tree-hovered,.tree-default>.tree-wholerow-ul .tree-selected{background:transparent;box-shadow:none;border-radius:0}.tree-default .tree-wholerow{box-sizing:border-box}.tree-default .tree-wholerow-hovered{background:#eee}.tree-default .tree-wholerow-clicked{background:#e1e1e1}.tree-default .tree-node{min-height:24px;line-height:24px;margin-left:30px;min-width:24px}.tree-default .tree-anchor,.tree-default .tree-icon{line-height:24px;height:24px}.tree-default .tree-icon{width:24px}.tree-default .tree-icon:empty{width:24px;height:24px;line-height:24px}.tree-default.tree-rtl .tree-node{margin-right:24px}.tree-default .tree-wholerow{height:24px}.tree-default .tree-icon,.tree-default .tree-node{background-image:url(\\\"\\\")}.tree-default .tree-node{background-position:-292px -4px;background-repeat:repeat-y}.tree-default .tree-last{background:transparent}.tree-default .tree-open>.tree-ocl{background-position:-132px -4px}.tree-default .tree-closed>.tree-ocl{background-position:-100px -4px}.tree-default .tree-leaf>.tree-ocl{background-position:-68px -4px}.tree-default .tree-themeicon{background-position:-260px -4px}.tree-default>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default>.tree-no-dots .tree-node{background:transparent}.tree-default>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -4px}.tree-default>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -4px}.tree-default .tree-disabled,.tree-default .tree-disabled.tree-hovered{background:transparent}.tree-default .tree-disabled.tree-selected{background:#efefef}.tree-default .tree-checkbox{background-position:-164px -4px}.tree-default .tree-checkbox:hover{background-position:-164px -36px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default .tree-checked>.tree-checkbox{background-position:-228px -4px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default .tree-checked>.tree-checkbox:hover{background-position:-228px -36px}.tree-default .tree-anchor>.tree-undetermined{background-position:-196px -4px}.tree-default .tree-anchor>.tree-undetermined:hover{background-position:-196px -36px}.tree-default .tree-checkbox-disabled{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default>.tree-striped{background-size:auto 48px}.tree-default.tree-rtl .tree-node{background-position:100% 1px;background-repeat:repeat-y}.tree-default.tree-rtl .tree-open>.tree-ocl{background-position:-132px -36px}.tree-default.tree-rtl .tree-closed>.tree-ocl{background-position:-100px -36px}.tree-default.tree-rtl .tree-leaf>.tree-ocl{background-position:-68px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -36px}.tree-default .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default .tree-node.tree-loading{background:none}.tree-default>.tree-container-ul .tree-loading>.tree-ocl{background:url(\\\"\\\") 50% no-repeat}.tree-default .tree-file{background:url(\\\"\\\") -100px -68px no-repeat}.tree-default .tree-folder{background:url(\\\"\\\") -260px -4px no-repeat}.tree-default>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default .tree-ellipsis{overflow:hidden}.tree-default .tree-ellipsis .tree-anchor{width:calc(100% - 29px);text-overflow:ellipsis;overflow:hidden}.tree-default .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default.tree-rtl .tree-node{background-image:url(\\\"\\\")}.tree-default.tree-rtl .tree-last{background:transparent}.tree-default-small .tree-node{min-height:18px;line-height:18px;margin-left:24px;min-width:18px}.tree-default-small .tree-anchor{line-height:18px;height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-icon:empty{width:18px;height:18px;line-height:18px}.tree-default-small.tree-rtl .tree-node{margin-right:18px}.tree-default-small .tree-wholerow{height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-node{background-image:url(\\\"\\\")}.tree-default-small .tree-node{background-position:-295px -7px;background-repeat:repeat-y}.tree-default-small .tree-last{background:transparent}.tree-default-small .tree-open>.tree-ocl{background-position:-135px -7px}.tree-default-small .tree-closed>.tree-ocl{background-position:-103px -7px}.tree-default-small .tree-leaf>.tree-ocl{background-position:-71px -7px}.tree-default-small .tree-themeicon{background-position:-263px -7px}.tree-default-small>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small>.tree-no-dots .tree-node{background:transparent}.tree-default-small>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -7px}.tree-default-small>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -7px}.tree-default-small .tree-disabled,.tree-default-small .tree-disabled.tree-hovered{background:transparent}.tree-default-small .tree-disabled.tree-selected{background:#efefef}.tree-default-small .tree-checkbox{background-position:-167px -7px}.tree-default-small .tree-checkbox:hover{background-position:-167px -39px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-small .tree-checked>.tree-checkbox{background-position:-231px -7px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-small .tree-checked>.tree-checkbox:hover{background-position:-231px -39px}.tree-default-small .tree-anchor>.tree-undetermined{background-position:-199px -7px}.tree-default-small .tree-anchor>.tree-undetermined:hover{background-position:-199px -39px}.tree-default-small .tree-checkbox-disabled{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-small>.tree-striped{background-size:auto 36px}.tree-default-small.tree-rtl .tree-node{background-image:url(\\\"\\\");background-position:100% 1px;background-repeat:repeat-y}.tree-default-small.tree-rtl .tree-open>.tree-ocl{background-position:-135px -39px}.tree-default-small.tree-rtl .tree-closed>.tree-ocl{background-position:-103px -39px}.tree-default-small.tree-rtl .tree-leaf>.tree-ocl{background-position:-71px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-small.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -39px}.tree-default-small .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-small .tree-node.tree-loading{background:none}.tree-default-small>.tree-container-ul .tree-loading>.tree-ocl{background:url(\\\"\\\") 50% no-repeat}.tree-default-small .tree-file{background:url(\\\"\\\") -103px -71px no-repeat}.tree-default-small .tree-folder{background:url(\\\"\\\") -263px -7px no-repeat}.tree-default-small>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-small .tree-ellipsis{overflow:hidden}.tree-default-small .tree-ellipsis .tree-anchor{width:calc(100% - 23px);text-overflow:ellipsis;overflow:hidden}.tree-default-small .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-small.tree-rtl .tree-node{background-image:url(\\\"\\\")}.tree-default-small.tree-rtl .tree-last{background:transparent}.tree-default-large .tree-node{min-height:32px;line-height:32px;margin-left:38px;min-width:32px}.tree-default-large .tree-anchor{line-height:32px;height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-icon:empty{width:32px;height:32px;line-height:32px}.tree-default-large.tree-rtl .tree-node{margin-right:32px}.tree-default-large .tree-wholerow{height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-node{background-image:url(\\\"\\\")}.tree-default-large .tree-node{background-position:-288px 0;background-repeat:repeat-y}.tree-default-large .tree-last{background:transparent}.tree-default-large .tree-open>.tree-ocl{background-position:-128px 0}.tree-default-large .tree-closed>.tree-ocl{background-position:-96px 0}.tree-default-large .tree-leaf>.tree-ocl{background-position:-64px 0}.tree-default-large .tree-themeicon{background-position:-256px 0}.tree-default-large>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large>.tree-no-dots .tree-node{background:transparent}.tree-default-large>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px 0}.tree-default-large>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 0}.tree-default-large .tree-disabled,.tree-default-large .tree-disabled.tree-hovered{background:transparent}.tree-default-large .tree-disabled.tree-selected{background:#efefef}.tree-default-large .tree-checkbox{background-position:-160px 0}.tree-default-large .tree-checkbox:hover{background-position:-160px -32px}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-large .tree-checked>.tree-checkbox{background-position:-224px 0}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-large .tree-checked>.tree-checkbox:hover{background-position:-224px -32px}.tree-default-large .tree-anchor>.tree-undetermined{background-position:-192px 0}.tree-default-large .tree-anchor>.tree-undetermined:hover{background-position:-192px -32px}.tree-default-large .tree-checkbox-disabled{opacity:.8;filter:url(\\\"data:image/svg+xml;utf8,#tree-grayscale\\\");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-large>.tree-striped{background-size:auto 64px}.tree-default-large.tree-rtl .tree-node{background-image:url(\\\"\\\");background-position:100% 1px;background-repeat:repeat-y}.tree-default-large.tree-rtl .tree-open>.tree-ocl{background-position:-128px -32px}.tree-default-large.tree-rtl .tree-closed>.tree-ocl{background-position:-96px -32px}.tree-default-large.tree-rtl .tree-leaf>.tree-ocl{background-position:-64px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-large.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 -32px}.tree-default-large .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-large .tree-node.tree-loading{background:none}.tree-default-large>.tree-container-ul .tree-loading>.tree-ocl{background:url(\\\"\\\") 50% no-repeat}.tree-default-large .tree-file{background:url(\\\"\\\") -96px -64px no-repeat}.tree-default-large .tree-folder{background:url(\\\"\\\") -256px 0 no-repeat}.tree-default-large>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-large .tree-ellipsis{overflow:hidden}.tree-default-large .tree-ellipsis .tree-anchor{width:calc(100% - 37px);text-overflow:ellipsis;overflow:hidden}.tree-default-large .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-large.tree-rtl .tree-node{background-image:url(\\\"\\\")}.tree-default-large.tree-rtl .tree-last{background:transparent}\", \"\"]);\n\n// exports\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/css-loader?minimize!./~/vue-loader/lib/style-compiler?{\"vue\":true,\"id\":\"data-v-7fbf202c\",\"scoped\":false,\"hasInlineConfig\":false}!./~/less-loader/dist/cjs.js!./~/vue-loader/lib/selector.js?type=styles&index=0!./src/tree.vue\n// module id = 5\n// module chunks = 0","/*\r\n\tMIT License http://www.opensource.org/licenses/mit-license.php\r\n\tAuthor Tobias Koppers @sokra\r\n*/\r\n// css base code, injected by the css-loader\r\nmodule.exports = function() {\r\n\tvar list = [];\r\n\r\n\t// return the list of modules as css string\r\n\tlist.toString = function toString() {\r\n\t\tvar result = [];\r\n\t\tfor(var i = 0; i < this.length; i++) {\r\n\t\t\tvar item = this[i];\r\n\t\t\tif(item[2]) {\r\n\t\t\t\tresult.push(\"@media \" + item[2] + \"{\" + item[1] + \"}\");\r\n\t\t\t} else {\r\n\t\t\t\tresult.push(item[1]);\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn result.join(\"\");\r\n\t};\r\n\r\n\t// import a list of modules into the list\r\n\tlist.i = function(modules, mediaQuery) {\r\n\t\tif(typeof modules === \"string\")\r\n\t\t\tmodules = [[null, modules, \"\"]];\r\n\t\tvar alreadyImportedModules = {};\r\n\t\tfor(var i = 0; i < this.length; i++) {\r\n\t\t\tvar id = this[i][0];\r\n\t\t\tif(typeof id === \"number\")\r\n\t\t\t\talreadyImportedModules[id] = true;\r\n\t\t}\r\n\t\tfor(i = 0; i < modules.length; i++) {\r\n\t\t\tvar item = modules[i];\r\n\t\t\t// skip already imported module\r\n\t\t\t// this implementation is not 100% perfect for weird media query combinations\r\n\t\t\t// when a module is imported multiple times with different media queries.\r\n\t\t\t// I hope this will never occur (Hey this way we have smaller bundles)\r\n\t\t\tif(typeof item[0] !== \"number\" || !alreadyImportedModules[item[0]]) {\r\n\t\t\t\tif(mediaQuery && !item[2]) {\r\n\t\t\t\t\titem[2] = mediaQuery;\r\n\t\t\t\t} else if(mediaQuery) {\r\n\t\t\t\t\titem[2] = \"(\" + item[2] + \") and (\" + mediaQuery + \")\";\r\n\t\t\t\t}\r\n\t\t\t\tlist.push(item);\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\treturn list;\r\n};\r\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/css-loader/lib/css-base.js\n// module id = 6\n// module chunks = 0","var Component = require(\"!../node_modules/vue-loader/lib/component-normalizer\")(\n /* script */\n require(\"!!babel-loader!../node_modules/vue-loader/lib/selector?type=script&index=0!./tree-item.vue\"),\n /* template */\n require(\"!!../node_modules/vue-loader/lib/template-compiler/index?{\\\"id\\\":\\\"data-v-5c46de44\\\",\\\"hasScoped\\\":false}!../node_modules/vue-loader/lib/selector?type=template&index=0!./tree-item.vue\"),\n /* styles */\n null,\n /* scopeId */\n null,\n /* moduleIdentifier (server only) */\n null\n)\n\nmodule.exports = Component.exports\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./src/tree-item.vue\n// module id = 7\n// module chunks = 0","module.exports={render:function (){var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;\n return _c('li', {\n class: _vm.classes,\n attrs: {\n \"role\": \"treeitem\",\n \"draggable\": _vm.draggable\n },\n on: {\n \"dragstart\": function($event) {\n $event.stopPropagation();\n _vm.onItemDragStart($event, _vm._self, _vm._self.model)\n },\n \"dragend\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.onItemDragEnd($event, _vm._self, _vm._self.model)\n },\n \"dragover\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.isDragEnter = true\n },\n \"dragenter\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.isDragEnter = true\n },\n \"dragleave\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.isDragEnter = false\n },\n \"drop\": function($event) {\n $event.stopPropagation();\n $event.preventDefault();\n _vm.handleItemDrop($event, _vm._self, _vm._self.model)\n }\n }\n }, [(_vm.isWholeRow) ? _c('div', {\n class: _vm.wholeRowClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }, [_vm._v(\" \")]) : _vm._e(), _vm._v(\" \"), _c('i', {\n staticClass: \"tree-icon tree-ocl\",\n attrs: {\n \"role\": \"presentation\"\n },\n on: {\n \"click\": _vm.handleItemToggle\n }\n }), _vm._v(\" \"), _c('div', _vm._g({\n class: _vm.anchorClasses\n }, _vm.events), [(_vm.showCheckbox && !_vm.model.loading) ? _c('i', {\n staticClass: \"tree-icon tree-checkbox\",\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _vm._t(\"default\", [(!_vm.model.loading) ? _c('i', {\n class: _vm.themeIconClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _c('span', {\n domProps: {\n \"innerHTML\": _vm._s(_vm.model[_vm.textFieldName])\n }\n })], {\n vm: this,\n model: _vm.model\n })], 2), _vm._v(\" \"), (_vm.isFolder) ? _c('ul', {\n ref: \"group\",\n staticClass: \"tree-children\",\n style: (_vm.groupStyle),\n attrs: {\n \"role\": \"group\"\n }\n }, _vm._l((_vm.model[_vm.childrenFieldName]), function(child, index) {\n return _c('tree-item', {\n key: index,\n attrs: {\n \"data\": child,\n \"text-field-name\": _vm.textFieldName,\n \"value-field-name\": _vm.valueFieldName,\n \"children-field-name\": _vm.childrenFieldName,\n \"item-events\": _vm.itemEvents,\n \"whole-row\": _vm.wholeRow,\n \"show-checkbox\": _vm.showCheckbox,\n \"allow-transition\": _vm.allowTransition,\n \"height\": _vm.height,\n \"parent-item\": _vm.model[_vm.childrenFieldName],\n \"draggable\": _vm.draggable,\n \"drag-over-background-color\": _vm.dragOverBackgroundColor,\n \"on-item-click\": _vm.onItemClick,\n \"on-item-toggle\": _vm.onItemToggle,\n \"on-item-drag-start\": _vm.onItemDragStart,\n \"on-item-drag-end\": _vm.onItemDragEnd,\n \"on-item-drop\": _vm.onItemDrop,\n \"klass\": index === _vm.model[_vm.childrenFieldName].length - 1 ? 'tree-last' : ''\n },\n scopedSlots: _vm._u([{\n key: \"default\",\n fn: function(_) {\n return [_vm._t(\"default\", [(!_vm.model.loading) ? _c('i', {\n class: _.vm.themeIconClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _c('span', {\n domProps: {\n \"innerHTML\": _vm._s(_.model[_vm.textFieldName])\n }\n })], {\n vm: _.vm,\n model: _.model\n })]\n }\n }])\n })\n })) : _vm._e()])\n},staticRenderFns: []}\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/vue-loader/lib/template-compiler?{\"id\":\"data-v-5c46de44\",\"hasScoped\":false}!./~/vue-loader/lib/selector.js?type=template&index=0!./src/tree-item.vue\n// module id = 8\n// module chunks = 0","module.exports={render:function (){var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;\n return _c('div', {\n class: _vm.classes,\n attrs: {\n \"role\": \"tree\",\n \"onselectstart\": \"return false\"\n }\n }, [_c('ul', {\n class: _vm.containerClasses,\n attrs: {\n \"role\": \"group\"\n }\n }, _vm._l((_vm.data), function(child, index) {\n return _c('tree-item', {\n key: index,\n attrs: {\n \"data\": child,\n \"text-field-name\": _vm.textFieldName,\n \"value-field-name\": _vm.valueFieldName,\n \"children-field-name\": _vm.childrenFieldName,\n \"item-events\": _vm.itemEvents,\n \"whole-row\": _vm.wholeRow,\n \"show-checkbox\": _vm.showCheckbox,\n \"allow-transition\": _vm.allowTransition,\n \"height\": _vm.sizeHeight,\n \"parent-item\": _vm.data,\n \"draggable\": _vm.draggable,\n \"drag-over-background-color\": _vm.dragOverBackgroundColor,\n \"on-item-click\": _vm.onItemClick,\n \"on-item-toggle\": _vm.onItemToggle,\n \"on-item-drag-start\": _vm.onItemDragStart,\n \"on-item-drag-end\": _vm.onItemDragEnd,\n \"on-item-drop\": _vm.onItemDrop,\n \"klass\": index === _vm.data.length - 1 ? 'tree-last' : ''\n },\n scopedSlots: _vm._u([{\n key: \"default\",\n fn: function(_) {\n return [_vm._t(\"default\", [(!_.model.loading) ? _c('i', {\n class: _.vm.themeIconClasses,\n attrs: {\n \"role\": \"presentation\"\n }\n }) : _vm._e(), _vm._v(\" \"), _c('span', {\n domProps: {\n \"innerHTML\": _vm._s(_.model[_vm.textFieldName])\n }\n })], {\n vm: _.vm,\n model: _.model\n })]\n }\n }])\n })\n }))])\n},staticRenderFns: []}\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/vue-loader/lib/template-compiler?{\"id\":\"data-v-7fbf202c\",\"hasScoped\":false}!./~/vue-loader/lib/selector.js?type=template&index=0!./src/tree.vue\n// module id = 9\n// module chunks = 0","// style-loader: Adds some css to the DOM by adding a {% endblock %} {% block content %}
+
+

{% trans 'Import' %}

+ + {% trans 'Bookmark Me!' %} +
+ -
-
-
+ +