@@ -7,6 +7,7 @@ from jsonfield import JSONField |
||
7 | 7 |
|
8 | 8 |
from kodo.basemodels import LensmanTypeBoolMixin |
9 | 9 |
from mch.models import SaleclerkInfo |
10 |
+from sales.models import SalesResponsibilityInfo |
|
10 | 11 |
|
11 | 12 |
|
12 | 13 |
class LensmanInfo(BaseModelMixin, LensmanTypeBoolMixin): |
@@ -383,6 +384,21 @@ class UserInfo(BaseModelMixin, LensmanTypeBoolMixin): |
||
383 | 384 |
} |
384 | 385 |
] |
385 | 386 |
|
387 |
+ @property |
|
388 |
+ def srinfo(self): |
|
389 |
+ try: |
|
390 |
+ sr = SalesResponsibilityInfo.objects.get(user_id=self.user_id) |
|
391 |
+ except SalesResponsibilityInfo.DoesNotExist: |
|
392 |
+ sr = None |
|
393 |
+ sr_id = sr.sr_id if sr else '' |
|
394 |
+ is_sr = True if sr else False |
|
395 |
+ is_super_sr = True if sr and sr.is_super else False |
|
396 |
+ return { |
|
397 |
+ 'sr_id': sr_id, |
|
398 |
+ 'is_sr': is_sr, |
|
399 |
+ 'is_super_sr': is_super_sr, |
|
400 |
+ } |
|
401 |
+ |
|
386 | 402 |
def brandata(self, brand_id=None): |
387 | 403 |
if self.unionid: |
388 | 404 |
try: |
@@ -406,6 +422,7 @@ class UserInfo(BaseModelMixin, LensmanTypeBoolMixin): |
||
406 | 422 |
'cardList': self.cardList, |
407 | 423 |
'saleclerk': bool(saleclerk_info), |
408 | 424 |
'saleclerk_info': saleclerk_info, |
425 |
+ 'sr_info': self.srinfo, |
|
409 | 426 |
} |
410 | 427 |
|
411 | 428 |
|
@@ -31,6 +31,7 @@ def clerk_add(request): |
||
31 | 31 |
|
32 | 32 |
SaleclerkInfo.objects.create( |
33 | 33 |
brand_id=administrator.brand_id, |
34 |
+ brand_name=administrator.brand_name, |
|
34 | 35 |
distributor_id=distributor_id, |
35 | 36 |
distributor_name=distributor.distributor_name, |
36 | 37 |
clerk_name=clerk_name, |
@@ -18,6 +18,8 @@ def distributor_add(request): |
||
18 | 18 |
province_code = request.POST.get('province_code', '') |
19 | 19 |
province_name = request.POST.get('province_name', '') |
20 | 20 |
|
21 |
+ sr_id = request.POST.get('sr_id', '') |
|
22 |
+ |
|
21 | 23 |
# 如果省份编码不存在,而省份名称存在 |
22 | 24 |
if not province_code and province_name: |
23 | 25 |
province_code = ProvinceShortModelMixin.PROVINCE_NAME_CODE_DICT.get(province_name, '') |
@@ -35,6 +37,7 @@ def distributor_add(request): |
||
35 | 37 |
distributor_short_name=distributor_short_name, |
36 | 38 |
distributor_province_code=province_code, |
37 | 39 |
distributor_province_name=province_name, |
40 |
+ sr_id=sr_id, |
|
38 | 41 |
) |
39 | 42 |
|
40 | 43 |
return response(200, 'Distributor Add Success', u'经销商添加成功') |
@@ -72,6 +75,8 @@ def distributor_update(request): |
||
72 | 75 |
|
73 | 76 |
admin_id = request.session.get('admin_id') |
74 | 77 |
|
78 |
+ sr_id = request.POST.get('sr_id', '') |
|
79 |
+ |
|
75 | 80 |
try: |
76 | 81 |
administrator = AdministratorInfo.objects.get(admin_id=admin_id, user_status=AdministratorInfo.ACTIVATED, status=True) |
77 | 82 |
except AdministratorInfo.DoesNotExist: |
@@ -94,6 +99,8 @@ def distributor_update(request): |
||
94 | 99 |
if not province_code: |
95 | 100 |
distributor.distributor_province_code = ProvinceShortModelMixin.PROVINCE_NAME_CODE_DICT.get(province_name, '') |
96 | 101 |
|
102 |
+ distributor.sr_id = sr_id |
|
103 |
+ |
|
97 | 104 |
distributor.save() |
98 | 105 |
|
99 | 106 |
return response(200, 'Distributor Update Success', u'经销商更新成功') |
@@ -27,6 +27,8 @@ def model_add(request): |
||
27 | 27 |
|
28 | 28 |
admin_id = request.session.get('admin_id') |
29 | 29 |
|
30 |
+ is_important = request.POST.get('is_important', 0) |
|
31 |
+ |
|
30 | 32 |
try: |
31 | 33 |
administrator = AdministratorInfo.objects.get(admin_id=admin_id, user_status=AdministratorInfo.ACTIVATED, status=True) |
32 | 34 |
except AdministratorInfo.DoesNotExist: |
@@ -45,6 +47,7 @@ def model_add(request): |
||
45 | 47 |
factory_yuan=factory_yuan, |
46 | 48 |
factory_fee=monetary.Yuan2Fen(factory_yuan), |
47 | 49 |
integral=integral, |
50 |
+ is_important=is_important, |
|
48 | 51 |
) |
49 | 52 |
|
50 | 53 |
return response(200, 'Model Add Success', u'型号添加成功') |
@@ -89,6 +92,8 @@ def model_update(request): |
||
89 | 92 |
|
90 | 93 |
admin_id = request.session.get('admin_id') |
91 | 94 |
|
95 |
+ is_important = request.POST.get('is_important', 0) |
|
96 |
+ |
|
92 | 97 |
try: |
93 | 98 |
administrator = AdministratorInfo.objects.get(admin_id=admin_id, user_status=AdministratorInfo.ACTIVATED, status=True) |
94 | 99 |
except AdministratorInfo.DoesNotExist: |
@@ -121,6 +126,8 @@ def model_update(request): |
||
121 | 126 |
if integral: |
122 | 127 |
modelObj.integral = integral |
123 | 128 |
|
129 |
+ modelObj.is_important = is_important |
|
130 |
+ |
|
124 | 131 |
modelObj.save() |
125 | 132 |
|
126 | 133 |
return response(200, 'Model Update Success', u'型号更新成功') |
@@ -0,0 +1,35 @@ |
||
1 |
+# -*- coding: utf-8 -*- |
|
2 |
+ |
|
3 |
+from __future__ import division |
|
4 |
+ |
|
5 |
+from django_logit import logit |
|
6 |
+from django_response import response |
|
7 |
+from paginator import pagination |
|
8 |
+ |
|
9 |
+from mch.models import AdministratorInfo |
|
10 |
+from sales.models import SalesResponsibilityInfo |
|
11 |
+from utils.error.errno_utils import AdministratorStatusCode |
|
12 |
+ |
|
13 |
+ |
|
14 |
+@logit |
|
15 |
+def sr_list(request): |
|
16 |
+ page = request.POST.get('page', 1) |
|
17 |
+ num = request.POST.get('num', 20) |
|
18 |
+ |
|
19 |
+ admin_id = request.session.get('admin_id') |
|
20 |
+ |
|
21 |
+ try: |
|
22 |
+ administrator = AdministratorInfo.objects.get(admin_id=admin_id, user_status=AdministratorInfo.ACTIVATED, status=True) |
|
23 |
+ except AdministratorInfo.DoesNotExist: |
|
24 |
+ return response(AdministratorStatusCode.ADMINISTRATOR_NOT_FOUND) |
|
25 |
+ |
|
26 |
+ srs = SalesResponsibilityInfo.objects.filter(brand_id=administrator.brand_id, status=True).order_by('-pk') |
|
27 |
+ count = srs.count() |
|
28 |
+ srs, left = pagination(srs, page, num) |
|
29 |
+ srs = [sr.admindata for sr in srs] |
|
30 |
+ |
|
31 |
+ return response(200, 'Get SR List Success', u'获取销售担当列表成功', { |
|
32 |
+ 'srs': srs, |
|
33 |
+ 'count': count, |
|
34 |
+ 'left': left, |
|
35 |
+ }) |
@@ -5,7 +5,7 @@ from django_file_upload import views as file_views |
||
5 | 5 |
|
6 | 6 |
from account import tourguide_views |
7 | 7 |
from account import views as account_views |
8 |
-from api import admin_views, clerk_views, distributor_views, encrypt_views, mch_views, model_views, operator_views |
|
8 |
+from api import admin_views, clerk_views, distributor_views, encrypt_views, mch_views, model_views, operator_views, sr_views |
|
9 | 9 |
from box import views as box_views |
10 | 10 |
from geo import views as geo_views |
11 | 11 |
from group import (groupuser_views, lensman_views, tourguidegroup_views, tourguidegroupadmin_views, |
@@ -17,6 +17,7 @@ from operation import views as op_views |
||
17 | 17 |
from page import oauth_views, sale_views, screen_views |
18 | 18 |
from pay import views as pay_views |
19 | 19 |
from photo import views as photo_views |
20 |
+from sales import views as sales_views |
|
20 | 21 |
from server import server_views |
21 | 22 |
from statistic import views as tj_views |
22 | 23 |
|
@@ -271,6 +272,10 @@ urlpatterns += [ |
||
271 | 272 |
] |
272 | 273 |
|
273 | 274 |
urlpatterns += [ |
275 |
+ url(r'^sr/list$', sr_views.sr_list, name='sr_list'), |
|
276 |
+] |
|
277 |
+ |
|
278 |
+urlpatterns += [ |
|
274 | 279 |
url(r'^screen/admin/loginqr$', screen_views.screen_admin_loginqr, name='screen_admin_loginqr'), |
275 | 280 |
url(r'^screen/admin/loginrst$', screen_views.screen_admin_loginrst, name='screen_admin_loginrst'), |
276 | 281 |
] |
@@ -280,3 +285,10 @@ urlpatterns += [ |
||
280 | 285 |
url(r'^admin/queryusedsn$', admin_views.queryusedsn, name='queryusedsn'), |
281 | 286 |
url(r'^admin/usecoupon$', admin_views.usecoupon, name='usecoupon'), |
282 | 287 |
] |
288 |
+ |
|
289 |
+urlpatterns += [ |
|
290 |
+ url(r'^sr/submit', sales_views.sr_submit_api, name='sr_submit_api'), |
|
291 |
+ url(r'^sr/distributor/list$', sales_views.sr_distributor_list, name='sr_distributor_list'), |
|
292 |
+ url(r'^sr/distributor/tj$', sales_views.sr_distributor_tj, name='sr_distributor_tj'), |
|
293 |
+ url(r'^supersr/sr/tj$', sales_views.supersr_sr_tj, name='supersr_sr_tj'), |
|
294 |
+] |
@@ -65,6 +65,7 @@ INSTALLED_APPS = ( |
||
65 | 65 |
'pay', |
66 | 66 |
'photo', |
67 | 67 |
'pre', |
68 |
+ 'sales', |
|
68 | 69 |
'server', |
69 | 70 |
'statistic', |
70 | 71 |
'website', |
@@ -402,6 +403,8 @@ KODO_CLERK_AUTH_URL = 'http://pai.ai/w/o?r=http%3A%2F%2Fkodo.xfoto.com.cn%2Fp%2F |
||
402 | 403 |
KODO_SCREEN_AUTH_URL = 'http://pai.ai/w/o?r=http%3A%2F%2Fkodo.xfoto.com.cn%2Fp%2Fscreen%2Fadmin%2Foauth%3Fbrand_id%3D{0}' |
403 | 404 |
KODO_SCREEN_LOGIN_URL = 'http://pai.ai/w/o?s=snsapi_base&r=http%3A%2F%2Fkodo.xfoto.com.cn%2Fp%2Fscreen%2Fadmin%2Flogin%3Fbrand_id%3D{0}%26token%3D{1}' |
404 | 405 |
|
406 |
+KODO_SUPERSR_AUTH_URL = 'http://pai.ai/w/o?r=http%3A%2F%2Fkodo.xfoto.com.cn%2Fp%2Fsupersr%2Foauth%3Fbrand_id%3D{0}' |
|
407 |
+ |
|
405 | 408 |
# 经纬度 |
406 | 409 |
GIS_2_ADMINISTRATIVE_DIVISION = 'http://116.196.105.215:1234/gis?auth_user=freevip&latitude={0}&longitude={1}' |
407 | 410 |
PHONE_2_ADMINISTRATIVE_DIVISION = 'https://www.baifubao.com/callback?cmd=1059&callback=phone&phone={0}' |
@@ -159,6 +159,8 @@ class ModelInfo(BaseModelMixin): |
||
159 | 159 |
|
160 | 160 |
display = models.BooleanField(_(u'display'), default=True, help_text=_(u'Display'), db_index=True) |
161 | 161 |
|
162 |
+ is_important = models.BooleanField(_(u'is_important'), default=True, help_text=_(u'是否重要型号'), db_index=True) |
|
163 |
+ |
|
162 | 164 |
class Meta: |
163 | 165 |
verbose_name = _(u'型号信息') |
164 | 166 |
verbose_name_plural = _(u'型号信息') |
@@ -221,6 +223,7 @@ class ModelInfo(BaseModelMixin): |
||
221 | 223 |
'image2_url': self.image2_url, |
222 | 224 |
'factory_yuan': self.factory_yuan, |
223 | 225 |
'integral': self.integral, |
226 |
+ 'is_important': self.is_important, |
|
224 | 227 |
} |
225 | 228 |
|
226 | 229 |
fulldata = admindata |
@@ -266,6 +269,8 @@ class DistributorInfo(BaseModelMixin): |
||
266 | 269 |
distributor_province_code = models.CharField(_(u'distributor_province_code'), max_length=6, blank=True, null=True, help_text=u'经销商所在省份编码') |
267 | 270 |
distributor_province_name = models.CharField(_(u'distributor_province_name'), max_length=3, choices=ProvinceShortModelMixin.PROVINCE_NAME_TUPLE, default=ProvinceShortModelMixin.PROVINCE_DEFAULT_NAME, blank=True, null=True, help_text=u'经销商所在省份名称') |
268 | 271 |
|
272 |
+ sr_id = models.CharField(_(u'sr_id'), max_length=32, blank=True, null=True, help_text=u'销售担当唯一标识', db_index=True) |
|
273 |
+ |
|
269 | 274 |
position = models.IntegerField(_(u'position'), default=1, help_text=u'排序') |
270 | 275 |
|
271 | 276 |
class Meta: |
@@ -293,6 +298,7 @@ class DistributorInfo(BaseModelMixin): |
||
293 | 298 |
'distributor_descr': self.distributor_descr, |
294 | 299 |
'province_code': self.distributor_province_code, |
295 | 300 |
'province_name': self.distributor_province_name, |
301 |
+ 'sr_id': self.sr_id, |
|
296 | 302 |
} |
297 | 303 |
|
298 | 304 |
|
@@ -0,0 +1,31 @@ |
||
1 |
+# -*- coding: utf-8 -*- |
|
2 |
+ |
|
3 |
+from django.conf import settings |
|
4 |
+from django_logit import logit |
|
5 |
+from json_render import json_render |
|
6 |
+ |
|
7 |
+from sales.models import SalesResponsibilityInfo |
|
8 |
+ |
|
9 |
+ |
|
10 |
+@logit |
|
11 |
+def supersr_oauthqr(request): |
|
12 |
+ brand_id = request.GET.get('brand_id', settings.KODO_DEFAULT_BRAND_ID) |
|
13 |
+ |
|
14 |
+ return json_render(request, 'page/supersr_oauth_qrcode.html', unjsondumpsdict={ |
|
15 |
+ 'qr': settings.KODO_SUPERSR_AUTH_URL.format(brand_id) |
|
16 |
+ }) |
|
17 |
+ |
|
18 |
+ |
|
19 |
+@logit |
|
20 |
+def supersr_oauth(request): |
|
21 |
+ brand_id = request.GET.get('brand_id', settings.KODO_DEFAULT_BRAND_ID) |
|
22 |
+ unionid = request.GET.get('unionid', '') |
|
23 |
+ |
|
24 |
+ SalesResponsibilityInfo.objects.update_or_create(brand_id=brand_id, unionid=unionid, defaults={ |
|
25 |
+ 'user_status': SalesResponsibilityInfo.ACTIVATED, |
|
26 |
+ 'is_auth': True, |
|
27 |
+ 'is_super': True, |
|
28 |
+ }) |
|
29 |
+ |
|
30 |
+ return json_render(request, 'page/supersr_oauth_success.html', unjsondumpsdict={ |
|
31 |
+ }) |
@@ -0,0 +1,64 @@ |
||
1 |
+{% load staticfiles %} |
|
2 |
+ |
|
3 |
+<!DOCTYPE html> |
|
4 |
+<html lang="zh-CN"> |
|
5 |
+ <head> |
|
6 |
+ <meta charset="utf-8"> |
|
7 |
+ <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> |
|
8 |
+ <meta name="format-detection" content="telephone=no,email=no,address=no"> |
|
9 |
+ <meta name="viewport" content="width=device-width,initial-scale=1.0,user-scalable=no"> |
|
10 |
+ <title>超级销售担当授权二维码</title> |
|
11 |
+ |
|
12 |
+ <link href="//res.wx.qq.com/open/libs/weui/0.4.3/weui.min.css" rel="stylesheet" type="text/css" /> |
|
13 |
+ |
|
14 |
+ <style> |
|
15 |
+ input:required:invalid { |
|
16 |
+ color: #E64340; |
|
17 |
+ } |
|
18 |
+ input:required:valid { |
|
19 |
+ color: rgb(0, 0, 0); |
|
20 |
+ } |
|
21 |
+ .hidden { |
|
22 |
+ display: none; |
|
23 |
+ } |
|
24 |
+ .qr { |
|
25 |
+ position: fixed; |
|
26 |
+ left: 50%; |
|
27 |
+ top: 50%; |
|
28 |
+ margin-left: -100px; |
|
29 |
+ margin-top: -100px; |
|
30 |
+ padding: 5px 5px 0 5px; |
|
31 |
+ border: 1px solid #000; |
|
32 |
+ border-radius: 5px; |
|
33 |
+ } |
|
34 |
+ </style> |
|
35 |
+ </head> |
|
36 |
+ <body> |
|
37 |
+ <div class="container" > |
|
38 |
+ <img id="qr_logo" class="hidden" src="{% static 'kodo/img/paiai_96_96.png' %}"> |
|
39 |
+ <div id="qr" class="qr"></div> |
|
40 |
+ </div> |
|
41 |
+ |
|
42 |
+ <script src="//cdnjs.cloudflare.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script> |
|
43 |
+ <script src="//cdnjs.cloudflare.com/ajax/libs/lrsjng.jquery-qrcode/0.14.0/jquery-qrcode.min.js"></script> |
|
44 |
+ <script> |
|
45 |
+ $("#qr").empty().qrcode({ |
|
46 |
+ render: 'canvas', |
|
47 |
+ mode: 0, |
|
48 |
+ text: '{{ qr }}' |
|
49 |
+ }); |
|
50 |
+ </script> |
|
51 |
+ <script type="text/javascript" src="//res.wx.qq.com/open/js/jweixin-1.2.0.js"></script> |
|
52 |
+ <script type="text/javascript" src="{% static 'kodo/js/jswe-0.0.4.js' %}"></script> |
|
53 |
+ <script> |
|
54 |
+ V.initWxData({ |
|
55 |
+ imgUrl: "http://pai.ai/static/kodo/img/paiai_96_96.png", |
|
56 |
+ link: 'http://pai.ai/w/o?r=http%3A%2F%2Fpai.ai%2Fp%2Floginqr', |
|
57 |
+ desc: "授权登录", |
|
58 |
+ title: "授权登录", |
|
59 |
+ timeLine: "" |
|
60 |
+ }, true); |
|
61 |
+ V.hideOptionMenu(); |
|
62 |
+ </script> |
|
63 |
+ </body> |
|
64 |
+</html> |
@@ -0,0 +1,13 @@ |
||
1 |
+<!DOCTYPE html> |
|
2 |
+<html lang="en"> |
|
3 |
+<head> |
|
4 |
+ <meta charset="UTF-8"> |
|
5 |
+ <meta name="viewport" content="width=device-width, initial-scale=1.0"> |
|
6 |
+ <meta http-equiv="X-UA-Compatible" content="ie=edge"> |
|
7 |
+ <title>授权成功</title> |
|
8 |
+</head> |
|
9 |
+<body> |
|
10 |
+ <div class="" style="text-align:center;margin-top:100px;font-size:25px;">授权成功</div> |
|
11 |
+ <div class="" style="text-align:center;font-size:15px;">您已成为超级销售担当</div> |
|
12 |
+</body> |
|
13 |
+</html> |
@@ -4,7 +4,7 @@ from django.conf.urls import url |
||
4 | 4 |
|
5 | 5 |
from account import tourguide_views |
6 | 6 |
from group import lensman_views |
7 |
-from page import info_views, oauth_views, page_views, preauth_views, sale_views, screen_views |
|
7 |
+from page import info_views, oauth_views, page_views, preauth_views, sale_views, screen_views, supersr_views |
|
8 | 8 |
|
9 | 9 |
|
10 | 10 |
urlpatterns = [ |
@@ -45,3 +45,9 @@ urlpatterns += [ |
||
45 | 45 |
url(r'^screen/admin/oauth$', screen_views.screen_admin_oauth, name='screen_admin_oauth'), |
46 | 46 |
url(r'^screen/admin/login$', screen_views.screen_admin_login, name='screen_admin_login'), |
47 | 47 |
] |
48 |
+ |
|
49 |
+# 超级销售担当 |
|
50 |
+urlpatterns += [ |
|
51 |
+ url(r'^supersr$', supersr_views.supersr_oauthqr, name='supersr_oauthqr'), # 授权二维码 |
|
52 |
+ url(r'^supersr/oauth$', supersr_views.supersr_oauth, name='supersr_oauth'), # 授权成功 |
|
53 |
+] |
@@ -0,0 +1,6 @@ |
||
1 |
+# -*- coding: utf-8 -*- |
|
2 |
+from __future__ import unicode_literals |
|
3 |
+ |
|
4 |
+from django.contrib import admin |
|
5 |
+ |
|
6 |
+# Register your models here. |
@@ -0,0 +1,8 @@ |
||
1 |
+# -*- coding: utf-8 -*- |
|
2 |
+from __future__ import unicode_literals |
|
3 |
+ |
|
4 |
+from django.apps import AppConfig |
|
5 |
+ |
|
6 |
+ |
|
7 |
+class SalesConfig(AppConfig): |
|
8 |
+ name = 'sales' |
@@ -0,0 +1,145 @@ |
||
1 |
+# -*- coding: utf-8 -*- |
|
2 |
+ |
|
3 |
+from django.db import models |
|
4 |
+from django.utils.translation import ugettext_lazy as _ |
|
5 |
+from django_models_ext import BaseModelMixin, upload_file_url, upload_path |
|
6 |
+from shortuuidfield import ShortUUIDField |
|
7 |
+ |
|
8 |
+ |
|
9 |
+class SalesResponsibilityInfo(BaseModelMixin): |
|
10 |
+ REFUSED = -1 |
|
11 |
+ UNVERIFIED = 0 |
|
12 |
+ ACTIVATED = 1 |
|
13 |
+ DISABLED = 2 |
|
14 |
+ DELETED = 3 |
|
15 |
+ ASSIGN = 10 |
|
16 |
+ |
|
17 |
+ USER_STATUS_TUPLE = ( |
|
18 |
+ (REFUSED, u'已拒绝'), |
|
19 |
+ (UNVERIFIED, u'未验证'), |
|
20 |
+ (ACTIVATED, u'已激活'), |
|
21 |
+ (DISABLED, u'已禁用'), |
|
22 |
+ (DELETED, u'已删除'), |
|
23 |
+ (ASSIGN, u'已分配'), |
|
24 |
+ ) |
|
25 |
+ |
|
26 |
+ brand_id = models.CharField(_(u'brand_id'), max_length=32, blank=True, null=True, help_text=u'品牌唯一标识', db_index=True) |
|
27 |
+ brand_name = models.CharField(_(u'brand_name'), max_length=255, blank=True, null=True, help_text=u'品牌名称') |
|
28 |
+ |
|
29 |
+ sr_id = ShortUUIDField(_(u'sr_id'), max_length=32, blank=True, null=True, help_text=u'销售担当唯一标识', db_index=True, unique=True) |
|
30 |
+ |
|
31 |
+ user_id = models.CharField(_(u'user_id'), max_length=32, blank=True, null=True, help_text=u'用户唯一标识', db_index=True) |
|
32 |
+ unionid = models.CharField(_(u'unionid'), max_length=32, blank=True, null=True, help_text=u'微信 UnionID', db_index=True) |
|
33 |
+ openid = models.CharField(_(u'openid'), max_length=32, blank=True, null=True, help_text=u'微信 OpenID', db_index=True) |
|
34 |
+ |
|
35 |
+ name = models.CharField(_(u'name'), max_length=255, blank=True, null=True, help_text=u'销售担当姓名') |
|
36 |
+ phone = models.CharField(_(u'phone'), max_length=11, blank=True, null=True, help_text=u'销售担当联系电话') |
|
37 |
+ |
|
38 |
+ user_status = models.IntegerField(_(u'user_status'), choices=USER_STATUS_TUPLE, default=UNVERIFIED, help_text=u'用户状态', db_index=True) |
|
39 |
+ refused_reason = models.TextField(_(u'refused_reason'), blank=True, null=True, help_text=u'审核拒绝原因') |
|
40 |
+ |
|
41 |
+ is_auth = models.BooleanField(_(u'is_auth'), default=False, help_text=_(u'是否已授权'), db_index=True) |
|
42 |
+ |
|
43 |
+ is_super = models.BooleanField(_(u'is_super'), default=True, help_text=_(u'是否超级销售担当'), db_index=True) |
|
44 |
+ |
|
45 |
+ class Meta: |
|
46 |
+ verbose_name = _(u'销售担当信息') |
|
47 |
+ verbose_name_plural = _(u'销售担当信息') |
|
48 |
+ |
|
49 |
+ unique_together = ( |
|
50 |
+ ('brand_id', 'sr_id'), |
|
51 |
+ ) |
|
52 |
+ |
|
53 |
+ def __unicode__(self): |
|
54 |
+ return u'{}-{}'.format(self.name, self.phone) |
|
55 |
+ |
|
56 |
+ @property |
|
57 |
+ def admindata(self): |
|
58 |
+ return { |
|
59 |
+ 'brand_id': self.brand_id, |
|
60 |
+ 'brand_name': self.brand_name, |
|
61 |
+ 'sr_id': self.sr_id, |
|
62 |
+ 'name': self.name, |
|
63 |
+ 'phone': self.phone, |
|
64 |
+ 'status': self.user_status, |
|
65 |
+ 'refused_reason': self.refused_reason, |
|
66 |
+ 'is_auth': self.is_auth, |
|
67 |
+ } |
|
68 |
+ |
|
69 |
+ data = admindata |
|
70 |
+ |
|
71 |
+ |
|
72 |
+class SalesResponsibilityInfoModelsSaleStatisticInfo(BaseModelMixin): |
|
73 |
+ brand_id = models.CharField(_(u'brand_id'), max_length=32, blank=True, null=True, help_text=u'品牌唯一标识', db_index=True) |
|
74 |
+ |
|
75 |
+ # 经销商 |
|
76 |
+ distributor_id = models.CharField(_(u'distributor_id'), max_length=32, help_text=u'经销商唯一标识', db_index=True) |
|
77 |
+ distributor_name = models.CharField(_(u'distributor_name'), max_length=255, blank=True, null=True, help_text=u'经销商名称') |
|
78 |
+ |
|
79 |
+ # 型号 |
|
80 |
+ model_id = models.CharField(_(u'model_id'), max_length=32, help_text=u'型号唯一标识', db_index=True) |
|
81 |
+ model_name = models.CharField(_(u'model_name'), max_length=255, blank=True, null=True, help_text=u'型号名称', db_index=True) |
|
82 |
+ is_important = models.BooleanField(_(u'is_important'), default=True, help_text=_(u'是否重要型号'), db_index=True) |
|
83 |
+ |
|
84 |
+ ymd = models.IntegerField(_(u'ymd'), default=0, help_text=u'年月日', db_index=True) # 例:20171208, tc.local_string(format='%Y%m%d'), 0 为全部 |
|
85 |
+ |
|
86 |
+ today_num = models.IntegerField(_(u'today_num'), default=0, help_text=u'支数(今日)') |
|
87 |
+ yesterday_num = models.IntegerField(_(u'yesterday_num'), default=0, help_text=u'支数(昨日)') |
|
88 |
+ current_month = models.IntegerField(_(u'current_month'), default=0, help_text=u'支数(本月)') |
|
89 |
+ last_month = models.IntegerField(_(u'last_month'), default=0, help_text=u'支数(上月)') |
|
90 |
+ |
|
91 |
+ position = models.IntegerField(_(u'position'), default=1, help_text=u'排序') |
|
92 |
+ |
|
93 |
+ class Meta: |
|
94 |
+ verbose_name = _(u'[销售担当]型号销量统计') |
|
95 |
+ verbose_name_plural = _(u'[销售担当]型号销量统计') |
|
96 |
+ |
|
97 |
+ def __unicode__(self): |
|
98 |
+ return unicode(self.pk) |
|
99 |
+ |
|
100 |
+ @property |
|
101 |
+ def data(self): |
|
102 |
+ return { |
|
103 |
+ 'model_id': self.model_id, |
|
104 |
+ 'model_name': self.model_name, |
|
105 |
+ 'is_important': self.is_important, |
|
106 |
+ 'today_num': self.today_num, |
|
107 |
+ 'yesterday_num': self.yesterday_num, |
|
108 |
+ 'current_month': self.current_month, |
|
109 |
+ 'last_month': self.last_month, |
|
110 |
+ } |
|
111 |
+ |
|
112 |
+ |
|
113 |
+class SuperSalesResponsibilityInfoModelsSaleStatisticInfo(BaseModelMixin): |
|
114 |
+ brand_id = models.CharField(_(u'brand_id'), max_length=32, blank=True, null=True, help_text=u'品牌唯一标识', db_index=True) |
|
115 |
+ |
|
116 |
+ # 销售担当 |
|
117 |
+ sr_id = models.CharField(_(u'sr_id'), max_length=32, help_text=u'销售担当唯一标识', db_index=True) |
|
118 |
+ sr_name = models.CharField(_(u'sr_name'), max_length=255, blank=True, null=True, help_text=u'销售担当名称') |
|
119 |
+ |
|
120 |
+ ymd = models.IntegerField(_(u'ymd'), default=0, help_text=u'年月日', db_index=True) # 例:20171208, tc.local_string(format='%Y%m%d'), 0 为全部 |
|
121 |
+ |
|
122 |
+ today_num = models.IntegerField(_(u'today_num'), default=0, help_text=u'支数(今日)') |
|
123 |
+ yesterday_num = models.IntegerField(_(u'yesterday_num'), default=0, help_text=u'支数(昨日)') |
|
124 |
+ current_month = models.IntegerField(_(u'current_month'), default=0, help_text=u'支数(本月)') |
|
125 |
+ last_month = models.IntegerField(_(u'last_month'), default=0, help_text=u'支数(上月)') |
|
126 |
+ |
|
127 |
+ position = models.IntegerField(_(u'position'), default=1, help_text=u'排序') |
|
128 |
+ |
|
129 |
+ class Meta: |
|
130 |
+ verbose_name = _(u'[超级销售担当]销售担当销量统计') |
|
131 |
+ verbose_name_plural = _(u'[超级销售担当]销售担当销量统计') |
|
132 |
+ |
|
133 |
+ def __unicode__(self): |
|
134 |
+ return unicode(self.pk) |
|
135 |
+ |
|
136 |
+ @property |
|
137 |
+ def data(self): |
|
138 |
+ return { |
|
139 |
+ 'sr_id': self.sr_id, |
|
140 |
+ 'sr_name': self.sr_name, |
|
141 |
+ 'today_num': self.today_num, |
|
142 |
+ 'yesterday_num': self.yesterday_num, |
|
143 |
+ 'current_month': self.current_month, |
|
144 |
+ 'last_month': self.last_month, |
|
145 |
+ } |
@@ -0,0 +1,6 @@ |
||
1 |
+# -*- coding: utf-8 -*- |
|
2 |
+from __future__ import unicode_literals |
|
3 |
+ |
|
4 |
+from django.test import TestCase |
|
5 |
+ |
|
6 |
+# Create your tests here. |
@@ -0,0 +1,82 @@ |
||
1 |
+# -*- coding: utf-8 -*- |
|
2 |
+ |
|
3 |
+from __future__ import division |
|
4 |
+ |
|
5 |
+from django.conf import settings |
|
6 |
+from django_logit import logit |
|
7 |
+from django_response import response |
|
8 |
+from TimeConvert import TimeConvert as tc |
|
9 |
+ |
|
10 |
+from mch.models import DistributorInfo |
|
11 |
+from sales.models import SalesResponsibilityInfo, SalesResponsibilityInfoModelsSaleStatisticInfo, SuperSalesResponsibilityInfoModelsSaleStatisticInfo |
|
12 |
+from utils.error.errno_utils import SalesResponsibilityStatusCode |
|
13 |
+ |
|
14 |
+ |
|
15 |
+@logit |
|
16 |
+def sr_submit_api(request): |
|
17 |
+ brand_id = request.POST.get('brand_id', settings.KODO_DEFAULT_BRAND_ID) |
|
18 |
+ user_id = request.POST.get('user_id', '') |
|
19 |
+ unionid = request.POST.get('unionid', '') |
|
20 |
+ openid = request.POST.get('openid', '') |
|
21 |
+ phone = request.POST.get('phone', '') |
|
22 |
+ |
|
23 |
+ if SalesResponsibilityInfo.objects.filter(brand_id=brand_id, phone=phone).exclude(user_id=user_id).exists(): |
|
24 |
+ return response(SalesResponsibilityStatusCode.SR_PHONE_ALREADY_EXISTS) |
|
25 |
+ |
|
26 |
+ SalesResponsibilityInfo.objects.update_or_create(brand_id=brand_id, user_id=user_id, defaults={ |
|
27 |
+ 'name': request.POST.get('name', ''), |
|
28 |
+ 'phone': request.POST.get('phone', ''), |
|
29 |
+ 'unionid': unionid, |
|
30 |
+ 'openid': openid, |
|
31 |
+ 'user_status': SalesResponsibilityInfo.UNVERIFIED, |
|
32 |
+ 'status': True, |
|
33 |
+ }) |
|
34 |
+ |
|
35 |
+ return response(200, 'Submit Success', u'提交成功', {}) |
|
36 |
+ |
|
37 |
+ |
|
38 |
+@logit |
|
39 |
+def sr_distributor_list(request): |
|
40 |
+ brand_id = request.POST.get('brand_id', settings.KODO_DEFAULT_BRAND_ID) |
|
41 |
+ user_id = request.POST.get('user_id', '') |
|
42 |
+ sr_id = request.POST.get('sr_id', '') |
|
43 |
+ |
|
44 |
+ # if not sr_id: |
|
45 |
+ # try: |
|
46 |
+ # sr = SalesResponsibilityInfo.objects.get(brand_id=brand_id, user_id=user_id) |
|
47 |
+ # except SalesResponsibilityInfo.DoesNotExist: |
|
48 |
+ # return response(SalesResponsibilityStatusCode.SR_NOT_FOUND) |
|
49 |
+ # sr_id = sr.sr_id |
|
50 |
+ |
|
51 |
+ distributors = DistributorInfo.objects.filter(brand_id=brand_id, sr_id=sr_id, status=True) |
|
52 |
+ distributors = [d.admindata for d in distributors] |
|
53 |
+ |
|
54 |
+ return response(data={ |
|
55 |
+ 'distributors': distributors, |
|
56 |
+ }) |
|
57 |
+ |
|
58 |
+ |
|
59 |
+@logit |
|
60 |
+def sr_distributor_tj(request): |
|
61 |
+ brand_id = request.POST.get('brand_id', settings.KODO_DEFAULT_BRAND_ID) |
|
62 |
+ distributor_id = request.POST.get('distributor_id', '') |
|
63 |
+ |
|
64 |
+ sales = SalesResponsibilityInfoModelsSaleStatisticInfo.objects.filter(brand_id=brand_id, distributor_id=distributor_id, ymd=tc.local_string(format='%Y%m%d'), status=True).order_by('-is_important') |
|
65 |
+ sales = [sale.data for sale in sales] |
|
66 |
+ |
|
67 |
+ return response(data={ |
|
68 |
+ 'sales': sales, |
|
69 |
+ }) |
|
70 |
+ |
|
71 |
+ |
|
72 |
+@logit |
|
73 |
+def supersr_sr_tj(request): |
|
74 |
+ brand_id = request.POST.get('brand_id', settings.KODO_DEFAULT_BRAND_ID) |
|
75 |
+ sr_id = request.POST.get('sr_id', '') |
|
76 |
+ |
|
77 |
+ sales = SuperSalesResponsibilityInfoModelsSaleStatisticInfo.objects.filter(brand_id=brand_id, sr_id=sr_id, ymd=tc.local_string(format='%Y%m%d'), status=True).order_by('-is_important') |
|
78 |
+ sales = [sale.data for sale in sales] |
|
79 |
+ |
|
80 |
+ return response(data={ |
|
81 |
+ 'sales': sales, |
|
82 |
+ }) |
@@ -16,6 +16,13 @@ class SaleclerkStatusCode(BaseStatusCode): |
||
16 | 16 |
DUPLICATE_SUBMIT = StatusCodeField(500199, 'Duplicate Submit', description=u'重复提交') |
17 | 17 |
|
18 | 18 |
|
19 |
+class SalesResponsibilityStatusCode(BaseStatusCode): |
|
20 |
+ """ 销售担当相关错误码 5002xx """ |
|
21 |
+ SR_NOT_FOUND = StatusCodeField(500201, 'SR Not Found', description=u'销售担当不存在') |
|
22 |
+ # 手机号 |
|
23 |
+ SR_PHONE_ALREADY_EXISTS = StatusCodeField(500205, 'SR Phone Already Exists', description=u'手机号已经存在') |
|
24 |
+ |
|
25 |
+ |
|
19 | 26 |
class ProductBrandStatusCode(BaseStatusCode): |
20 | 27 |
""" 品牌相关错误码 5010xx """ |
21 | 28 |
BRAND_NOT_FOUND = StatusCodeField(501001, 'Brand Not Found', description=u'品牌不存在') |