s-num lines-num-old">
+ font-weight: bold;
+ color: #020001;
+ }
+ .course_time {
+ font-size: 12px;
+ color: #999;
+ }
+ .course_cover>img {
+ width: 80%;
+ border-radius: 5px;
+ }
+ </style>
+ </head>
+ <body>
+ <div class="container" >
+ {% for course in courses %}
+ <div class="course_wrapper" data-courseid="{{ course.course_id }}">
+ <div class="course_name">{{ course.course_name }}</div>
+ <div class="course_time">{{ course.course_time }}分钟</div>
+ <div class="course_cover"><img src="{{ course.course_cover_url }}"></div>
+ </div>
+ {% endfor %}
+ </div>
+
+ <script src="//cdn.bootcss.com/zepto/1.1.6/zepto.min.js"></script>
+ <script>
+ $(function() {
+ $('.course_wrapper').click(function () {
+ window.location.href = '{{ domain }}/page/course/info?course_id=' + $(this).attr('data-courseid') + '&{{ params|safe }}';
+ })
+ });
+ </script>
+ <script type="text/javascript" src="//res.wx.qq.com/open/js/jweixin-1.2.0.js"></script>
+ <script type="text/javascript" src="{% static 'course/js/jswe-0.0.1.js' %}"></script>
+ <script>
+ V.initWxData({
+ imgUrl: 'http://pai.ai/static/pai2/img/paiai_96_96.png',
+ link: 'http://api.pai.ai/wx_oauth2?redirect_url=http://tamron.xfoto.com.cn/page/clerk',
+ desc: '店员授权',
+ title: '店员授权',
+ timeLine: ''
+ }, true);
+{# V.hideOptionMenu();#}
+ </script>
+ </body>
+</html>
@@ -0,0 +1,4 @@ |
||
| 1 |
+from django.test import TestCase |
|
| 2 |
+ |
|
| 3 |
+ |
|
| 4 |
+# Create your tests here. |
@@ -0,0 +1,12 @@ |
||
| 1 |
+# -*- coding: utf-8 -*- |
|
| 2 |
+ |
|
| 3 |
+from django.conf.urls import url |
|
| 4 |
+ |
|
| 5 |
+from page import code_views, info_views, list_views |
|
| 6 |
+ |
|
| 7 |
+ |
|
| 8 |
+urlpatterns = [ |
|
| 9 |
+ url(r'^course/code$', code_views.course_code, name='course_code'), |
|
| 10 |
+ url(r'^course/list$', list_views.course_list, name='course_list'), |
|
| 11 |
+ url(r'^course/info$', info_views.course_info, name='course_info'), |
|
| 12 |
+] |
@@ -0,0 +1,4 @@ |
||
| 1 |
+from django.shortcuts import render |
|
| 2 |
+ |
|
| 3 |
+ |
|
| 4 |
+# Create your views here. |
@@ -0,0 +1,9 @@ |
||
| 1 |
+#!/bin/bash |
|
| 2 |
+ |
|
| 3 |
+# Ignoring autogenerated files |
|
| 4 |
+# -- Migration directories |
|
| 5 |
+# Ignoring error codes |
|
| 6 |
+# -- E128 continuation line under-indented for visual indent |
|
| 7 |
+# -- E501 line too long |
|
| 8 |
+ |
|
| 9 |
+pep8 --exclude=migrations --ignore=E128,E501 . |
@@ -0,0 +1,40 @@ |
||
| 1 |
+-e git+https://github.com/Brightcells/django-q.git#egg=django-q |
|
| 2 |
+CodeConvert==2.0.4 |
|
| 3 |
+Django==1.11.3 |
|
| 4 |
+MySQL-python==1.2.5 |
|
| 5 |
+Pillow==3.4.2 |
|
| 6 |
+StatusCode==1.0.0 |
|
| 7 |
+TimeConvert==1.4.1 |
|
| 8 |
+cryptography==2.0.3 |
|
| 9 |
+django-curtail-uuid==1.0.0 |
|
| 10 |
+django-detect==1.0.5 |
|
| 11 |
+django-file-md5==1.0.1 |
|
| 12 |
+django-ip==1.0.1 |
|
| 13 |
+django-json-response==1.1.5 |
|
| 14 |
+django-logit==1.0.6 |
|
| 15 |
+django-multidomain==1.1.4 |
|
| 16 |
+django-paginator2==1.0.3 |
|
| 17 |
+django-rlog==1.0.7 |
|
| 18 |
+django-shortuuidfield==0.1.3 |
|
| 19 |
+django-six==1.0.2 |
|
| 20 |
+django-uniapi==1.0.0 |
|
| 21 |
+django-we==1.0.14 |
|
| 22 |
+djangorestframework==3.6.3 |
|
| 23 |
+furl==1.0.1 |
|
| 24 |
+hiredis==0.2.0 |
|
| 25 |
+isoweek==1.3.3 |
|
| 26 |
+jsonfield==2.0.2 |
|
| 27 |
+mock==2.0.0 |
|
| 28 |
+pep8==1.7.0 |
|
| 29 |
+pysnippets==1.0.4 |
|
| 30 |
+pywe-miniapp==1.0.0 |
|
| 31 |
+pywe-oauth==1.0.5 |
|
| 32 |
+pywe-response==1.0.1 |
|
| 33 |
+qiniu==7.1.5 |
|
| 34 |
+redis==2.10.6 |
|
| 35 |
+redis-extensions==1.1.1 |
|
| 36 |
+requests==2.18.4 |
|
| 37 |
+rlog==0.2 |
|
| 38 |
+shortuuid==0.5.0 |
|
| 39 |
+uWSGI==2.0.15 |
|
| 40 |
+versions==0.10.0 |
@@ -0,0 +1,51 @@ |
||
| 1 |
+# -*- coding: utf-8 -*- |
|
| 2 |
+ |
|
| 3 |
+from StatusCode import BaseStatusCode, StatusCodeField |
|
| 4 |
+ |
|
| 5 |
+ |
|
| 6 |
+class ProfileStatusCode(BaseStatusCode): |
|
| 7 |
+ """ 用户相关错误码 4000xx """ |
|
| 8 |
+ PROFILE_NOT_FOUND = StatusCodeField(400001, 'Profile Not Found', description=u'用户不存在') |
|
| 9 |
+ |
|
| 10 |
+ |
|
| 11 |
+class CourseCodeStatusCode(BaseStatusCode): |
|
| 12 |
+ """ 课程兑换码相关错误码 4001xx """ |
|
| 13 |
+ COURSE_CODE_NOT_FOUND = StatusCodeField(400101, 'Course Code Not Found', description=u'课程兑换码不存在') |
|
| 14 |
+ COURSE_CODE_HAS_EXCHANGED = StatusCodeField(400102, 'Course Code Has Exchanged', description=u'课程兑换码已兑换') |
|
| 15 |
+ |
|
| 16 |
+ |
|
| 17 |
+class CourseStatusCode(BaseStatusCode): |
|
| 18 |
+ """ 课程相关错误码 4002xx """ |
|
| 19 |
+ COURSE_NOT_FOUND = StatusCodeField(400201, 'Course Not Found', description=u'课程不存在') |
|
| 20 |
+ |
|
| 21 |
+ |
|
| 22 |
+class OrderStatusCode(BaseStatusCode): |
|
| 23 |
+ """ 订单/支付相关错误码 4040xx """ |
|
| 24 |
+ WX_UNIFIED_ORDER_FAIL = StatusCodeField(404000, 'WX Unified Order Fail', description=u'微信统一下单失败') |
|
| 25 |
+ WX_ORDER_NOT_FOUND = StatusCodeField(404001, 'WX Order Not Found', description=u'订单不存在') |
|
| 26 |
+ WX_ORDER_NOT_PAY = StatusCodeField(404002, 'WX Order Not Pay', description=u'订单未支付') |
|
| 27 |
+ WX_ORDER_PAYING = StatusCodeField(404003, 'WX Order Paying', description=u'订单支付中') |
|
| 28 |
+ WX_ORDER_PAY_FAIL = StatusCodeField(404009, 'WX Order Pay Fail', description=u'微信支付失败') |
|
| 29 |
+ SIGN_CHECK_FAIL = StatusCodeField(404010, 'Sign Check Fail', description=u'签名校验失败') |
|
| 30 |
+ FEE_CHECK_FAIL = StatusCodeField(404011, 'FEE Check Fail', description=u'金额校验失败') |
|
| 31 |
+ NO_DETAIL_PERMISSION = StatusCodeField(404015, 'No Detail Permission', description=u'无详情权限') |
|
| 32 |
+ WX_ORDER_PAID_ALREADY_EXISTS = StatusCodeField(404020, 'WX Order Paid Already Exists', description=u'照片已购买') |
|
| 33 |
+ |
|
| 34 |
+ |
|
| 35 |
+class PayStatusCode(BaseStatusCode): |
|
| 36 |
+ """ 支付相关错误码 4041xx """ |
|
| 37 |
+ |
|
| 38 |
+ |
|
| 39 |
+class WithdrawStatusCode(BaseStatusCode): |
|
| 40 |
+ """ 提现相关错误码 4042xx """ |
|
| 41 |
+ BALANCE_NOT_ENOUGH = StatusCodeField(404200, 'Balance Not Enough', description=u'提现金额不足') |
|
| 42 |
+ |
|
| 43 |
+ |
|
| 44 |
+class MessageStatusCode(BaseStatusCode): |
|
| 45 |
+ """ 消息相关错误码 4090xx """ |
|
| 46 |
+ MESSAGE_NOT_FOUND = StatusCodeField(409001, 'Message Not Found', description=u'消息不存在') |
|
| 47 |
+ |
|
| 48 |
+ |
|
| 49 |
+class TokenStatusCode(BaseStatusCode): |
|
| 50 |
+ """ 票据相关错误码 4090xx """ |
|
| 51 |
+ TOKEN_NOT_FOUND = StatusCodeField(409901, 'Token Not Found', description=u'票据不存在') |
@@ -0,0 +1,18 @@ |
||
| 1 |
+# -*- coding: utf-8 -*- |
|
| 2 |
+ |
|
| 3 |
+from django.http import JsonResponse |
|
| 4 |
+from StatusCode import StatusCodeField |
|
| 5 |
+ |
|
| 6 |
+ |
|
| 7 |
+def response_data(status_code=200, message=None, description=None, data={}, **kwargs):
|
|
| 8 |
+ return dict({
|
|
| 9 |
+ 'status': status_code, |
|
| 10 |
+ 'message': message, |
|
| 11 |
+ 'description': description, |
|
| 12 |
+ 'data': data, |
|
| 13 |
+ }, **kwargs) |
|
| 14 |
+ |
|
| 15 |
+ |
|
| 16 |
+def response(status_code=200, message=None, description=None, data={}, **kwargs):
|
|
| 17 |
+ message, description = (message or status_code.message, description or status_code.description) if isinstance(status_code, StatusCodeField) else (message, description) |
|
| 18 |
+ return JsonResponse(response_data(status_code, message, description, data, **kwargs), safe=False) |
@@ -0,0 +1,6 @@ |
||
| 1 |
+# -*- coding: utf-8 -*- |
|
| 2 |
+ |
|
| 3 |
+from django.conf import settings |
|
| 4 |
+ |
|
| 5 |
+ |
|
| 6 |
+r = settings.REDIS_CACHE |
@@ -0,0 +1,68 @@ |
||
| 1 |
+# -*- coding: utf-8 -*- |
|
| 2 |
+ |
|
| 3 |
+# 唯一标识相关 |
|
| 4 |
+UUID_LIST = 'uuid:list' # List,唯一标识列表 |
|
| 5 |
+ |
|
| 6 |
+# 用户相关 |
|
| 7 |
+PROFILE_INFO = 'profile:info:%s' # STRING,用户信息,user_id |
|
| 8 |
+ |
|
| 9 |
+# 导游相关 |
|
| 10 |
+TOUR_GUIDE_GROUP_GEO_INFO = 'tour:guide:group:geo:info:%s' # ZSET,旅游团地理位置信息,group_id |
|
| 11 |
+TOUR_GUIDE_GROUP_GEO_SUBMIT_DT = 'tour:guide:group:geo:submit:dt:%s' # ZSET,旅游团地理位置最后上传时间,group_id |
|
| 12 |
+TOUR_GUIDE_GROUP_CUR_SESSION = 'tour:guide:group:cur:session:%s' # STRING,旅游团当前Session,group_id,导游设置集合时间的时候更新 |
|
| 13 |
+TOUR_GUIDE_GROUP_CUR_GATHER_INFO = 'tour:guide:group:cur:gather:info:%s' # STRING,旅游团当前Session,group_id,导游设置集合时间的时候更新 |
|
| 14 |
+TOUR_GUIDE_GROUP_USER_GEO_LIST = 'tour:guide:group:user:geo:list:%s:%s:%s' # LIST,旅游团当前用户地理位置列表,group_id、session_id、user_id |
|
| 15 |
+ |
|
| 16 |
+TOUR_GUIDE_GROUP_USER_OWN = 'tour:guide:group:user:own:%s' # STRING,导游当前拥有的旅行团,user_id,导游创建旅行团的时候更新 |
|
| 17 |
+TOUR_GUIDE_GROUP_USER_BELONG = 'tour:guide:group:user:belong:%s' # STRING,用户当前所属旅行团,user_id,用户加入旅行团的时候更新 |
|
| 18 |
+ |
|
| 19 |
+# 群组相关 |
|
| 20 |
+GROUP_INFO = 'group:info:%s' # STRING,群组信息,group_id |
|
| 21 |
+ |
|
| 22 |
+# 群组用户相关 |
|
| 23 |
+GROUP_USERS_INFO = 'group:users:info:%s' # STRING,群组用户信息,group_id |
|
| 24 |
+GROUP_USERS_KV_INFO = 'group:users:kv:info:%s' # STRING,群组用户信息,group_id |
|
| 25 |
+GROUP_USERS_APPLYING_SET = 'group:users:applying:set:%s' # SET,群组用户申请集合,group_id |
|
| 26 |
+GROUP_USERS_PASSED_SET = 'group:users:passed:set:%s' # SET,群组用户通过集合,group_id |
|
| 27 |
+GROUP_USERS_REFUSED_SET = 'group:users:refused:set:%s' # SET,群组用户拒绝集合,group_id |
|
| 28 |
+GROUP_USERS_DELETED_SET = 'group:users:deleted:set:%s' # SET,群组用户移除集合,group_id |
|
| 29 |
+GROUP_USERS_QUIT_SET = 'group:users:quit:set:%s' # SET,群组用户退出集合,group_id |
|
| 30 |
+ |
|
| 31 |
+# 群组照片相关 |
|
| 32 |
+GROUP_PHOTO_DATA = 'group:photo:data:%s' # STRING,群组数据记录,group_id |
|
| 33 |
+GROUP_PHOTO_THUMB_UP = 'group:photo:thumb:up:%s:%s' # STRING,群组照片用户点赞记录,photo_id、user_id |
|
| 34 |
+GROUP_PHOTO_COMMENT_LIST = 'group:photo:comment:list:%s' # STRING,群组照片用户评论列表,photo_id |
|
| 35 |
+GROUP_PHOTO_THUMB_UP_LIST = 'group:photo:thumb:up:list:%s' # STRING,群组照片用户点赞列表,photo_id |
|
| 36 |
+GROUP_PHOTO_WATCHER_SET = 'group:photo:watcher:set:%s' # SET,群组照片用户关注集合,photo_id,关注即评论点赞 |
|
| 37 |
+GROUP_LAST_PHOTO_PK = 'group:last:photo:pk:%s' # STRING,群组最后一张照片PK,group_id |
|
| 38 |
+ |
|
| 39 |
+# 摄影师照片相关 |
|
| 40 |
+LENSMAN_PHOTO_ORDER_RECORD = 'lensman:photo:order:record:%s:%s' # STRING,摄影师照片购买记录,photo_id、user_id |
|
| 41 |
+ |
|
| 42 |
+# 摄影师简报相关 |
|
| 43 |
+# 收入 |
|
| 44 |
+TOTAL_INCOME = 'total:income:%s:%s' # STRING,总收入,user_id、photo_type |
|
| 45 |
+WEEK_INCOME = 'week:income:%s:%s:%s' # STRING,周收入,user_id、photo_type、Week.thisweek().isoformat() |
|
| 46 |
+TODAY_INCOME = 'today:income:%s:%s:%s' # STRING,日收入,user_id、photo_type、tc.local_string(format='%Y%m%d') |
|
| 47 |
+# 上传 |
|
| 48 |
+TODAY_UPLOAD_PHOTO_AMOUNT = 'today:upload:photo:amount:%s:%s' # STRING,日上传照片数量,user_id、tc.local_string(format='%Y%m%d') |
|
| 49 |
+# 售出 |
|
| 50 |
+WEEK_SOLD = 'week:sold:%s:%s:%s' # STRING,周售出,user_id、photo_type、Week.thisweek().isoformat() |
|
| 51 |
+ |
|
| 52 |
+# 摄影师定价相关 |
|
| 53 |
+LENSMAN_PHOTO_PRICE_FIXED = 'lensman:photo:price:fixed:%s' # STRING,摄影师照片定价(单位:分),user_id |
|
| 54 |
+ |
|
| 55 |
+# 系统消息相关 |
|
| 56 |
+SYSTEM_MESSAGE_READ_INFO = 'system:message:read:info:%s' # STRING,系统消息读取信息,user_id |
|
| 57 |
+SYSTEM_MESSAGE_DELETED_INFO = 'system:message:deleted:info:%s' # STRING,系统消息删除信息,user_id |
|
| 58 |
+ |
|
| 59 |
+# 游客入口相关 |
|
| 60 |
+GUEST_ENTRANCE_CONTROL_INFO = 'guest:entrance:control:info:%s' # STRING,游客入口控制信息,src |
|
| 61 |
+ |
|
| 62 |
+# APP 相关 |
|
| 63 |
+LATEST_APP_INFO = 'latest:app:info:%s' # STRING,最新 APP 信息,src |
|
| 64 |
+APP_SETTINGS_INFO = 'app:settings:info:%s:%s:%s' # STRING,APP 设置信息,platform、channel、version |
|
| 65 |
+APP_PATCH_INFO = 'app:patch:info:%s:%s:%s' # STRING,APP 补丁信息,platform、version、src |
|
| 66 |
+ |
|
| 67 |
+# BOX 相关 |
|
| 68 |
+BOX_PROGRAM_VERSION_INFO = 'box:program:version:info' # STRING,BOX 程序版本信息 |
@@ -0,0 +1,7 @@ |
||
| 1 |
+# -*- coding: utf-8 -*- |
|
| 2 |
+ |
|
| 3 |
+from django.conf import settings |
|
| 4 |
+ |
|
| 5 |
+ |
|
| 6 |
+def upload_file_url(file_path): |
|
| 7 |
+ return file_path and ('{}{}'.format(settings.DOMAIN, file_path.url)) or ''
|