Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. import os
  2. from django.conf import settings
  3. from django.contrib import messages
  4. from django.core import exceptions
  5. from django.core.urlresolvers import reverse
  6. from django.db.models.loading import get_model
  7. from django.http import HttpResponseRedirect
  8. from django.shortcuts import get_object_or_404
  9. from django.template.loader import render_to_string
  10. from django.utils.translation import ungettext, ugettext_lazy as _
  11. from django.views.generic import (ListView, DeleteView, CreateView, UpdateView)
  12. from oscar.views.generic import BulkEditMixin
  13. from oscar.core.loading import get_classes
  14. Range = get_model('offer', 'Range')
  15. Product = get_model('catalogue', 'Product')
  16. RangeForm, RangeProductForm = get_classes('dashboard.ranges.forms',
  17. ['RangeForm', 'RangeProductForm'])
  18. RangeProductFileUpload = get_model('ranges', 'RangeProductFileUpload')
  19. class RangeListView(ListView):
  20. model = Range
  21. context_object_name = 'ranges'
  22. template_name = 'dashboard/ranges/range_list.html'
  23. class RangeCreateView(CreateView):
  24. model = Range
  25. template_name = 'dashboard/ranges/range_form.html'
  26. form_class = RangeForm
  27. def get_success_url(self):
  28. if 'action' in self.request.POST:
  29. return reverse('dashboard:range-products',
  30. kwargs={'pk': self.object.id})
  31. else:
  32. msg = render_to_string(
  33. 'dashboard/ranges/messages/range_saved.html',
  34. {'range': self.object})
  35. messages.success(self.request, msg, extra_tags='safe noicon')
  36. return reverse('dashboard:range-list')
  37. def get_context_data(self, **kwargs):
  38. ctx = super(RangeCreateView, self).get_context_data(**kwargs)
  39. ctx['title'] = _("Create range")
  40. return ctx
  41. class RangeUpdateView(UpdateView):
  42. model = Range
  43. template_name = 'dashboard/ranges/range_form.html'
  44. form_class = RangeForm
  45. def get_object(self):
  46. obj = super(RangeUpdateView, self).get_object()
  47. if not obj.is_editable:
  48. raise exceptions.PermissionDenied("Not allowed")
  49. return obj
  50. def get_success_url(self):
  51. if 'action' in self.request.POST:
  52. return reverse('dashboard:range-products',
  53. kwargs={'pk': self.object.id})
  54. else:
  55. msg = render_to_string(
  56. 'dashboard/ranges/messages/range_saved.html',
  57. {'range': self.object})
  58. messages.success(self.request, msg, extra_tags='safe noicon')
  59. return reverse('dashboard:range-list')
  60. def get_context_data(self, **kwargs):
  61. ctx = super(RangeUpdateView, self).get_context_data(**kwargs)
  62. ctx['range'] = self.object
  63. ctx['title'] = self.object.name
  64. return ctx
  65. class RangeDeleteView(DeleteView):
  66. model = Range
  67. template_name = 'dashboard/ranges/range_delete.html'
  68. context_object_name = 'range'
  69. def get_success_url(self):
  70. messages.warning(self.request, _("Range deleted"))
  71. return reverse('dashboard:range-list')
  72. class RangeProductListView(BulkEditMixin, ListView):
  73. model = Product
  74. template_name = 'dashboard/ranges/range_product_list.html'
  75. context_object_name = 'products'
  76. actions = ('remove_selected_products', 'add_products')
  77. form_class = RangeProductForm
  78. def post(self, request, *args, **kwargs):
  79. self.object_list = self.get_queryset()
  80. if request.POST.get('action', None) == 'add_products':
  81. return self.add_products(request)
  82. return super(RangeProductListView, self).post(request, *args, **kwargs)
  83. def get_range(self):
  84. if not hasattr(self, '_range'):
  85. self._range = get_object_or_404(Range, id=self.kwargs['pk'])
  86. return self._range
  87. def get_queryset(self):
  88. return self.get_range().included_products.all()
  89. def get_context_data(self, **kwargs):
  90. ctx = super(RangeProductListView, self).get_context_data(**kwargs)
  91. range = self.get_range()
  92. ctx['range'] = range
  93. if 'form' not in ctx:
  94. ctx['form'] = self.form_class(range)
  95. return ctx
  96. def remove_selected_products(self, request, products):
  97. range = self.get_range()
  98. for product in products:
  99. range.included_products.remove(product)
  100. num_products = len(products)
  101. messages.success(request, ungettext("Removed %d product from range",
  102. "Removed %d products from range",
  103. num_products) % num_products)
  104. return HttpResponseRedirect(self.get_success_url(request))
  105. def add_products(self, request):
  106. range = self.get_range()
  107. form = self.form_class(range, request.POST, request.FILES)
  108. if not form.is_valid():
  109. ctx = self.get_context_data(form=form,
  110. object_list=self.object_list)
  111. return self.render_to_response(ctx)
  112. self.handle_query_products(request, range, form)
  113. self.handle_file_products(request, range, form)
  114. return HttpResponseRedirect(self.get_success_url(request))
  115. def handle_query_products(self, request, range, form):
  116. products = form.get_products()
  117. if not products:
  118. return
  119. for product in products:
  120. range.included_products.add(product)
  121. num_products = len(products)
  122. messages.success(request, ungettext("%d product added to range",
  123. "%d products added to range",
  124. num_products) % num_products)
  125. dupe_skus = form.get_duplicate_skus()
  126. if dupe_skus:
  127. messages.warning(
  128. request,
  129. _("The products with SKUs or UPCs matching %s are already in this range") % ", ".join(dupe_skus))
  130. missing_skus = form.get_missing_skus()
  131. if missing_skus:
  132. messages.warning(request,
  133. _("No product(s) were found with SKU or UPC matching %s") % ", ".join(missing_skus))
  134. def handle_file_products(self, request, range, form):
  135. if not 'file_upload' in request.FILES:
  136. return
  137. upload = self.create_upload_object(request, range)
  138. upload.process()
  139. if not upload.was_processing_successful():
  140. messages.error(request, upload.error_message)
  141. else:
  142. msg = render_to_string(
  143. 'dashboard/ranges/messages/range_products_saved.html',
  144. {'range': range,
  145. 'upload': upload})
  146. messages.success(request, msg, extra_tags='safe noicon block')
  147. upload.delete_file()
  148. def create_upload_object(self, request, range):
  149. f = request.FILES['file_upload']
  150. destination_path = os.path.join(settings.OSCAR_UPLOAD_ROOT, f.name)
  151. with open(destination_path, 'wb+') as dest:
  152. for chunk in f.chunks():
  153. dest.write(chunk)
  154. upload = RangeProductFileUpload.objects.create(
  155. range=range,
  156. uploaded_by=request.user,
  157. filepath=destination_path,
  158. size=f.size
  159. )
  160. return upload