views.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. # coding=utf-8
  2. from rest_framework.views import APIView
  3. from django.db.models import Q
  4. import traceback
  5. import datetime
  6. from django.db import transaction
  7. from rest_framework.serializers import ValidationError
  8. from django.utils import timezone
  9. from rest_framework.decorators import action
  10. from utils.permission import isLogin, permission_required, check_permission
  11. from utils.custom_modelviewset import CustomModelViewSet
  12. from utils import response_ok, response_error
  13. from utils.exceptions import CustomError
  14. from apps.log.models import BizLog
  15. from apps.option.models import Option
  16. from apps.option.serializers import OptionComboboxSerializer
  17. from .models import ReportCustomer,NewCustomer,Review
  18. from .serializers import ReportCustomerSerializer,NewCustomerSerializer, ReviewSerializer
  19. from .filters import ReportCustomerFilter,NewCustomerFilter,ReviewFilter
  20. from django.contrib.auth import get_user_model
  21. User = get_user_model()
  22. from apps.order.serializers import Order, OrderSerializer
  23. class ReportCustomerViewSet(CustomModelViewSet):
  24. permission_classes = [isLogin]
  25. queryset = ReportCustomer.objects.filter()
  26. serializer_class = ReportCustomerSerializer
  27. @permission_required('customer.view_report_customer')
  28. def filter_queryset(self, queryset):
  29. queryset = queryset.filter(
  30. Q(store_id__in=self.request.user.get_manager_range()) |
  31. Q(create_user=self.request.user))
  32. f = ReportCustomerFilter(self.request.GET, queryset=queryset)
  33. return f.qs
  34. @permission_required('customer.add_report_customer')
  35. def perform_create(self, serializer):
  36. super(ReportCustomerViewSet, self).perform_create(serializer)
  37. instance = serializer.instance
  38. validated_data = serializer.validated_data
  39. BizLog.objects.addnew(self.request.user, BizLog.INSERT,
  40. u'添加客户报备[%s],id=%d' % (instance.name, instance.id), validated_data)
  41. @permission_required('customer.add_report_customer')
  42. def perform_update(self, serializer):
  43. super(ReportCustomerViewSet, self).perform_update(serializer)
  44. instance = serializer.instance
  45. validated_data = serializer.validated_data
  46. BizLog.objects.addnew(self.request.user, BizLog.UPDATE,
  47. u'修改客户报备[%s],id=%d' % (instance.name, instance.id), validated_data)
  48. @permission_required('customer.delete_report_customer')
  49. def perform_destroy(self, instance):
  50. BizLog.objects.addnew(self.request.user, BizLog.DELETE,
  51. u'删除客户报备[%s],id=%d' % (instance.name, instance.id))
  52. super(ReportCustomerViewSet, self).perform_destroy(instance)
  53. @action(methods=['post'], detail=True)
  54. def dispatch_customer(self, request, pk):
  55. check_permission(request, 'customer.check_report_customer')
  56. # 报备客户审核 分配
  57. user_id = request.POST.get('user')
  58. try:
  59. instance = ReportCustomer.objects.filter(id=pk).first()
  60. if not instance:
  61. raise CustomError('当前客户报备信息有误,请刷新重试!')
  62. instance.check_user = self.request.user
  63. instance.report_status = ReportCustomer.CHECKED
  64. instance.check_time = timezone.now()
  65. instance.save()
  66. BizLog.objects.addnew(request.user, BizLog.INSERT,
  67. u'分配客户报备[%s],id=%d' % (instance.name, instance.id))
  68. # 创建潜客跟踪表
  69. potential_level = Option.objects.filter(type=Option.POTENTIAL_LEVEL, enable=True).order_by('sort').first()
  70. stage_progress = Option.objects.filter(type=Option.STAGE_PROGRESS, enable=True).order_by('sort').first()
  71. if not potential_level.track_day:
  72. raise CustomError('当前潜客级别,没有可用跟踪天数!')
  73. next_time = (timezone.now() + datetime.timedelta(days=potential_level.track_day)).strftime('%Y-%m-%d')
  74. customer = NewCustomer.objects.create(report_customer=instance, potential_level=potential_level,
  75. track_user_id=user_id, next_time=next_time, stage_progress=stage_progress,
  76. create_user=instance.create_user,store=instance.store,
  77. name=instance.name,tel=instance.tel,village=instance.village,
  78. address=instance.address,source=instance.source,notes=instance.notes,
  79. )
  80. projects = instance.project.all()
  81. for project in projects:
  82. customer.project.add(project.id)
  83. except CustomError as e:
  84. return response_error(e.get_error_msg())
  85. except Exception as e:
  86. return response_error(str(e))
  87. return response_ok()
  88. @action(methods=['post'], detail=True)
  89. def sign_hit(self, request, pk):
  90. check_permission(request, 'customer.check_report_customer')
  91. # 报备客户标记撞单
  92. try:
  93. instance = ReportCustomer.objects.filter(id=pk).first()
  94. if not instance:
  95. raise CustomError('当前客户报备信息有误,请刷新重试!')
  96. instance.check_user = self.request.user
  97. instance.report_status = ReportCustomer.REPEAT_REPORT
  98. instance.check_time = timezone.now()
  99. instance.save()
  100. BizLog.objects.addnew(request.user, BizLog.INSERT,
  101. u'标记客户[%s]为撞单,id=%d' % (instance.name, instance.id))
  102. except CustomError as e:
  103. return response_error(e.get_error_msg())
  104. except Exception as e:
  105. return response_error(str(e))
  106. return response_ok()
  107. class ReportCustomerDictView(APIView):
  108. permission_classes = [isLogin]
  109. def get(self, request):
  110. source = Option.objects.filter(type=Option.CUSTOMER_SOURCE, enable=True)
  111. project = Option.objects.filter(type=Option.CATEGORY, enable=True)
  112. level = Option.objects.filter(type=Option.POTENTIAL_LEVEL, enable=True)
  113. serializer_source = OptionComboboxSerializer(source, many=True)
  114. serializer_project = OptionComboboxSerializer(project, many=True)
  115. serializer_level = OptionComboboxSerializer(level, many=True)
  116. return response_ok({
  117. 'source': serializer_source.data,
  118. 'project': serializer_project.data,
  119. 'level': serializer_level.data,
  120. })
  121. class UserDictView(APIView):
  122. permission_classes = [isLogin]
  123. def get(self, request):
  124. store = request.user.store
  125. if not store:
  126. return response_error('当前账号未绑定门店,禁止分配客户!')
  127. users = User.objects.filter(store=store).values('id', 'name')
  128. users = [{'value': user['id'], 'lable': user['name']} for user in users]
  129. return response_ok(users)
  130. class NewCustomerViewSet(CustomModelViewSet):
  131. permission_classes = []
  132. queryset = NewCustomer.objects.filter()
  133. serializer_class = NewCustomerSerializer
  134. def filter_queryset(self, queryset):
  135. queryset = queryset.filter()
  136. f = NewCustomerFilter(self.request.GET, queryset=queryset)
  137. return f.qs
  138. def perform_update(self, serializer):
  139. super(NewCustomerViewSet, self).perform_update(serializer)
  140. instance = serializer.instance
  141. validated_data = serializer.validated_data
  142. def paginate_queryset(self, queryset):
  143. if self.paginator is None or self.request.GET.get('export'):
  144. return None
  145. return self.paginator.paginate_queryset(queryset, self.request, view=self)
  146. @action(methods=['post'], detail=True)
  147. def add_review(self, request, pk):
  148. # 添加跟踪报告
  149. check_permission(request, 'customer.add_review')
  150. description = request.POST.get('description')
  151. instruction = request.POST.get('instruction')
  152. is_entry = request.POST.get('is_entry') == '1'
  153. is_giveup = request.POST.get('is_giveup') == '1'
  154. potential_level = request.POST.get('potential_level')
  155. next_time = request.POST.get('next_time')
  156. try:
  157. instance = NewCustomer.objects.filter(id=pk).first()
  158. if not instance:
  159. raise CustomError('当前客户信息有误,请刷新重试!')
  160. data = {
  161. 'customer':instance.id,
  162. 'store':request.user.store.id,
  163. 'create_user':request.user.id,
  164. 'description':description,
  165. 'instruction':instruction,
  166. 'is_entry':is_entry,
  167. 'is_giveup':is_giveup,
  168. }
  169. serializer = ReviewSerializer(data=data)
  170. if serializer.is_valid(raise_exception=True):
  171. serializer.save()
  172. sort = 1
  173. track_day = 1
  174. try:
  175. sort = int(instance.stage_progress.sort)
  176. track_day = int(instance.stage_progress.track_day)
  177. except:
  178. pass
  179. # 预定客户,sort排序大于1,提交跟踪报告此字段为空
  180. if sort > 1 or not next_time:
  181. next_time = (timezone.now() + datetime.timedelta(days=track_day)).strftime('%Y-%m-%d')
  182. data = {
  183. 'potential_level':potential_level or None, #预定客户,sort排序大于1,,无客户等级
  184. 'next_time':next_time,
  185. 'end_time':timezone.now(),
  186. 'stage_progress':instance.stage_progress.id,
  187. }
  188. ser = self.serializer_class(instance, data=data, partial=True)
  189. if ser.is_valid(raise_exception=True):
  190. ser.save()
  191. except ValidationError as e:
  192. traceback.print_exc()
  193. return response_error('数据格式有误')
  194. except CustomError as e:
  195. return response_error(e.get_error_msg())
  196. except Exception as e:
  197. return response_error(str(e))
  198. return response_ok()
  199. @action(methods=['post'], detail=True)
  200. def add_order(self, request, pk):
  201. # 更新订单进度
  202. check_permission(request, 'order.update_order_process')
  203. stage_progress = request.POST.get('stage_progress')
  204. try:
  205. with transaction.atomic():
  206. instance = NewCustomer.objects.filter(id=pk).first()
  207. if not instance:
  208. raise CustomError('当前客户信息有误,请刷新重试!')
  209. order = Order.objects.filter(customer=instance).first()
  210. if order:
  211. if order.stage_progress.end_stage:
  212. raise CustomError('当前客户订单进度已到最后阶段,请勿重复更新!')
  213. if order.status == Order.WAIT_DISPATCH:
  214. raise CustomError('当前客户订单等待分配中,请勿重复更新!')
  215. data = {
  216. 'status': Order.WAIT_DISPATCH,
  217. 'stage_progress': stage_progress,
  218. }
  219. ser = OrderSerializer(order, data=data, partial=True)
  220. if ser.is_valid(raise_exception=True):
  221. ser.save()
  222. else:
  223. data = {
  224. 'status': Order.WAIT_DISPATCH,
  225. 'stage_progress_id': stage_progress,
  226. 'customer': instance,
  227. 'service_user': instance.track_user,
  228. }
  229. ser = Order.objects.create(**data)
  230. ser.no = ser.get_no()
  231. ser.save()
  232. projects = instance.project.all()
  233. for project in projects:
  234. ser.project.add(project.id)
  235. # TODO 订单流程
  236. except ValidationError as e:
  237. traceback.print_exc()
  238. return response_error('数据格式有误')
  239. except CustomError as e:
  240. return response_error(e.get_error_msg())
  241. except Exception as e:
  242. return response_error(str(e))
  243. return response_ok()
  244. class StageCountView(APIView):
  245. permission_classes = [isLogin]
  246. def get(self,request):
  247. data = []
  248. stage_progress = Option.objects.filter(type=Option.STAGE_PROGRESS, enable=True).order_by('sort')
  249. user = request.user
  250. time_now = timezone.now().strftime('%Y-%m-%d')
  251. for stage in stage_progress:
  252. new_customers = NewCustomer.objects.filter(Q(track_user=user) |
  253. Q(track_user__store_id__in=request.user.get_manager_range()),
  254. stage_progress=stage)
  255. stage_count = {
  256. 'stage_id': stage.id, # 阶段名称
  257. 'stage': stage.name, # 阶段名称
  258. 'total_count': new_customers.count(), # 总人数
  259. 'today_count': new_customers.filter(next_time=time_now).count(), # 今日人数
  260. 'overdue_count': new_customers.filter(next_time__lt=time_now).count() # 逾期人数
  261. }
  262. data.append(stage_count)
  263. return response_ok(data)
  264. class ReviewViewSet(CustomModelViewSet):
  265. permission_classes = [isLogin]
  266. queryset = Review.objects.filter()
  267. serializer_class = ReviewSerializer
  268. def filter_queryset(self, queryset):
  269. queryset = queryset.filter()
  270. f = ReviewFilter(self.request.GET, queryset=queryset)
  271. return f.qs
  272. @action(methods=['post'], detail=True)
  273. def track_review(self, request, pk):
  274. check_permission(request, 'customer.check_review')
  275. # 跟踪审核
  276. check_status = request.POST.get('check_status')
  277. check_comment = request.POST.get('check_comment')
  278. next_time = request.POST.get('next_time')
  279. try:
  280. instance = Review.objects.filter(id=pk).first()
  281. if int(check_status) == Review.KEEPUP:
  282. instance.check_user = request.user
  283. instance.check_status = check_status
  284. instance.check_comment = check_comment
  285. instance.check_time = timezone.now()
  286. instance.save()
  287. if int(check_status) == Review.ABANDON:
  288. instance.check_user = request.user
  289. instance.check_status = check_status
  290. instance.check_comment = check_comment
  291. instance.check_time = timezone.now()
  292. instance.save()
  293. instance.customer.status = NewCustomer.ABANDONED
  294. instance.customer.next_time = next_time
  295. instance.customer.save()
  296. except CustomError as e:
  297. return response_error(e.get_error_msg())
  298. except Exception as e:
  299. return response_error(str(e))
  300. return response_ok()