diff --git a/cookbook/forms.py b/cookbook/forms.py
index 94f287eeb..239b17012 100644
--- a/cookbook/forms.py
+++ b/cookbook/forms.py
@@ -155,11 +155,13 @@ class ImportExportBase(forms.Form):
RECIPESAGE = 'RECIPESAGE'
DOMESTICA = 'DOMESTICA'
MEALMASTER = 'MEALMASTER'
+ MELARECIPES = 'MELARECIPES'
REZKONV = 'REZKONV'
OPENEATS = 'OPENEATS'
PLANTOEAT = 'PLANTOEAT'
COOKBOOKAPP = 'COOKBOOKAPP'
COPYMETHAT = 'COPYMETHAT'
+ COOKMATE = 'COOKMATE'
PDF = 'PDF'
type = forms.ChoiceField(choices=(
@@ -167,7 +169,8 @@ class ImportExportBase(forms.Form):
(MEALIE, 'Mealie'), (CHOWDOWN, 'Chowdown'), (SAFFRON, 'Saffron'), (CHEFTAP, 'ChefTap'),
(PEPPERPLATE, 'Pepperplate'), (RECETTETEK, 'RecetteTek'), (RECIPESAGE, 'Recipe Sage'), (DOMESTICA, 'Domestica'),
(MEALMASTER, 'MealMaster'), (REZKONV, 'RezKonv'), (OPENEATS, 'Openeats'), (RECIPEKEEPER, 'Recipe Keeper'),
- (PLANTOEAT, 'Plantoeat'), (COOKBOOKAPP, 'CookBookApp'), (COPYMETHAT, 'CopyMeThat'), (PDF, 'PDF'),
+ (PLANTOEAT, 'Plantoeat'), (COOKBOOKAPP, 'CookBookApp'), (COPYMETHAT, 'CopyMeThat'), (PDF, 'PDF'), (MELARECIPES, 'Melarecipes'),
+ (COOKMATE, 'Cookmate')
))
diff --git a/cookbook/helper/ingredient_parser.py b/cookbook/helper/ingredient_parser.py
index b3f4a1c03..4987e0b5b 100644
--- a/cookbook/helper/ingredient_parser.py
+++ b/cookbook/helper/ingredient_parser.py
@@ -203,11 +203,14 @@ class IngredientParser:
def parse(self, x):
# initialize default values
amount = 0
- unit = ''
+ unit = None
ingredient = ''
note = ''
unit_note = ''
+ if len(x) == 0:
+ raise ValueError('string to parse cannot be empty')
+
# if the string contains parenthesis early on remove it and place it at the end
# because its likely some kind of note
if re.match('(.){1,6}\s\((.[^\(\)])+\)\s', x):
@@ -271,4 +274,9 @@ class IngredientParser:
if unit_note not in note:
note += ' ' + unit_note
- return amount, self.apply_unit_automation(unit.strip()), self.apply_food_automation(ingredient.strip()), note.strip()
+ try:
+ unit = self.apply_unit_automation(unit.strip())
+ except Exception:
+ pass
+
+ return amount, unit, self.apply_food_automation(ingredient.strip()), note.strip()
diff --git a/cookbook/helper/permission_helper.py b/cookbook/helper/permission_helper.py
index a24483b24..0331a5993 100644
--- a/cookbook/helper/permission_helper.py
+++ b/cookbook/helper/permission_helper.py
@@ -1,6 +1,3 @@
-"""
-Source: https://djangosnippets.org/snippets/1703/
-"""
from django.conf import settings
from django.contrib import messages
from django.contrib.auth.decorators import user_passes_test
@@ -12,7 +9,7 @@ from django.utils.translation import gettext as _
from rest_framework import permissions
from rest_framework.permissions import SAFE_METHODS
-from cookbook.models import ShareLink
+from cookbook.models import ShareLink, Recipe, UserPreference
def get_allowed_groups(groups_required):
@@ -262,3 +259,38 @@ class CustomIsShare(permissions.BasePermission):
if share:
return share_link_valid(obj, share)
return False
+
+
+def above_space_limit(space): # TODO add file storage limit
+ """
+ Test if the space has reached any limit (e.g. max recipes, users, ..)
+ :param space: Space to test for limits
+ :return: Tuple (True if above or equal any limit else false, message)
+ """
+ r_limit, r_msg = above_space_recipe_limit(space)
+ u_limit, u_msg = above_space_user_limit(space)
+ return r_limit or u_limit, (r_msg + ' ' + u_msg).strip()
+
+
+def above_space_recipe_limit(space):
+ """
+ Test if a space has reached its recipe limit
+ :param space: Space to test for limits
+ :return: Tuple (True if above or equal limit else false, message)
+ """
+ limit = space.max_recipes != 0 and Recipe.objects.filter(space=space).count() >= space.max_recipes
+ if limit:
+ return True, _('You have reached the maximum number of recipes for your space.')
+ return False, ''
+
+
+def above_space_user_limit(space):
+ """
+ Test if a space has reached its user limit
+ :param space: Space to test for limits
+ :return: Tuple (True if above or equal limit else false, message)
+ """
+ limit = space.max_users != 0 and UserPreference.objects.filter(space=space).count() > space.max_users
+ if limit:
+ return True, _('You have more users than allowed in your space.')
+ return False, ''
diff --git a/cookbook/helper/recipe_html_import.py b/cookbook/helper/recipe_html_import.py
index acf72917b..7fa7beaf2 100644
--- a/cookbook/helper/recipe_html_import.py
+++ b/cookbook/helper/recipe_html_import.py
@@ -58,18 +58,6 @@ def get_recipe_from_source(text, url, request):
})
return kid_list
- recipe_json = {
- 'name': '',
- 'url': '',
- 'description': '',
- 'image': '',
- 'keywords': [],
- 'recipeIngredient': [],
- 'recipeInstructions': '',
- 'servings': '',
- 'prepTime': '',
- 'cookTime': ''
- }
recipe_tree = []
parse_list = []
html_data = []
diff --git a/cookbook/helper/recipe_url_import.py b/cookbook/helper/recipe_url_import.py
index b8b3797e5..1d9b9e549 100644
--- a/cookbook/helper/recipe_url_import.py
+++ b/cookbook/helper/recipe_url_import.py
@@ -14,6 +14,9 @@ from cookbook.helper.ingredient_parser import IngredientParser
from cookbook.models import Keyword
+# from recipe_scrapers._utils import get_minutes ## temporary until/unless upstream incorporates get_minutes() PR
+
+
def get_from_scraper(scrape, request):
# converting the scrape_me object to the existing json format based on ld+json
recipe_json = {}
@@ -28,7 +31,7 @@ def get_from_scraper(scrape, request):
recipe_json['name'] = ''
try:
- description = scrape.description() or None
+ description = scrape.description() or None
except Exception:
description = None
if not description:
@@ -37,7 +40,7 @@ def get_from_scraper(scrape, request):
except Exception:
description = ''
- recipe_json['description'] = parse_description(description)
+ recipe_json['internal'] = True
try:
servings = scrape.yields() or None
@@ -48,34 +51,31 @@ def get_from_scraper(scrape, request):
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)
+
+ recipe_json['servings'] = parse_servings(servings)
+ recipe_json['servings_text'] = parse_servings_text(servings)
try:
- recipe_json['prepTime'] = get_minutes(scrape.prep_time()) or 0
+ recipe_json['working_time'] = get_minutes(scrape.prep_time()) or 0
except Exception:
try:
- recipe_json['prepTime'] = get_minutes(scrape.schema.data.get("prepTime")) or 0
+ recipe_json['working_time'] = get_minutes(scrape.schema.data.get("prepTime")) or 0
except Exception:
- recipe_json['prepTime'] = 0
+ recipe_json['working_time'] = 0
try:
- recipe_json['cookTime'] = get_minutes(scrape.cook_time()) or 0
+ recipe_json['waiting_time'] = get_minutes(scrape.cook_time()) or 0
except Exception:
try:
- recipe_json['cookTime'] = get_minutes(scrape.schema.data.get("cookTime")) or 0
+ recipe_json['waiting_time'] = get_minutes(scrape.schema.data.get("cookTime")) or 0
except Exception:
- recipe_json['cookTime'] = 0
+ recipe_json['waiting_time'] = 0
- if recipe_json['cookTime'] + recipe_json['prepTime'] == 0:
+ if recipe_json['working_time'] + recipe_json['waiting_time'] == 0:
try:
- recipe_json['prepTime'] = get_minutes(scrape.total_time()) or 0
+ recipe_json['working_time'] = get_minutes(scrape.total_time()) or 0
except Exception:
try:
- recipe_json['prepTime'] = get_minutes(scrape.schema.data.get("totalTime")) or 0
+ recipe_json['working_time'] = get_minutes(scrape.schema.data.get("totalTime")) or 0
except Exception:
pass
@@ -113,6 +113,14 @@ def get_from_scraper(scrape, request):
keywords += listify_keywords(scrape.schema.data.get("recipeCuisine"))
except Exception:
pass
+
+ if source_url := scrape.canonical_url():
+ recipe_json['source_url'] = source_url
+ try:
+ keywords.append(source_url.replace('http://', '').replace('https://', '').split('/')[0])
+ except Exception:
+ pass
+
try:
recipe_json['keywords'] = parse_keywords(list(set(map(str.casefold, keywords))), request.space)
except AttributeError:
@@ -120,54 +128,49 @@ def get_from_scraper(scrape, request):
ingredient_parser = IngredientParser(request, True)
- ingredients = []
+ recipe_json['steps'] = []
+
+ for i in parse_instructions(scrape.instructions()):
+ recipe_json['steps'].append({'instruction': i, 'ingredients': [], })
+ if len(recipe_json['steps']) == 0:
+ recipe_json['steps'].append({'instruction': '', 'ingredients': [], })
+
+ if len(parse_description(description)) > 256: # split at 256 as long descriptions dont look good on recipe cards
+ recipe_json['steps'][0]['instruction'] = f'*{parse_description(description)}* \n\n' + recipe_json['steps'][0]['instruction']
+ else:
+ recipe_json['description'] = parse_description(description)[:512]
+
try:
for x in scrape.ingredients():
try:
amount, unit, ingredient, note = ingredient_parser.parse(x)
- ingredients.append(
- {
- 'amount': amount,
- 'unit': {
- 'text': unit,
- 'id': random.randrange(10000, 99999)
- },
- 'ingredient': {
- 'text': ingredient,
- 'id': random.randrange(10000, 99999)
- },
- 'note': note,
- 'original_text': x
- }
- )
+ ingredient = {
+ 'amount': amount,
+ 'food': {
+ 'name': ingredient,
+ },
+ 'unit': None,
+ 'note': note,
+ 'original_text': x
+ }
+ if unit:
+ ingredient['unit'] = {'name': unit, }
+ recipe_json['steps'][0]['ingredients'].append(ingredient)
except Exception:
- ingredients.append(
+ recipe_json['steps'][0]['ingredients'].append(
{
'amount': 0,
- 'unit': {
- 'text': '',
- 'id': random.randrange(10000, 99999)
- },
- 'ingredient': {
- 'text': x,
- 'id': random.randrange(10000, 99999)
+ 'unit': None,
+ 'food': {
+ 'name': x,
},
'note': '',
'original_text': x
}
)
- recipe_json['recipeIngredient'] = ingredients
except Exception:
- recipe_json['recipeIngredient'] = ingredients
+ pass
- try:
- recipe_json['recipeInstructions'] = parse_instructions(scrape.instructions())
- except Exception:
- recipe_json['recipeInstructions'] = ""
-
- if scrape.canonical_url():
- recipe_json['url'] = scrape.canonical_url()
- recipe_json['recipeInstructions'] += "\n\n" + _("Imported from") + ": " + scrape.canonical_url()
return recipe_json
@@ -180,102 +183,46 @@ def parse_name(name):
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_text': 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_text': 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
-
- normalized_string = normalize_string(instructions)
+def clean_instruction_string(instruction):
+ normalized_string = normalize_string(instruction)
normalized_string = normalized_string.replace('\n', ' \n')
normalized_string = normalized_string.replace(' \n \n', '\n\n')
return normalized_string
+def parse_instructions(instructions):
+ """
+ Convert arbitrary instructions object from website import and turn it into a flat list of strings
+ :param instructions: any instructions object from import
+ :return: list of strings (from one to many elements depending on website)
+ """
+ instruction_list = []
+
+ if type(instructions) == list:
+ for i in instructions:
+ if type(i) == str:
+ instruction_list.append(clean_instruction_string(i))
+ else:
+ if 'text' in i:
+ instruction_list.append(clean_instruction_string(i['text']))
+ elif 'itemListElement' in i:
+ for ile in i['itemListElement']:
+ if type(ile) == str:
+ instruction_list.append(clean_instruction_string(ile))
+ elif 'text' in ile:
+ instruction_list.append(clean_instruction_string(ile['text']))
+ else:
+ instruction_list.append(clean_instruction_string(str(i)))
+ else:
+ instruction_list.append(clean_instruction_string(instructions))
+
+ return instruction_list
+
+
def parse_image(image):
# check if list of images is returned, take first if so
if not image:
@@ -310,40 +257,31 @@ def parse_servings(servings):
return servings
-def parse_cooktime(cooktime):
- if type(cooktime) not in [int, float]:
+def parse_servings_text(servings):
+ if type(servings) == str:
try:
- cooktime = float(re.search(r'\d+', cooktime).group())
+ servings = re.sub("\d+", '', servings).strip()
+ except Exception:
+ servings = ''
+ return servings
+
+
+def parse_time(recipe_time):
+ if type(recipe_time) not in [int, float]:
+ try:
+ recipe_time = float(re.search(r'\d+', recipe_time).group())
except (ValueError, AttributeError):
try:
- cooktime = round(iso_parse_duration(cooktime).seconds / 60)
+ recipe_time = round(iso_parse_duration(recipe_time).seconds / 60)
except ISO8601Error:
try:
- if (type(cooktime) == list and len(cooktime) > 0):
- cooktime = cooktime[0]
- cooktime = round(parse_duration(cooktime).seconds / 60)
+ if (type(recipe_time) == list and len(recipe_time) > 0):
+ recipe_time = recipe_time[0]
+ recipe_time = round(parse_duration(recipe_time).seconds / 60)
except AttributeError:
- cooktime = 0
+ recipe_time = 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
+ return recipe_time
def parse_keywords(keyword_json, space):
@@ -353,9 +291,9 @@ def parse_keywords(keyword_json, space):
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.name)})
+ keywords.append({'label': str(k), 'name': k.name, 'id': k.id})
else:
- keywords.append({'id': random.randrange(1111111, 9999999, 1), 'text': kw})
+ keywords.append({'label': kw, 'name': kw})
return keywords
diff --git a/cookbook/integration/cookmate.py b/cookbook/integration/cookmate.py
new file mode 100644
index 000000000..19c0e936e
--- /dev/null
+++ b/cookbook/integration/cookmate.py
@@ -0,0 +1,77 @@
+import base64
+import json
+from io import BytesIO
+
+from gettext import gettext as _
+
+import requests
+from lxml import etree
+
+from cookbook.helper.ingredient_parser import IngredientParser
+from cookbook.helper.recipe_url_import import parse_servings, parse_time, parse_servings_text
+from cookbook.integration.integration import Integration
+from cookbook.models import Ingredient, Keyword, Recipe, Step
+
+
+class Cookmate(Integration):
+
+ def import_file_name_filter(self, zip_info_object):
+ return zip_info_object.filename.endswith('.xml')
+
+ def get_files_from_recipes(self, recipes, el, cookie):
+ raise NotImplementedError('Method not implemented in storage integration')
+
+ def get_recipe_from_file(self, file):
+ recipe_xml = file
+
+ recipe = Recipe.objects.create(
+ name=recipe_xml.find('title').text.strip(),
+ created_by=self.request.user, internal=True, space=self.request.space)
+
+ if recipe_xml.find('preptime') is not None:
+ recipe.working_time = parse_time(recipe_xml.find('preptime').text.strip())
+
+ if recipe_xml.find('cooktime') is not None:
+ recipe.waiting_time = parse_time(recipe_xml.find('cooktime').text.strip())
+
+ if recipe_xml.find('quantity') is not None:
+ recipe.servings = parse_servings(recipe_xml.find('quantity').text.strip())
+ recipe.servings_text = parse_servings_text(recipe_xml.find('quantity').text.strip())
+
+ if recipe_xml.find('url') is not None:
+ recipe.source_url = recipe_xml.find('url').text.strip()
+
+ if recipe_xml.find('description') is not None: # description is a list of
's with text
+ if len(recipe_xml.find('description')) > 0:
+ recipe.description = recipe_xml.find('description')[0].text[:512]
+
+ for step in recipe_xml.find('recipetext').getchildren():
+ step = Step.objects.create(
+ instruction=step.text.strip(), space=self.request.space,
+ )
+ recipe.steps.add(step)
+
+ ingredient_parser = IngredientParser(self.request, True)
+
+ for ingredient in recipe_xml.find('ingredient').getchildren():
+ if ingredient.text.strip() != '':
+ amount, unit, food, note = ingredient_parser.parse(ingredient.text.strip())
+ f = ingredient_parser.get_food(food)
+ u = ingredient_parser.get_unit(unit)
+ recipe.steps.first().ingredients.add(Ingredient.objects.create(
+ food=f, unit=u, amount=amount, note=note, original_text=ingredient.text.strip(), space=self.request.space,
+ ))
+
+ if recipe_xml.find('imageurl') is not None:
+ try:
+ response = requests.get(recipe_xml.find('imageurl').text.strip())
+ self.import_recipe_image(recipe, BytesIO(response.content))
+ except Exception as e:
+ print('failed to import image ', str(e))
+
+ recipe.save()
+
+ return recipe
+
+ def get_file_from_recipe(self, recipe):
+ raise NotImplementedError('Method not implemented in storage integration')
diff --git a/cookbook/integration/copymethat.py b/cookbook/integration/copymethat.py
index c961a51a5..7a2a532f9 100644
--- a/cookbook/integration/copymethat.py
+++ b/cookbook/integration/copymethat.py
@@ -32,7 +32,14 @@ class CopyMeThat(Integration):
recipe.servings = parse_servings(file.find("a", {"id": "recipeYield"}).text.strip())
recipe.working_time = iso_duration_to_minutes(file.find("span", {"meta": "prepTime"}).text.strip())
recipe.waiting_time = iso_duration_to_minutes(file.find("span", {"meta": "cookTime"}).text.strip())
- recipe.save()
+ recipe.description = (file.find("div ", {"id": "description"}).text.strip())[:512]
+
+ except AttributeError:
+ pass
+
+ try:
+ if len(file.find("span", {"id": "starred"}).text.strip()) > 0:
+ recipe.keywords.add(Keyword.objects.get_or_create(space=self.request.space, name=_('Favorite'))[0])
except AttributeError:
pass
diff --git a/cookbook/integration/integration.py b/cookbook/integration/integration.py
index d1d82c57c..6493c266d 100644
--- a/cookbook/integration/integration.py
+++ b/cookbook/integration/integration.py
@@ -5,6 +5,8 @@ import traceback
import uuid
from io import BytesIO, StringIO
from zipfile import BadZipFile, ZipFile
+
+import lxml
from django.core.cache import cache
import datetime
@@ -16,6 +18,7 @@ from django.http import HttpResponse
from django.utils.formats import date_format
from django.utils.translation import gettext as _
from django_scopes import scope
+from lxml import etree
from cookbook.forms import ImportExportBase
from cookbook.helper.image_processing import get_filetype, handle_image
@@ -144,7 +147,7 @@ class Integration:
il.imported_recipes += 1
il.save()
import_zip.close()
- elif '.zip' in f['name'] or '.paprikarecipes' in f['name']:
+ elif '.zip' in f['name'] or '.paprikarecipes' in f['name'] or '.mcb' in f['name']:
import_zip = ZipFile(f['file'])
file_list = []
for z in import_zip.filelist:
@@ -157,9 +160,16 @@ class Integration:
file_list = self.split_recipe_file(BytesIO(import_zip.read('recipes.html')))
il.total_recipes += len(file_list)
+ if isinstance(self, cookbook.integration.cookmate.Cookmate):
+ new_file_list = []
+ for file in file_list:
+ new_file_list += etree.parse(BytesIO(import_zip.read(file.filename))).getroot().getchildren()
+ il.total_recipes = len(new_file_list)
+ file_list = new_file_list
+
for z in file_list:
try:
- if isinstance(z, Tag):
+ if not hasattr(z, 'filename'):
recipe = self.get_recipe_from_file(z)
else:
recipe = self.get_recipe_from_file(BytesIO(import_zip.read(z.filename)))
@@ -172,7 +182,7 @@ class Integration:
traceback.print_exc()
self.handle_exception(e, log=il, message=f'-------------------- \nERROR \n{e}\n--------------------\n')
import_zip.close()
- elif '.json' in f['name'] or '.txt' in f['name'] or '.mmf' in f['name'] or '.rk' in f['name']:
+ elif '.json' in f['name'] or '.txt' in f['name'] or '.mmf' in f['name'] or '.rk' in f['name'] or '.melarecipe' in f['name']:
data_list = self.split_recipe_file(f['file'])
il.total_recipes += len(data_list)
for d in data_list:
diff --git a/cookbook/integration/melarecipes.py b/cookbook/integration/melarecipes.py
new file mode 100644
index 000000000..88ff4d355
--- /dev/null
+++ b/cookbook/integration/melarecipes.py
@@ -0,0 +1,83 @@
+import base64
+import json
+from io import BytesIO
+
+from gettext import gettext as _
+from cookbook.helper.ingredient_parser import IngredientParser
+from cookbook.helper.recipe_url_import import parse_servings, parse_time
+from cookbook.integration.integration import Integration
+from cookbook.models import Ingredient, Keyword, Recipe, Step
+
+
+class MelaRecipes(Integration):
+
+ def split_recipe_file(self, file):
+ return [json.loads(file.getvalue().decode("utf-8"))]
+
+ def get_files_from_recipes(self, recipes, el, cookie):
+ raise NotImplementedError('Method not implemented in storage integration')
+
+ def get_recipe_from_file(self, file):
+ recipe_json = file
+
+ recipe = Recipe.objects.create(
+ name=recipe_json['title'].strip(),
+ created_by=self.request.user, internal=True, space=self.request.space)
+
+ if 'yield' in recipe_json:
+ recipe.servings = parse_servings(recipe_json['yield'])
+
+ if 'cookTime' in recipe_json:
+ recipe.waiting_time = parse_time(recipe_json['cookTime'])
+
+ if 'prepTime' in recipe_json:
+ recipe.working_time = parse_time(recipe_json['prepTime'])
+
+ if 'favorite' in recipe_json and recipe_json['favorite']:
+ recipe.keywords.add(Keyword.objects.get_or_create(space=self.request.space, name=_('Favorite'))[0])
+
+ if 'categories' in recipe_json:
+ try:
+ for x in recipe_json['categories']:
+ recipe.keywords.add(Keyword.objects.get_or_create(space=self.request.space, name=x)[0])
+ except Exception:
+ pass
+
+ instruction = ''
+ if 'text' in recipe_json:
+ instruction += f'*{recipe_json["text"].strip()}* \n'
+
+ if 'instructions' in recipe_json:
+ instruction += recipe_json["instructions"].strip() + ' \n'
+
+ if 'notes' in recipe_json:
+ instruction += recipe_json["notes"].strip() + ' \n'
+
+ if 'link' in recipe_json:
+ recipe.source_url = recipe_json['link']
+
+ step = Step.objects.create(
+ instruction=instruction, space=self.request.space,
+ )
+
+ ingredient_parser = IngredientParser(self.request, True)
+ for ingredient in recipe_json['ingredients'].split('\n'):
+ if ingredient.strip() != '':
+ amount, unit, food, note = ingredient_parser.parse(ingredient)
+ f = ingredient_parser.get_food(food)
+ u = ingredient_parser.get_unit(unit)
+ step.ingredients.add(Ingredient.objects.create(
+ food=f, unit=u, amount=amount, note=note, original_text=ingredient, space=self.request.space,
+ ))
+ recipe.steps.add(step)
+
+ if recipe_json.get("images", None):
+ try:
+ self.import_recipe_image(recipe, BytesIO(base64.b64decode(recipe_json['images'][0])), filetype='.jpeg')
+ except Exception:
+ pass
+
+ return recipe
+
+ def get_file_from_recipe(self, recipe):
+ raise NotImplementedError('Method not implemented in storage integration')
diff --git a/cookbook/integration/nextcloud_cookbook.py b/cookbook/integration/nextcloud_cookbook.py
index d4720222a..c302aff50 100644
--- a/cookbook/integration/nextcloud_cookbook.py
+++ b/cookbook/integration/nextcloud_cookbook.py
@@ -31,6 +31,9 @@ class NextcloudCookbook(Integration):
except Exception:
pass
+ if 'url' in recipe_json:
+ recipe.source_url = recipe_json['url'].strip()
+
if 'recipeCategory' in recipe_json:
try:
recipe.keywords.add(Keyword.objects.get_or_create(space=self.request.space, name=recipe_json['recipeCategory'])[0])
diff --git a/cookbook/integration/paprika.py b/cookbook/integration/paprika.py
index 7a1255000..9a82c4aa7 100644
--- a/cookbook/integration/paprika.py
+++ b/cookbook/integration/paprika.py
@@ -6,6 +6,7 @@ from gettext import gettext as _
from io import BytesIO
from cookbook.helper.ingredient_parser import IngredientParser
+from cookbook.helper.recipe_url_import import parse_servings, parse_servings_text
from cookbook.integration.integration import Integration
from cookbook.models import Ingredient, Keyword, Recipe, Step
@@ -26,10 +27,9 @@ class Paprika(Integration):
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']):
- s = recipe_json['servings'].split(' ')
- recipe.servings = s[0]
- recipe.servings_text = s[1]
+ if 'servings' in recipe_json['servings']:
+ recipe.servings = parse_servings(recipe_json['servings'])
+ recipe.servings_text = parse_servings_text(recipe_json['servings'])
if len(recipe_json['cook_time'].strip()) > 0:
recipe.waiting_time = re.findall(r'\d+', recipe_json['cook_time'])[0]
diff --git a/cookbook/migrations/0173_recipe_source_url.py b/cookbook/migrations/0173_recipe_source_url.py
new file mode 100644
index 000000000..dc84e64eb
--- /dev/null
+++ b/cookbook/migrations/0173_recipe_source_url.py
@@ -0,0 +1,18 @@
+# Generated by Django 3.2.12 on 2022-03-04 13:39
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+
+ dependencies = [
+ ('cookbook', '0172_ingredient_original_text'),
+ ]
+
+ operations = [
+ migrations.AddField(
+ model_name='recipe',
+ name='source_url',
+ field=models.CharField(blank=True, default=None, max_length=1024, null=True),
+ ),
+ ]
diff --git a/cookbook/models.py b/cookbook/models.py
index bd3d8bd4d..2ed79cc51 100644
--- a/cookbook/models.py
+++ b/cookbook/models.py
@@ -241,7 +241,7 @@ class Space(ExportModelOperationsMixin('space'), models.Model):
max_users = models.IntegerField(default=0)
allow_sharing = models.BooleanField(default=True)
demo = models.BooleanField(default=False)
- food_inherit = models.ManyToManyField(FoodInheritField, blank=True)
+ food_inherit = models.ManyToManyField(FoodInheritField, blank=True)
show_facet_count = models.BooleanField(default=False)
def __str__(self):
@@ -337,7 +337,7 @@ class UserPreference(models.Model, PermissionModelMixin):
default_delay = models.DecimalField(default=4, max_digits=8, decimal_places=4)
shopping_recent_days = models.PositiveIntegerField(default=7)
csv_delim = models.CharField(max_length=2, default=",")
- csv_prefix = models.CharField(max_length=10, blank=True,)
+ csv_prefix = models.CharField(max_length=10, blank=True, )
created_at = models.DateTimeField(auto_now_add=True)
space = models.ForeignKey(Space, on_delete=models.CASCADE, null=True)
@@ -496,11 +496,11 @@ class Food(ExportModelOperationsMixin('food'), TreeModel, PermissionModelMixin):
ignore_shopping = models.BooleanField(default=False) # inherited field
onhand_users = models.ManyToManyField(User, blank=True)
description = models.TextField(default='', blank=True)
- inherit_fields = models.ManyToManyField(FoodInheritField, blank=True)
+ inherit_fields = models.ManyToManyField(FoodInheritField, blank=True)
substitute = models.ManyToManyField("self", blank=True)
substitute_siblings = models.BooleanField(default=False)
substitute_children = models.BooleanField(default=False)
- child_inherit_fields = models.ManyToManyField(FoodInheritField, blank=True, related_name='child_inherit')
+ child_inherit_fields = models.ManyToManyField(FoodInheritField, blank=True, related_name='child_inherit')
space = models.ForeignKey(Space, on_delete=models.CASCADE)
objects = ScopedManager(space='space', _manager_class=TreeManager)
@@ -533,7 +533,7 @@ class Food(ExportModelOperationsMixin('food'), TreeModel, PermissionModelMixin):
if food:
# if child inherit fields is preset children should be set to that, otherwise inherit this foods inherited fields
inherit = list((food.child_inherit_fields.all() or food.inherit_fields.all()).values('id', 'field'))
- tree_filter = Q(path__startswith=food.path, space=space, depth=food.depth+1)
+ tree_filter = Q(path__startswith=food.path, space=space, depth=food.depth + 1)
else:
inherit = list(space.food_inherit.all().values('id', 'field'))
tree_filter = Q(space=space)
@@ -593,6 +593,8 @@ class Ingredient(ExportModelOperationsMixin('ingredient'), models.Model, Permiss
order = models.IntegerField(default=0)
original_text = models.CharField(max_length=512, null=True, blank=True, default=None)
+ original_text = models.CharField(max_length=512, null=True, blank=True, default=None)
+
space = models.ForeignKey(Space, on_delete=models.CASCADE)
objects = ScopedManager(space='space')
@@ -663,9 +665,7 @@ class Recipe(ExportModelOperationsMixin('recipe'), models.Model, PermissionModel
servings = models.IntegerField(default=1)
servings_text = models.CharField(default='', blank=True, max_length=32)
image = models.ImageField(upload_to='recipes/', blank=True, null=True)
- storage = models.ForeignKey(
- Storage, on_delete=models.PROTECT, blank=True, null=True
- )
+ storage = models.ForeignKey(Storage, on_delete=models.PROTECT, blank=True, null=True)
file_uid = models.CharField(max_length=256, default="", blank=True)
file_path = models.CharField(max_length=512, default="", blank=True)
link = models.CharField(max_length=512, null=True, blank=True)
@@ -675,9 +675,9 @@ class Recipe(ExportModelOperationsMixin('recipe'), models.Model, PermissionModel
working_time = models.IntegerField(default=0)
waiting_time = models.IntegerField(default=0)
internal = models.BooleanField(default=False)
- nutrition = models.ForeignKey(
- NutritionInformation, blank=True, null=True, on_delete=models.CASCADE
- )
+ nutrition = models.ForeignKey(NutritionInformation, blank=True, null=True, on_delete=models.CASCADE)
+
+ source_url = models.CharField(max_length=1024, default=None, blank=True, null=True)
created_by = models.ForeignKey(User, on_delete=models.PROTECT)
created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)
diff --git a/cookbook/serializer.py b/cookbook/serializer.py
index e70a7e50a..d913be516 100644
--- a/cookbook/serializer.py
+++ b/cookbook/serializer.py
@@ -12,6 +12,7 @@ from rest_framework.exceptions import NotFound, ValidationError
from cookbook.helper.CustomStorageClass import CachedS3Boto3Storage
from cookbook.helper.HelperFunctions import str2bool
+from cookbook.helper.permission_helper import above_space_limit
from cookbook.helper.shopping_helper import RecipeShoppingEditor
from cookbook.models import (Automation, BookmarkletImport, Comment, CookLog, CustomFilter,
ExportLog, Food, FoodInheritField, ImportLog, Ingredient, Keyword,
@@ -649,6 +650,12 @@ class RecipeSerializer(RecipeBaseSerializer):
)
read_only_fields = ['image', 'created_by', 'created_at']
+ def validate(self, data):
+ above_limit, msg = above_space_limit(self.context['request'].space)
+ if above_limit:
+ raise serializers.ValidationError(msg)
+ return super().validate(data)
+
def create(self, validated_data):
validated_data['created_by'] = self.context['request'].user
validated_data['space'] = self.context['request'].space
@@ -656,9 +663,12 @@ class RecipeSerializer(RecipeBaseSerializer):
class RecipeImageSerializer(WritableNestedModelSerializer):
+ image = serializers.ImageField(required=False, allow_null=True)
+ image_url = serializers.CharField(max_length=4096, required=False, allow_null=True)
+
class Meta:
model = Recipe
- fields = ['image', ]
+ fields = ['image', 'image_url', ]
class RecipeImportSerializer(SpacedModelSerializer):
@@ -968,12 +978,19 @@ class AutomationSerializer(serializers.ModelSerializer):
# CORS, REST and Scopes aren't currently working
# Scopes are evaluating before REST has authenticated the user assigning 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):
+class BookmarkletImportListSerializer(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', 'created_by', 'created_at')
+ read_only_fields = ('created_by', 'space')
+
+
+class BookmarkletImportSerializer(BookmarkletImportListSerializer):
class Meta:
model = BookmarkletImport
fields = ('id', 'url', 'html', 'created_by', 'created_at')
diff --git a/cookbook/static/js/vue-jstree.js b/cookbook/static/js/vue-jstree.js
deleted file mode 100644
index 50d5ade52..000000000
--- a/cookbook/static/js/vue-jstree.js
+++ /dev/null
@@ -1,2 +0,0 @@
-!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 %}
+{% block content_fluid %}
-
-
- {% trans 'URL' %}
- {% trans 'App' %}
- {% trans 'Source' %}
- {% trans 'Text' %}
- {% trans 'File' %}
-
-
-
-
-
-
-
- Tandoor
-
- Cheftap
- Chowdown
- CookBookApp
- CopyMeThat
- Domestica
- Mealie
- Mealmaster
- Nextcloud Cookbook
- Openeats
- Paprika
- Pepperplate
- Plantoeat
- RecetteTek
- Recipekeeper
- Recipesage
- Rezkonv
- Safron
-
-
-
- {% trans 'To prevent duplicates recipes with the same name as existing ones are ignored. Check this box to import everything.' %}
-
-
-
-
- {% trans 'Import' %}
-
-
-
-
-
-
-
- Automatic
-
-
-
- Manual
-
-
-
-
-
-
{% trans 'Import' %}
-
-
-
-
-
-
-
-
- {% if not request.user.is_authenticated or request.user.userpreference.theme == request.user.userpreference.TANDOOR %}
-
- {% else %}
-
- {% endif %}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
[[recipe_json.description]]
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
[[recipe_json.servings]]
-
-
-
-
-
-
-
-
[[recipe_json.prepTime]]
-
-
-
-
-
-
-
-
[[recipe_json.cookTime]]
-
-
-
-
-
-
-
-
-
-
-
-
- [[i.amount]]
- [[i.unit.text]]
- [[i.ingredient.text]]
- [[i.note]]
-
-
-
-
-
-
-
-
-
-
-
-
[[recipe_json.recipeInstructions]]
-
-
-
-
-
-
-
-
{% trans 'Import' %}
-
-
-
-
-
-
-
-
-
- json
-
-
- html
-
-
- images
-
-
-
-
-
-
-
- [[blank_field]]
-
-
-
-
-
-
-
-
-
-
-
-
- {% verbatim %}
- [[_.model.name]]
- {% endverbatim %}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- [[error.msg]]
-
-
-
-
-
+
-
-
{% endblock %}
+
+
+{% block script %}
+ {% if debug %}
+
+ {% else %}
+
+ {% endif %}
+
+
+
+ {% render_bundle 'import_view' %}
+{% endblock %}
\ No newline at end of file
diff --git a/cookbook/templatetags/custom_tags.py b/cookbook/templatetags/custom_tags.py
index 3b119cf3f..f34eac8f4 100644
--- a/cookbook/templatetags/custom_tags.py
+++ b/cookbook/templatetags/custom_tags.py
@@ -136,7 +136,7 @@ def bookmarklet(request):
if (api_token := Token.objects.filter(user=request.user).first()) is None:
api_token = Token.objects.create(user=request.user)
- bookmark = "javascript: \
+ bookmark = "Test "
+ return re.sub(r"[\n\t]*", "", bookmark)
@register.simple_tag
diff --git a/cookbook/views/api.py b/cookbook/views/api.py
index 3719c1e69..1eb3d55e3 100644
--- a/cookbook/views/api.py
+++ b/cookbook/views/api.py
@@ -5,6 +5,7 @@ import uuid
from collections import OrderedDict
import requests
+from PIL import UnidentifiedImageError
from annoying.decorators import ajax_request
from annoying.functions import get_object_or_None
from django.contrib import messages
@@ -23,6 +24,7 @@ from django.utils.translation import gettext as _
from django_scopes import scopes_disabled
from icalendar import Calendar, Event
from recipe_scrapers import NoSchemaFoundInWildMode, WebsiteNotImplementedError, scrape_me
+from requests.exceptions import MissingSchema
from rest_framework import decorators, status, viewsets
from rest_framework.exceptions import APIException, PermissionDenied
from rest_framework.pagination import PageNumberPagination
@@ -68,7 +70,7 @@ from cookbook.serializer import (AutomationSerializer, BookmarkletImportSerializ
SupermarketCategorySerializer, SupermarketSerializer,
SyncLogSerializer, SyncSerializer, UnitSerializer,
UserFileSerializer, UserNameSerializer, UserPreferenceSerializer,
- ViewLogSerializer, IngredientSimpleSerializer)
+ ViewLogSerializer, IngredientSimpleSerializer, BookmarkletImportListSerializer)
from recipes import settings
@@ -767,20 +769,33 @@ class RecipeViewSet(viewsets.ModelViewSet):
serializer = self.serializer_class(obj, data=request.data, partial=True)
- if self.request.space.demo:
- raise PermissionDenied(detail='Not available in demo', code=None)
-
if serializer.is_valid():
serializer.save()
+ image = None
- if serializer.validated_data == {}:
- obj.image = None
- else:
- img, filetype = handle_image(request, obj.image)
+ if 'image' in serializer.validated_data:
+ image = obj.image
+ elif 'image_url' in serializer.validated_data:
+ try:
+ response = requests.get(serializer.validated_data['image_url'])
+ image = File(io.BytesIO(response.content))
+ print('test')
+ except UnidentifiedImageError as e:
+ print(e)
+ pass
+ except MissingSchema as e:
+ print(e)
+ pass
+ except Exception as e:
+ print(e)
+ pass
+
+ if image is not None:
+ img, filetype = handle_image(request, image)
obj.image = File(img, name=f'{uuid.uuid4()}_{obj.pk}{filetype}')
- obj.save()
+ obj.save()
+ return Response(serializer.data)
- return Response(serializer.data)
return Response(serializer.errors, 400)
# TODO: refactor API to use post/put/delete or leave as put and change VUE to use list_recipe after creating
@@ -959,6 +974,11 @@ class BookmarkletImportViewSet(viewsets.ModelViewSet):
serializer_class = BookmarkletImportSerializer
permission_classes = [CustomIsUser]
+ def get_serializer_class(self):
+ if self.action == 'list':
+ return BookmarkletImportListSerializer
+ return self.serializer_class
+
def get_queryset(self):
return self.queryset.filter(space=self.request.space).all()
@@ -1132,102 +1152,55 @@ def get_plan_ical(request, from_date, to_date):
@group_required('user')
def recipe_from_source(request):
- url = request.POST.get('url', None)
- data = request.POST.get('data', None)
- mode = request.POST.get('mode', None)
- auto = request.POST.get('auto', 'true')
+ """
+ function to retrieve a recipe from a given url or source string
+ :param request: standard request with additional post parameters
+ - url: url to use for importing recipe
+ - data: if no url is given recipe is imported from provided source data
+ - (optional) bookmarklet: id of bookmarklet import to use, overrides URL and data attributes
+ :return: JsonResponse containing the parsed json, original html,json and images
+ """
+ request_payload = json.loads(request.body.decode('utf-8'))
+ url = request_payload.get('url', None)
+ data = request_payload.get('data', None)
+ bookmarklet = request_payload.get('bookmarklet', None)
- HEADERS = {
- "User-Agent": "Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.9.0.7) Gecko/2009021910 Firefox/3.0.7"
- }
+ if bookmarklet := BookmarkletImport.objects.filter(pk=bookmarklet).first():
+ url = bookmarklet.url
+ data = bookmarklet.html
+ bookmarklet.delete()
- if (not url and not data) or (mode == 'url' and not url) or (mode == 'source' and not data):
- return JsonResponse(
- {
- 'error': True,
- 'msg': _('Nothing to do.')
- },
- status=400
- )
+ # headers to use for request to external sites
+ external_request_headers = {"User-Agent": "Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.9.0.7) Gecko/2009021910 Firefox/3.0.7"}
- if mode == 'url' and auto == 'true':
+ if not url and not data:
+ return JsonResponse({
+ 'error': True,
+ 'msg': _('Nothing to do.')
+ }, status=400)
+
+ # in manual mode request complete page to return it later
+ if url:
try:
- scrape = scrape_me(url)
- except (WebsiteNotImplementedError, AttributeError):
- try:
- scrape = scrape_me(url, wild_mode=True)
- except NoSchemaFoundInWildMode:
- return JsonResponse(
- {
- 'error': True,
- 'msg': _('The requested site provided malformed data and cannot be read.') # noqa: E501
- },
- status=400)
- except ConnectionError:
- return JsonResponse(
- {
- 'error': True,
- 'msg': _('The requested page could not be found.')
- },
- status=400
- )
-
- try:
- instructions = scrape.instructions()
- except Exception:
- instructions = ""
- try:
- ingredients = scrape.ingredients()
- except Exception:
- ingredients = []
- if len(ingredients) + len(instructions) == 0:
- return JsonResponse(
- {
- 'error': True,
- 'msg': _(
- 'The requested site does not provide any recognized data format to import the recipe from.')
- # noqa: E501
- },
- status=400)
- else:
- return JsonResponse({"recipe_json": get_from_scraper(scrape, request)})
- elif (mode == 'source') or (mode == 'url' and auto == 'false'):
- if not data or data == 'undefined':
- try:
- data = requests.get(url, headers=HEADERS).content
- except requests.exceptions.ConnectionError:
- return JsonResponse(
- {
- 'error': True,
- 'msg': _('Connection Refused.')
- },
- status=400
- )
- recipe_json, recipe_tree, recipe_html, images = get_recipe_from_source(data, url, request)
- if len(recipe_tree) == 0 and len(recipe_json) == 0:
- return JsonResponse(
- {
- 'error': True,
- 'msg': _('No usable data could be found.')
- },
- status=400
- )
- else:
+ data = requests.get(url, headers=external_request_headers).content
+ except requests.exceptions.ConnectionError:
return JsonResponse({
- 'recipe_tree': recipe_tree,
- 'recipe_json': recipe_json,
- 'recipe_html': recipe_html,
- 'images': images,
- })
-
- else:
- return JsonResponse(
- {
'error': True,
- 'msg': _('I couldn\'t find anything to do.')
- },
- status=400
- )
+ 'msg': _('Connection Refused.')
+ }, status=400)
+ recipe_json, recipe_tree, recipe_html, recipe_images = get_recipe_from_source(data, url, request)
+ if len(recipe_tree) == 0 and len(recipe_json) == 0:
+ return JsonResponse({
+ 'error': True,
+ 'msg': _('No usable data could be found.')
+ }, status=400)
+ else:
+ return JsonResponse({
+ 'recipe_json': recipe_json,
+ 'recipe_tree': recipe_tree,
+ 'recipe_html': recipe_html,
+ 'recipe_images': list(dict.fromkeys(recipe_images)),
+ })
@group_required('admin')
diff --git a/cookbook/views/data.py b/cookbook/views/data.py
index 6362f3efd..8dd1e969f 100644
--- a/cookbook/views/data.py
+++ b/cookbook/views/data.py
@@ -1,41 +1,27 @@
-import json
-import uuid
from datetime import datetime
-from io import BytesIO
-import requests
from django.contrib import messages
-from django.core.exceptions import ObjectDoesNotExist
-from django.core.files import File
-from django.db.transaction import atomic
-from django.http import HttpResponse, HttpResponseRedirect
+from django.http import HttpResponseRedirect
from django.shortcuts import redirect, render
from django.urls import reverse
from django.utils.translation import gettext as _
from django.utils.translation import ngettext
from django_tables2 import RequestConfig
-from PIL import UnidentifiedImageError
-from requests.exceptions import MissingSchema
+from rest_framework.authtoken.models import Token
from cookbook.forms import BatchEditForm, SyncForm
-from cookbook.helper.image_processing import handle_image
-from cookbook.helper.ingredient_parser import IngredientParser
-from cookbook.helper.permission_helper import group_required, has_group_permission
-from cookbook.helper.recipe_url_import import parse_cooktime
-from cookbook.models import (Comment, Food, Ingredient, Keyword, Recipe, RecipeImport, Step, Sync,
- Unit, UserPreference)
+from cookbook.helper.permission_helper import group_required, has_group_permission, above_space_limit
+from cookbook.models import (Comment, Food, Keyword, Recipe, RecipeImport, Sync,
+ Unit, UserPreference, BookmarkletImport)
from cookbook.tables import SyncTable
from recipes import settings
@group_required('user')
def sync(request):
- if request.space.max_recipes != 0 and Recipe.objects.filter(space=request.space).count() >= request.space.max_recipes: # TODO move to central helper function
- messages.add_message(request, messages.WARNING, _('You have reached the maximum number of recipes for your space.'))
- return HttpResponseRedirect(reverse('index'))
-
- if request.space.max_users != 0 and UserPreference.objects.filter(space=request.space).count() > request.space.max_users:
- messages.add_message(request, messages.WARNING, _('You have more users than allowed in your space.'))
+ limit, msg = above_space_limit(request.space)
+ if limit:
+ messages.add_message(request, messages.WARNING, msg)
return HttpResponseRedirect(reverse('index'))
if request.space.demo or settings.HOSTED:
@@ -123,103 +109,21 @@ def batch_edit(request):
@group_required('user')
-@atomic
def import_url(request):
- if request.space.max_recipes != 0 and Recipe.objects.filter(space=request.space).count() >= request.space.max_recipes: # TODO move to central helper function
- messages.add_message(request, messages.WARNING, _('You have reached the maximum number of recipes for your space.'))
+ limit, msg = above_space_limit(request.space)
+ if limit:
+ messages.add_message(request, messages.WARNING, msg)
return HttpResponseRedirect(reverse('index'))
- if request.space.max_users != 0 and UserPreference.objects.filter(space=request.space).count() > request.space.max_users:
- messages.add_message(request, messages.WARNING, _('You have more users than allowed in your space.'))
- return HttpResponseRedirect(reverse('index'))
-
- if request.method == 'POST':
- data = json.loads(request.body)
- data['cookTime'] = parse_cooktime(data.get('cookTime', ''))
- data['prepTime'] = parse_cooktime(data.get('prepTime', ''))
-
- recipe = Recipe.objects.create(
- name=data['name'],
- description=data['description'],
- waiting_time=data['cookTime'],
- working_time=data['prepTime'],
- servings=data['servings'],
- internal=True,
- created_by=request.user,
- space=request.space,
- )
-
- step = Step.objects.create(
- instruction=data['recipeInstructions'], space=request.space,
- )
-
- recipe.steps.add(step)
-
- for kw in data['keywords']:
- if data['all_keywords']: # do not remove this check :) https://github.com/vabene1111/recipes/issues/645
- k, created = Keyword.objects.get_or_create(name=kw['text'], space=request.space)
- recipe.keywords.add(k)
- else:
- try:
- k = Keyword.objects.get(name=kw['text'], space=request.space)
- recipe.keywords.add(k)
- except ObjectDoesNotExist:
- pass
-
- ingredient_parser = IngredientParser(request, True)
- for ing in data['recipeIngredient']:
- original = ing.pop('original', None) or ing.pop('original_text', None)
- ingredient = Ingredient(original_text=original, space=request.space, )
-
- if food_text := ing['ingredient']['text'].strip():
- ingredient.food = ingredient_parser.get_food(food_text)
-
- if ing['unit']:
- if unit_text := ing['unit']['text'].strip():
- ingredient.unit = ingredient_parser.get_unit(unit_text)
-
- # TODO properly handle no_amount recipes
- if isinstance(ing['amount'], str):
- try:
- ingredient.amount = float(ing['amount'].replace(',', '.'))
- except ValueError:
- ingredient.no_amount = True
- pass
- elif isinstance(ing['amount'], float) \
- or isinstance(ing['amount'], int):
- ingredient.amount = ing['amount']
- ingredient.note = ing['note'].strip() if 'note' in ing else ''
-
- ingredient.save()
- step.ingredients.add(ingredient)
-
- if 'image' in data and data['image'] != '' and data['image'] is not None:
- try:
- response = requests.get(data['image'])
-
- img, filetype = handle_image(request, File(BytesIO(response.content), name='image'))
- recipe.image = File(
- img, name=f'{uuid.uuid4()}_{recipe.pk}{filetype}'
- )
- recipe.save()
- except UnidentifiedImageError as e:
- print(e)
- pass
- except MissingSchema as e:
- print(e)
- pass
- except Exception as e:
- print(e)
- pass
-
- return HttpResponse(reverse('view_recipe', args=[recipe.pk]))
+ if (api_token := Token.objects.filter(user=request.user).first()) is None:
+ api_token = Token.objects.create(user=request.user)
+ bookmarklet_import_id = -1
if 'id' in request.GET:
- context = {'bookmarklet': request.GET.get('id', '')}
- else:
- context = {}
+ if bookmarklet_import := BookmarkletImport.objects.filter(id=request.GET['id']).first():
+ bookmarklet_import_id = bookmarklet_import.pk
- return render(request, 'url_import.html', context)
+ return render(request, 'url_import.html', {'api_token': api_token, 'bookmarklet_import_id': bookmarklet_import_id})
class Object(object):
diff --git a/cookbook/views/edit.py b/cookbook/views/edit.py
index ff3134662..50e878662 100644
--- a/cookbook/views/edit.py
+++ b/cookbook/views/edit.py
@@ -9,7 +9,7 @@ from django.views.generic import UpdateView
from django.views.generic.edit import FormMixin
from cookbook.forms import CommentForm, ExternalRecipeForm, MealPlanForm, StorageForm, SyncForm
-from cookbook.helper.permission_helper import GroupRequiredMixin, OwnerRequiredMixin, group_required
+from cookbook.helper.permission_helper import GroupRequiredMixin, OwnerRequiredMixin, group_required, above_space_limit
from cookbook.models import (Comment, MealPlan, MealType, Recipe, RecipeImport, Storage, Sync,
UserPreference)
from cookbook.provider.dropbox import Dropbox
@@ -39,12 +39,9 @@ def convert_recipe(request, pk):
@group_required('user')
def internal_recipe_update(request, pk):
- if request.space.max_recipes != 0 and Recipe.objects.filter(space=request.space).count() > request.space.max_recipes: # TODO move to central helper function
- messages.add_message(request, messages.WARNING, _('You have reached the maximum number of recipes for your space.'))
- return HttpResponseRedirect(reverse('view_recipe', args=[pk]))
-
- if request.space.max_users != 0 and UserPreference.objects.filter(space=request.space).count() > request.space.max_users:
- messages.add_message(request, messages.WARNING, _('You have more users than allowed in your space.'))
+ limit, msg = above_space_limit(request.space)
+ if limit:
+ messages.add_message(request, messages.WARNING, msg)
return HttpResponseRedirect(reverse('view_recipe', args=[pk]))
recipe_instance = get_object_or_404(Recipe, pk=pk, space=request.space)
diff --git a/cookbook/views/import_export.py b/cookbook/views/import_export.py
index 581416878..d81777fce 100644
--- a/cookbook/views/import_export.py
+++ b/cookbook/views/import_export.py
@@ -10,16 +10,18 @@ from django.urls import reverse
from django.utils.translation import gettext as _
from cookbook.forms import ExportForm, ImportExportBase, ImportForm
-from cookbook.helper.permission_helper import group_required
+from cookbook.helper.permission_helper import group_required, above_space_limit
from cookbook.helper.recipe_search import RecipeSearch
from cookbook.integration.cheftap import ChefTap
from cookbook.integration.chowdown import Chowdown
from cookbook.integration.cookbookapp import CookBookApp
+from cookbook.integration.cookmate import Cookmate
from cookbook.integration.copymethat import CopyMeThat
from cookbook.integration.default import Default
from cookbook.integration.domestica import Domestica
from cookbook.integration.mealie import Mealie
from cookbook.integration.mealmaster import MealMaster
+from cookbook.integration.melarecipes import MelaRecipes
from cookbook.integration.nextcloud_cookbook import NextcloudCookbook
from cookbook.integration.openeats import OpenEats
from cookbook.integration.paprika import Paprika
@@ -74,16 +76,17 @@ def get_integration(request, export_type):
return CopyMeThat(request, export_type)
if export_type == ImportExportBase.PDF:
return PDFexport(request, export_type)
+ if export_type == ImportExportBase.MELARECIPES:
+ return MelaRecipes(request, export_type)
+ if export_type == ImportExportBase.COOKMATE:
+ return Cookmate(request, export_type)
@group_required('user')
def import_recipe(request):
- if request.space.max_recipes != 0 and Recipe.objects.filter(space=request.space).count() >= request.space.max_recipes: # TODO move to central helper function
- messages.add_message(request, messages.WARNING, _('You have reached the maximum number of recipes for your space.'))
- return HttpResponseRedirect(reverse('index'))
-
- if request.space.max_users != 0 and UserPreference.objects.filter(space=request.space).count() > request.space.max_users:
- messages.add_message(request, messages.WARNING, _('You have more users than allowed in your space.'))
+ limit, msg = above_space_limit(request.space)
+ if limit:
+ messages.add_message(request, messages.WARNING, msg)
return HttpResponseRedirect(reverse('index'))
if request.method == "POST":
@@ -100,7 +103,7 @@ def import_recipe(request):
t.setDaemon(True)
t.start()
- return JsonResponse({'import_id': [il.pk]})
+ return JsonResponse({'import_id': il.pk})
except NotImplementedError:
return JsonResponse(
{
diff --git a/cookbook/views/views.py b/cookbook/views/views.py
index a15ac0447..f47eed694 100644
--- a/cookbook/views/views.py
+++ b/cookbook/views/views.py
@@ -662,11 +662,10 @@ def test(request):
if not settings.DEBUG:
return HttpResponseRedirect(reverse('index'))
- with scopes_disabled():
- result = ShoppingList.objects.filter(
- Q(created_by=request.user) | Q(shared=request.user)).filter(
- space=request.space).values().distinct()
- return JsonResponse(list(result), safe=False, json_dumps_params={'indent': 2})
+ if (api_token := Token.objects.filter(user=request.user).first()) is None:
+ api_token = Token.objects.create(user=request.user)
+
+ return render(request, 'test.html', {'api_token': api_token})
def test2(request):
diff --git a/docs/features/import_export.md b/docs/features/import_export.md
index 98424158d..a7b448794 100644
--- a/docs/features/import_export.md
+++ b/docs/features/import_export.md
@@ -38,6 +38,8 @@ Overview of the capabilities of the different integrations.
| Plantoeat | ✔️ | ❌ | ✔ |
| CookBookApp | ✔️ | ⌚ | ✔️ |
| CopyMeThat | ✔️ | ❌ | ✔️ |
+| Melarecipes | ✔️ | ⌚ | ✔️ |
+| Cookmate | ✔️ | ⌚ | ✔️ |
| PDF (experimental) | ⌚️ | ✔️ | ✔️ |
✔️ = implemented, ❌ = not implemented and not possible/planned, ⌚ = not yet implemented
@@ -225,6 +227,19 @@ CookBookApp can export .zip files containing .html files. Upload the entire ZIP
CopyMeThat can export .zip files containing an `.html` file as well as a folder containing all the images. Upload the entire ZIP to Tandoor to import all included recipes.
+## Cookmate
+Cookmate allows you to export a `.mcb` file which you can simply upload to tandoor and import all your recipes.
+
+## RecetteTek
+RecetteTek exports are `.rtk` files which can simply be uploaded to tandoor to import all your recipes.
+
+## Melarecipes
+
+Melarecipes provides multiple export formats but only the `MelaRecipes` format can export the complete collection.
+Perform this export and open the `.melarecipes` file using your favorite archive opening program (e.g 7zip).
+Repeat this if the file contains another `.melarecipes` file until you get a list of one or many `.melarecipe` files.
+Upload all `.melarecipe` files you want to import to tandoor and start the import.
+
## PDF
The PDF Exporter is an experimental feature that uses the puppeteer browser renderer to render each recipe and export it to PDF.
diff --git a/openapitools.json b/openapitools.json
new file mode 100644
index 000000000..e69de29bb
diff --git a/vue/src/apps/ImportView/ImportView.vue b/vue/src/apps/ImportView/ImportView.vue
new file mode 100644
index 000000000..6e6f39b7a
--- /dev/null
+++ b/vue/src/apps/ImportView/ImportView.vue
@@ -0,0 +1,671 @@
+
+
+
+
+
+
{{ $t('Import') }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Website
+
+
+
+
+
+
+
+
+ Multiple Recipes Single Recipe
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Import
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Options
+
+
+
+
+
+
+
+
+
{{
+ recipe_json.name
+ }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Click the image you want to import for this
+ recipe
+
No additional images found in source.
+
+
+
+
+
+
+
+
+
+ Keywords
+
+ {x.show = true; return x}))">
+
+ {x.show = false; return x}))">
+
+
+
+
+
+
+
+
+ {{
+ k.label
+ }}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Import
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Import &
+ View
+
+ Import & Edit
+
+ Import &
+ Restart
+
+ Restart
+
+
+
+
+
+
+
+
+
+
+ {{ i.name }}
+
+
+
+ {{ $t('import_duplicates') }}
+
+
+ {{
+ $t('Help')
+ }}
+
+
+
+ {{ $t('Import') }}
+
+
+
+
+
+
+
+
+
+
+ {{ $t('Import') }}
+
+
+
+
+
+
+ Some pages cannot be imported from their URL, the Bookmarklet can be used to import from
+ some of them anyway.
+ 1. Drag the following button to your bookmarks bar Import into
+ Tandoor
+
+ 2. Open the page you want to import from
+ 3. Click on the bookmark to perform the import
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/vue/src/apps/ImportView/ImportViewStepEditor.vue b/vue/src/apps/ImportView/ImportViewStepEditor.vue
new file mode 100644
index 000000000..f9983e2e8
--- /dev/null
+++ b/vue/src/apps/ImportView/ImportViewStepEditor.vue
@@ -0,0 +1,151 @@
+
+
+
Steps
+
+
+ {{ $t('All') }}
+
+ {{ $t('All') }}
+
+
+
+
+
+
+ {{ i.original_text }}
+
+
+
+
+
+
+
+
+ x === s),1)">
+
+
+
+
+
+
+
+
+
+
+ x === s) +1,0,{ingredients:[], instruction: ''})">
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/vue/src/apps/ImportView/main.js b/vue/src/apps/ImportView/main.js
new file mode 100644
index 000000000..f55808a72
--- /dev/null
+++ b/vue/src/apps/ImportView/main.js
@@ -0,0 +1,18 @@
+import Vue from 'vue'
+import App from './ImportView.vue'
+import i18n from '@/i18n'
+
+Vue.config.productionTip = false
+
+// TODO move this and other default stuff to centralized JS file (verify nothing breaks)
+let publicPath = localStorage.STATIC_URL + 'vue/'
+if (process.env.NODE_ENV === 'development') {
+ publicPath = 'http://localhost:8080/'
+}
+export default __webpack_public_path__ = publicPath // eslint-disable-line
+
+
+new Vue({
+ i18n,
+ render: h => h(App),
+}).$mount('#app')
diff --git a/vue/src/apps/RecipeEditView/RecipeEditView.vue b/vue/src/apps/RecipeEditView/RecipeEditView.vue
index 17a18dbae..038507708 100644
--- a/vue/src/apps/RecipeEditView/RecipeEditView.vue
+++ b/vue/src/apps/RecipeEditView/RecipeEditView.vue
@@ -370,6 +370,9 @@
+
+ {{ingredient.original_text}}
+
diff --git a/vue/src/components/KeywordsComponent.vue b/vue/src/components/KeywordsComponent.vue
index cac06d38c..294021762 100644
--- a/vue/src/components/KeywordsComponent.vue
+++ b/vue/src/components/KeywordsComponent.vue
@@ -1,6 +1,6 @@
-
+
{{ k.label }}
diff --git a/vue/src/components/RecipeCard.vue b/vue/src/components/RecipeCard.vue
index 51bb4409e..6871f475d 100755
--- a/vue/src/components/RecipeCard.vue
+++ b/vue/src/components/RecipeCard.vue
@@ -2,7 +2,7 @@
-
+
@@ -24,7 +24,7 @@
-
+
{{ recipe.description.substr(0, text_length) + "\u2026" }}
@@ -78,6 +78,7 @@ export default {
footer_text: String,
footer_icon: String,
detailed: { type: Boolean, default: true },
+ show_context_menu: { type: Boolean, default: true }
},
mounted() {},
computed: {
diff --git a/vue/src/locales/en.json b/vue/src/locales/en.json
index 3f89c6a22..e319332ce 100644
--- a/vue/src/locales/en.json
+++ b/vue/src/locales/en.json
@@ -336,6 +336,11 @@
"ingredient_list": "Ingredient List",
"explain": "Explain",
"filter": "Filter",
+ "Website": "Website",
+ "App": "App",
+ "Bookmarklet": "Bookmarklet",
+ "import_duplicates": "To prevent duplicates recipes with the same name as existing ones are ignored. Check this box to import everything.",
+ "paste_json": "Paste json or html source here to load recipe.",
"search_no_recipes": "Could not find any recipes!",
"search_import_help_text": "Import a recipe from an external website or application.",
"search_create_help_text": "Create a new recipe directly in Tandoor.",
diff --git a/vue/src/utils/integration.js b/vue/src/utils/integration.js
new file mode 100644
index 000000000..f5bace760
--- /dev/null
+++ b/vue/src/utils/integration.js
@@ -0,0 +1,24 @@
+// containing all data and functions regarding the different integrations
+
+export const INTEGRATIONS = [
+ {id: 'DEFAULT', name: "Tandoor", import: true, export: true, help_url: 'https://docs.tandoor.dev/features/import_export/#default'},
+ {id: 'CHEFTAP', name: "Cheftap", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#cheftap'},
+ {id: 'CHOWDOWN', name: "Chowdown", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#chowdown'},
+ {id: 'COOKBOOKAPP', name: "CookBookApp", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#cookbookapp'},
+ {id: 'COOKMATE', name: "Cookmate", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#cookmate'},
+ {id: 'COPYMETHAT', name: "CopyMeThat", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#copymethat'},
+ {id: 'DOMESTICA', name: "Domestica", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#domestica'},
+ {id: 'MEALIE', name: "Mealie", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#mealie'},
+ {id: 'MEALMASTER', name: "Mealmaster", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#mealmaster'},
+ {id: 'MELARECIPES', name: "Melarecipes", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#melarecipes'},
+ {id: 'NEXTCLOUD', name: "Nextcloud Cookbook", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#nextcloud'},
+ {id: 'OPENEATS', name: "Openeats", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#openeats'},
+ {id: 'PAPRIKA', name: "Paprika", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#paprika'},
+ {id: 'PEPPERPLATE', name: "Pepperplate", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#pepperplate'},
+ {id: 'PLANTOEAT', name: "Plantoeat", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#plantoeat'},
+ {id: 'RECETTETEK', name: "RecetteTek", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#recettetek'},
+ {id: 'RECIPEKEEPER', name: "Recipekeeper", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#recipekeeper'},
+ {id: 'RECIPESAGE', name: "Recipesage", import: true, export: true, help_url: 'https://docs.tandoor.dev/features/import_export/#recipesage'},
+ {id: 'REZKONV', name: "Rezkonv", import: true, export: false, help_url: 'https://docs.tandoor.dev/features/import_export/#rezkonv'},
+ {id: 'SAFRON', name: "Safron", import: true, export: true, help_url: 'https://docs.tandoor.dev/features/import_export/#safron'},
+]
diff --git a/vue/src/utils/openapi/api.ts b/vue/src/utils/openapi/api.ts
index b9c9b2553..c88cb5e21 100644
--- a/vue/src/utils/openapi/api.ts
+++ b/vue/src/utils/openapi/api.ts
@@ -472,7 +472,7 @@ export interface FoodRecipe {
* @type {string}
* @memberof FoodRecipe
*/
- name: string;
+ name?: string;
/**
*
* @type {string}
@@ -537,7 +537,7 @@ export interface FoodSubstitute {
* @type {string}
* @memberof FoodSubstitute
*/
- name: string;
+ name?: string;
}
/**
*
@@ -752,12 +752,6 @@ export interface Ingredient {
* @memberof Ingredient
*/
original_text?: string | null;
- /**
- *
- * @type {string}
- * @memberof Ingredient
- */
- used_in_recipes?: string;
}
/**
*
@@ -1862,6 +1856,12 @@ export interface RecipeImage {
* @memberof RecipeImage
*/
image?: any | null;
+ /**
+ *
+ * @type {string}
+ * @memberof RecipeImage
+ */
+ image_url?: string | null;
}
/**
*
@@ -1923,12 +1923,6 @@ export interface RecipeIngredients {
* @memberof RecipeIngredients
*/
original_text?: string | null;
- /**
- *
- * @type {string}
- * @memberof RecipeIngredients
- */
- used_in_recipes?: string;
}
/**
*
@@ -2197,7 +2191,7 @@ export interface RecipeSimple {
* @type {string}
* @memberof RecipeSimple
*/
- name: string;
+ name?: string;
/**
*
* @type {string}
@@ -5239,10 +5233,11 @@ export const ApiApiAxiosParamCreator = function (configuration?: Configuration)
*
* @param {string} id A unique integer value identifying this recipe.
* @param {any} [image]
+ * @param {string} [imageUrl]
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
- imageRecipe: async (id: string, image?: any, options: any = {}): Promise => {
+ imageRecipe: async (id: string, image?: any, imageUrl?: string, options: any = {}): Promise => {
// verify required parameter 'id' is not null or undefined
assertParamExists('imageRecipe', 'id', id)
const localVarPath = `/api/recipe/{id}/image/`
@@ -5264,6 +5259,10 @@ export const ApiApiAxiosParamCreator = function (configuration?: Configuration)
localVarFormParams.append('image', image as any);
}
+ if (imageUrl !== undefined) {
+ localVarFormParams.append('image_url', imageUrl as any);
+ }
+
localVarHeaderParameter['Content-Type'] = 'multipart/form-data';
@@ -10353,11 +10352,12 @@ export const ApiApiFp = function(configuration?: Configuration) {
*
* @param {string} id A unique integer value identifying this recipe.
* @param {any} [image]
+ * @param {string} [imageUrl]
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
- async imageRecipe(id: string, image?: any, options?: any): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise> {
- const localVarAxiosArgs = await localVarAxiosParamCreator.imageRecipe(id, image, options);
+ async imageRecipe(id: string, image?: any, imageUrl?: string, options?: any): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise> {
+ const localVarAxiosArgs = await localVarAxiosParamCreator.imageRecipe(id, image, imageUrl, options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
},
/**
@@ -12186,11 +12186,12 @@ export const ApiApiFactory = function (configuration?: Configuration, basePath?:
*
* @param {string} id A unique integer value identifying this recipe.
* @param {any} [image]
+ * @param {string} [imageUrl]
* @param {*} [options] Override http request option.
* @throws {RequiredError}
*/
- imageRecipe(id: string, image?: any, options?: any): AxiosPromise {
- return localVarFp.imageRecipe(id, image, options).then((request) => request(axios, basePath));
+ imageRecipe(id: string, image?: any, imageUrl?: string, options?: any): AxiosPromise {
+ return localVarFp.imageRecipe(id, image, imageUrl, options).then((request) => request(axios, basePath));
},
/**
*
@@ -14004,12 +14005,13 @@ export class ApiApi extends BaseAPI {
*
* @param {string} id A unique integer value identifying this recipe.
* @param {any} [image]
+ * @param {string} [imageUrl]
* @param {*} [options] Override http request option.
* @throws {RequiredError}
* @memberof ApiApi
*/
- public imageRecipe(id: string, image?: any, options?: any) {
- return ApiApiFp(this.configuration).imageRecipe(id, image, options).then((request) => request(this.axios, this.basePath));
+ public imageRecipe(id: string, image?: any, imageUrl?: string, options?: any) {
+ return ApiApiFp(this.configuration).imageRecipe(id, image, imageUrl, options).then((request) => request(this.axios, this.basePath));
}
/**
diff --git a/vue/src/utils/utils.js b/vue/src/utils/utils.js
index 88c2b7905..f76469822 100644
--- a/vue/src/utils/utils.js
+++ b/vue/src/utils/utils.js
@@ -50,37 +50,37 @@ export class StandardToasts {
static FAIL_MOVE = "FAIL_MOVE"
static FAIL_MERGE = "FAIL_MERGE"
- static makeStandardToast(toast, err_details = undefined) {
+ static makeStandardToast(toast, err_details = undefined) { //TODO err_details render very ugly, improve this maybe by using a custom toast component (in conjunction with error logging maybe)
switch (toast) {
case StandardToasts.SUCCESS_CREATE:
- makeToast(i18n.tc("Success"), i18n.tc("success_creating_resource"), "success")
+ makeToast(i18n.tc("Success"), i18n.tc("success_creating_resource") + (err_details ? "\n" + err_details : ""), "success")
break
case StandardToasts.SUCCESS_FETCH:
- makeToast(i18n.tc("Success"), i18n.tc("success_fetching_resource"), "success")
+ makeToast(i18n.tc("Success"), i18n.tc("success_fetching_resource") + (err_details ? "\n" + err_details : ""), "success")
break
case StandardToasts.SUCCESS_UPDATE:
- makeToast(i18n.tc("Success"), i18n.tc("success_updating_resource"), "success")
+ makeToast(i18n.tc("Success"), i18n.tc("success_updating_resource") + (err_details ? "\n" + err_details : ""), "success")
break
case StandardToasts.SUCCESS_DELETE:
- makeToast(i18n.tc("Success"), i18n.tc("success_deleting_resource"), "success")
+ makeToast(i18n.tc("Success"), i18n.tc("success_deleting_resource") + (err_details ? "\n" + err_details : ""), "success")
break
case StandardToasts.SUCCESS_MOVE:
- makeToast(i18n.tc("Success"), i18n.tc("success_moving_resource"), "success")
+ makeToast(i18n.tc("Success"), i18n.tc("success_moving_resource") + (err_details ? "\n" + err_details : ""), "success")
break
case StandardToasts.SUCCESS_MERGE:
- makeToast(i18n.tc("Success"), i18n.tc("success_merging_resource"), "success")
+ makeToast(i18n.tc("Success"), i18n.tc("success_merging_resource") + (err_details ? "\n" + err_details : ""), "success")
break
case StandardToasts.FAIL_CREATE:
- makeToast(i18n.tc("Failure"), i18n.tc("err_creating_resource"), "danger")
+ makeToast(i18n.tc("Failure"), i18n.tc("err_creating_resource") + (err_details ? "\n" + err_details : ""), "danger")
break
case StandardToasts.FAIL_FETCH:
- makeToast(i18n.tc("Failure"), i18n.tc("err_fetching_resource"), "danger")
+ makeToast(i18n.tc("Failure"), i18n.tc("err_fetching_resource") + (err_details ? "\n" + err_details : ""), "danger")
break
case StandardToasts.FAIL_UPDATE:
- makeToast(i18n.tc("Failure"), i18n.tc("err_updating_resource"), "danger")
+ makeToast(i18n.tc("Failure"), i18n.tc("err_updating_resource") + (err_details ? "\n" + err_details : ""), "danger")
break
case StandardToasts.FAIL_DELETE:
- makeToast(i18n.tc("Failure"), i18n.tc("err_deleting_resource"), "danger")
+ makeToast(i18n.tc("Failure"), i18n.tc("err_deleting_resource") + (err_details ? "\n" + err_details : ""), "danger")
break
case StandardToasts.FAIL_DELETE_PROTECTED:
makeToast(i18n.tc("Protected"), i18n.tc("err_deleting_protected_resource"), "danger")
@@ -149,6 +149,27 @@ export function resolveDjangoUrl(url, params = null) {
}
}
+/*
+ * Utility functions to use djangos static files
+ * */
+
+export const StaticMixin = {
+ methods: {
+ /**
+ * access django static files from javascript
+ * @param {string} param path to static file
+ */
+ resolveDjangoStatic: function (param) {
+ return resolveDjangoStatic(param)
+ },
+ },
+}
+
+export function resolveDjangoStatic(param) {
+ let url = localStorage.getItem('STATIC_URL') + param
+ return url.replace('//','/') //replace // with / in case param started with / which resulted in // after the static base url
+}
+
/*
* other utilities
* */
diff --git a/vue/vue.config.js b/vue/vue.config.js
index 27436cc77..776579905 100644
--- a/vue/vue.config.js
+++ b/vue/vue.config.js
@@ -13,6 +13,10 @@ const pages = {
entry: "./src/apps/OfflineView/main.js",
chunks: ["chunk-vendors"],
},
+ import_view: {
+ entry: "./src/apps/ImportView/main.js",
+ chunks: ["chunk-vendors"],
+ },
import_response_view: {
entry: "./src/apps/ImportResponseView/main.js",
chunks: ["chunk-vendors"],
diff --git a/vue/yarn.lock b/vue/yarn.lock
index bf4fc6a6c..371d69f4d 100644
--- a/vue/yarn.lock
+++ b/vue/yarn.lock
@@ -12697,6 +12697,11 @@ vue-infinite-loading@^2.4.5:
resolved "https://registry.yarnpkg.com/vue-infinite-loading/-/vue-infinite-loading-2.4.5.tgz#cc20fd40af7f20188006443c99b60470cf1de1b3"
integrity sha512-xhq95Mxun060bRnsOoLE2Be6BR7jYwuC89kDe18+GmCLVrRA/dU0jrGb12Xu6NjmKs+iTW0AA6saSEmEW4cR7g==
+vue-jstree@^2.1.6:
+ version "2.1.6"
+ resolved "https://registry.yarnpkg.com/vue-jstree/-/vue-jstree-2.1.6.tgz#44827ad72953ed77da6590ce4e8f37f7787f8653"
+ integrity sha512-vtUmhLbfE2JvcnYNRXauJPkNJSRO/f9BTsbxV+ESXP/mMQPVUIYI4EkSHKSEOxVDHTU7SfLp/AxplmaAl6ctcg==
+
"vue-loader-v16@npm:vue-loader@^16.1.0":
version "16.8.3"
resolved "https://registry.yarnpkg.com/vue-loader/-/vue-loader-16.8.3.tgz#d43e675def5ba9345d6c7f05914c13d861997087"