commit
0b6eaf08d1
@ -0,0 +1,77 @@
|
||||
package au.com.royalpay.payment.manage.billqrcode.bean;
|
||||
|
||||
import com.alibaba.fastjson.JSONObject;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
|
||||
import java.math.BigDecimal;
|
||||
|
||||
public class NewBillQrCodeBean {
|
||||
|
||||
private BigDecimal order_amount;
|
||||
|
||||
private String currency = "AUD";
|
||||
|
||||
private String remark;
|
||||
|
||||
private String cancle_time;
|
||||
|
||||
private String client_order_id;
|
||||
|
||||
public JSONObject toJson(){
|
||||
JSONObject record = new JSONObject();
|
||||
if(order_amount != null){
|
||||
record.put("order_amount",order_amount);
|
||||
}
|
||||
if(StringUtils.isNotEmpty(remark)){
|
||||
record.put("remark",remark);
|
||||
}
|
||||
if(StringUtils.isNotEmpty(client_order_id)){
|
||||
record.put("client_order_id",client_order_id);
|
||||
}
|
||||
if (StringUtils.isNotEmpty(cancle_time)) {
|
||||
record.put("cancle_time",cancle_time);
|
||||
}
|
||||
record.put("currency",currency);
|
||||
return record;
|
||||
}
|
||||
|
||||
public BigDecimal getOrder_amount() {
|
||||
return order_amount;
|
||||
}
|
||||
|
||||
public void setOrder_amount(BigDecimal order_amount) {
|
||||
this.order_amount = order_amount;
|
||||
}
|
||||
|
||||
public String getCurrency() {
|
||||
return currency;
|
||||
}
|
||||
|
||||
public void setCurrency(String currency) {
|
||||
this.currency = currency;
|
||||
}
|
||||
|
||||
public String getRemark() {
|
||||
return remark;
|
||||
}
|
||||
|
||||
public void setRemark(String remark) {
|
||||
this.remark = remark;
|
||||
}
|
||||
|
||||
public String getCancle_time() {
|
||||
return cancle_time;
|
||||
}
|
||||
|
||||
public void setCancle_time(String cancle_time) {
|
||||
this.cancle_time = cancle_time;
|
||||
}
|
||||
|
||||
public String getClient_order_id() {
|
||||
return client_order_id;
|
||||
}
|
||||
|
||||
public void setClient_order_id(String client_order_id) {
|
||||
this.client_order_id = client_order_id;
|
||||
}
|
||||
}
|
@ -0,0 +1,46 @@
|
||||
package au.com.royalpay.payment.manage.billqrcode.bean;
|
||||
|
||||
import com.alibaba.fastjson.JSONObject;
|
||||
import com.sun.tools.hat.internal.model.JavaObject;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
|
||||
/**
|
||||
* Created by yuan on 2018/5/9.
|
||||
*/
|
||||
public class QueryBillBean {
|
||||
private int limit = 10;
|
||||
private int page = 1;
|
||||
private String status;
|
||||
|
||||
public JSONObject toJson(){
|
||||
JSONObject jason = new JSONObject();
|
||||
if(StringUtils.isNotEmpty(status)){
|
||||
jason.put("status",status);
|
||||
}
|
||||
return jason;
|
||||
}
|
||||
|
||||
public int getLimit() {
|
||||
return limit;
|
||||
}
|
||||
|
||||
public void setLimit(int limit) {
|
||||
this.limit = limit;
|
||||
}
|
||||
|
||||
public int getPage() {
|
||||
return page;
|
||||
}
|
||||
|
||||
public void setPage(int page) {
|
||||
this.page = page;
|
||||
}
|
||||
|
||||
public String getStatus() {
|
||||
return status;
|
||||
}
|
||||
|
||||
public void setStatus(String status) {
|
||||
this.status = status;
|
||||
}
|
||||
}
|
@ -0,0 +1,20 @@
|
||||
package au.com.royalpay.payment.manage.billqrcode.core;
|
||||
|
||||
import au.com.royalpay.payment.manage.billqrcode.bean.NewBillQrCodeBean;
|
||||
import au.com.royalpay.payment.manage.billqrcode.bean.QueryBillBean;
|
||||
import com.alibaba.fastjson.JSONObject;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Created by yuan on 2018/5/3.
|
||||
*/
|
||||
public interface PartnerBillService {
|
||||
String saveBill(JSONObject partner, NewBillQrCodeBean newBillQrCodeBean);
|
||||
|
||||
List<JSONObject> listBills(JSONObject partner, QueryBillBean queryBillBean);
|
||||
|
||||
void dailyCheckDirectedBillCode();
|
||||
|
||||
void disableBills(JSONObject partner,String bill_code_id);
|
||||
}
|
@ -0,0 +1,128 @@
|
||||
package au.com.royalpay.payment.manage.billqrcode.core.impl;
|
||||
|
||||
import au.com.royalpay.payment.manage.billqrcode.bean.NewBillQrCodeBean;
|
||||
import au.com.royalpay.payment.manage.billqrcode.bean.QueryBillBean;
|
||||
import au.com.royalpay.payment.manage.billqrcode.core.PartnerBillService;
|
||||
import au.com.royalpay.payment.manage.mappers.billqrcode.DirectedBillCodeMapper;
|
||||
import au.com.royalpay.payment.manage.mappers.system.ClientMapper;
|
||||
import au.com.royalpay.payment.manage.merchants.core.ClientManager;
|
||||
import au.com.royalpay.payment.tools.env.PlatformEnvironment;
|
||||
import au.com.royalpay.payment.tools.exceptions.BadRequestException;
|
||||
import au.com.royalpay.payment.tools.exceptions.NotFoundException;
|
||||
import au.com.royalpay.payment.tools.utils.QRCodeUtils;
|
||||
import com.alibaba.fastjson.JSONObject;
|
||||
import com.github.miemiedev.mybatis.paginator.domain.Order;
|
||||
import com.github.miemiedev.mybatis.paginator.domain.PageBounds;
|
||||
import javafx.scene.chart.PieChart;
|
||||
import org.apache.commons.lang3.RandomStringUtils;
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.apache.commons.lang3.time.DateFormatUtils;
|
||||
import org.apache.commons.lang3.time.DateUtils;
|
||||
import org.slf4j.LoggerFactory;
|
||||
import org.springframework.stereotype.Service;
|
||||
import org.springframework.transaction.annotation.Transactional;
|
||||
|
||||
import javax.annotation.Resource;
|
||||
import java.util.Calendar;
|
||||
import java.util.Date;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Created by yuan on 2018/5/3.
|
||||
*/
|
||||
@Service
|
||||
public class PartnerBillServiceImpl implements PartnerBillService {
|
||||
private org.slf4j.Logger logger = LoggerFactory.getLogger(getClass());
|
||||
@Resource
|
||||
private ClientMapper clientMapper;
|
||||
|
||||
@Resource
|
||||
private ClientManager clientManager;
|
||||
|
||||
@Resource
|
||||
private DirectedBillCodeMapper directedBillCodeMapper;
|
||||
|
||||
|
||||
@Override
|
||||
@Transactional
|
||||
public String saveBill(JSONObject partner, NewBillQrCodeBean newBillQrCodeBean) {
|
||||
JSONObject client = clientMapper.findClientByMoniker(partner.getString("client_moniker"));
|
||||
if (client == null) {
|
||||
throw new NotFoundException("client info not found");
|
||||
}
|
||||
if(StringUtils.isNotEmpty(newBillQrCodeBean.getClient_order_id())){
|
||||
JSONObject bill = directedBillCodeMapper.findOne(newBillQrCodeBean.getClient_order_id());
|
||||
if(bill != null){
|
||||
if(bill.getIntValue("status") == 2){
|
||||
throw new BadRequestException("该订单已关闭,请确认");
|
||||
}
|
||||
if(bill.getIntValue("status") == 3){
|
||||
throw new BadRequestException("该订单已完成,请确认");
|
||||
}
|
||||
if(bill.getIntValue("status") == 1){
|
||||
if(bill.getBigDecimal("order_amount") == newBillQrCodeBean.getOrder_amount()){
|
||||
return QRCodeUtils.qrcodeImageCode(bill.getString("code_url"), 250, false);
|
||||
}
|
||||
throw new BadRequestException("已有相同订单号正在被支付,详情请查看订单列表");
|
||||
}
|
||||
}
|
||||
}
|
||||
JSONObject record = newBillQrCodeBean.toJson();
|
||||
if(record.getString("client_order_id") == null){
|
||||
record.put("client_order_id","Bill_" + DateFormatUtils.format(new Date(), "yyyyMMddHHmmss") + "_" + RandomStringUtils.random(5, true, true).toUpperCase());
|
||||
}
|
||||
record.put("client_id",partner.getIntValue("client_id"));
|
||||
record.put("create_time", new Date());
|
||||
record.put("status",1);
|
||||
directedBillCodeMapper.save(record);
|
||||
String code_url = getQRCodeImg(record);
|
||||
record.put("code_url",code_url);
|
||||
directedBillCodeMapper.update(record);
|
||||
return QRCodeUtils.qrcodeImageCode(code_url, 250, false);
|
||||
}
|
||||
|
||||
private String getQRCodeImg(JSONObject bill){
|
||||
return PlatformEnvironment.getEnv().concatUrl("api/v1.0/share_code/business/bills/"+bill.getString("bill_code_id"));
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<JSONObject> listBills(JSONObject partner, QueryBillBean queryBillBean) {
|
||||
int client_id = partner.getInteger("client_id");
|
||||
JSONObject client = clientManager.getClientInfo(client_id);
|
||||
if (client == null) {
|
||||
throw new NotFoundException("client info not found");
|
||||
}
|
||||
JSONObject params = queryBillBean.toJson();
|
||||
params.put("client_id",client_id);
|
||||
List<JSONObject> bills = directedBillCodeMapper.findByClientId(params,new PageBounds(queryBillBean.getPage(),queryBillBean.getLimit(), Order.formString("create_time.desc")));
|
||||
bills.stream().filter(t->t.getString("code_url")!= null).forEach(t->t.put("code_url",QRCodeUtils.qrcodeImageCode(t.getString("code_url"), 250, false)));
|
||||
return bills;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void disableBills(JSONObject partner, String bill_code_id) {
|
||||
int client_id = partner.getInteger("client_id");
|
||||
JSONObject client = clientManager.getClientInfo(client_id);
|
||||
if (client == null) {
|
||||
throw new NotFoundException("client info not found");
|
||||
}
|
||||
JSONObject bill = directedBillCodeMapper.findOneByBillCodeId(bill_code_id);
|
||||
if(bill.getIntValue("status")==1){
|
||||
if(bill.getString("order_id") != null){
|
||||
bill.put("status",2);
|
||||
directedBillCodeMapper.update(bill);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void dailyCheckDirectedBillCode() {
|
||||
List<JSONObject> listAllBill = directedBillCodeMapper.listAllBills();
|
||||
for (JSONObject bill: listAllBill) {
|
||||
if(DateUtils.addDays(bill.getDate("cancle_time"),1).before(new Date())){
|
||||
bill.put("status",2);
|
||||
directedBillCodeMapper.update(bill);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,37 @@
|
||||
package au.com.royalpay.payment.manage.billqrcode.web;
|
||||
|
||||
import au.com.royalpay.payment.manage.billqrcode.bean.NewBillQrCodeBean;
|
||||
import au.com.royalpay.payment.manage.billqrcode.bean.QueryBillBean;
|
||||
import au.com.royalpay.payment.manage.billqrcode.core.PartnerBillService;
|
||||
import au.com.royalpay.payment.manage.permission.manager.PartnerMapping;
|
||||
import au.com.royalpay.payment.tools.CommonConsts;
|
||||
import com.alibaba.fastjson.JSONObject;
|
||||
import org.springframework.web.bind.annotation.*;
|
||||
|
||||
import javax.annotation.Resource;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Created by yuan on 2018/5/3.
|
||||
*/
|
||||
@RestController
|
||||
@RequestMapping("/partner/qrcode")
|
||||
public class PartnerBillController {
|
||||
|
||||
@Resource
|
||||
private PartnerBillService partnerBillService;
|
||||
|
||||
@PartnerMapping(value = "/bills", method = RequestMethod.POST)
|
||||
public String addBill(@ModelAttribute(CommonConsts.PARTNER_STATUS) JSONObject partner, @RequestBody NewBillQrCodeBean newBillQrCodeBean) {
|
||||
return partnerBillService.saveBill(partner, newBillQrCodeBean);
|
||||
}
|
||||
|
||||
@PartnerMapping(value = "", method = RequestMethod.GET)
|
||||
public List<JSONObject> listBills(@ModelAttribute(CommonConsts.PARTNER_STATUS) JSONObject partner, QueryBillBean queryBillBean) {
|
||||
return partnerBillService.listBills(partner,queryBillBean);
|
||||
}
|
||||
@PartnerMapping(value = "/{bill_code_id}", method = RequestMethod.DELETE)
|
||||
public void disableBills(@ModelAttribute(CommonConsts.PARTNER_STATUS) JSONObject partner,@PathVariable String bill_code_id) {
|
||||
partnerBillService.disableBills(partner,bill_code_id);
|
||||
}
|
||||
}
|
@ -0,0 +1,33 @@
|
||||
package au.com.royalpay.payment.manage.mappers.billqrcode;
|
||||
|
||||
import cn.yixblog.support.mybatis.autosql.annotations.AdvanceSelect;
|
||||
import cn.yixblog.support.mybatis.autosql.annotations.AutoMapper;
|
||||
import cn.yixblog.support.mybatis.autosql.annotations.AutoSql;
|
||||
import cn.yixblog.support.mybatis.autosql.annotations.SqlType;
|
||||
import com.alibaba.fastjson.JSONObject;
|
||||
import com.github.miemiedev.mybatis.paginator.domain.PageBounds;
|
||||
import org.apache.ibatis.annotations.Param;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
@AutoMapper(tablename = "pmt_directed_bill_code", pkName = "bill_code_id")
|
||||
public interface DirectedBillCodeMapper {
|
||||
@AutoSql(type = SqlType.INSERT)
|
||||
int save(JSONObject record);
|
||||
|
||||
@AutoSql(type = SqlType.SELECT)
|
||||
JSONObject findOne(@Param("client_order_id") String client_order_id);
|
||||
|
||||
@AutoSql(type = SqlType.SELECT)
|
||||
JSONObject findOneByBillCodeId(@Param("bill_code_id") String bill_code_id);
|
||||
|
||||
@AutoSql(type = SqlType.UPDATE)
|
||||
int update(JSONObject record);
|
||||
|
||||
@AutoSql(type = SqlType.SELECT)
|
||||
List<JSONObject> findByClientId(JSONObject params, PageBounds pageBounds);
|
||||
|
||||
@AutoSql(type = SqlType.SELECT)
|
||||
@AdvanceSelect(addonWhereClause = "status=1")
|
||||
List<JSONObject> listAllBills();
|
||||
}
|
@ -0,0 +1,23 @@
|
||||
package au.com.royalpay.payment.manage.task;
|
||||
|
||||
import au.com.royalpay.payment.manage.billqrcode.core.PartnerBillService;
|
||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
|
||||
import org.springframework.scheduling.annotation.Scheduled;
|
||||
import org.springframework.stereotype.Component;
|
||||
|
||||
import javax.annotation.Resource;
|
||||
|
||||
/**
|
||||
* Created by yuan on 2018/5/4.
|
||||
*/
|
||||
@Component
|
||||
@ConditionalOnProperty(value = "app.run-tasks", havingValue = "true")
|
||||
public class DirectedBillCodeStatusDailyCheck {
|
||||
@Resource
|
||||
private PartnerBillService partnerBillService;
|
||||
|
||||
@Scheduled(cron = "0 0 6 * * ?")
|
||||
public void statusDailyCheck(){
|
||||
partnerBillService.dailyCheckDirectedBillCode();
|
||||
}
|
||||
}
|
@ -0,0 +1,151 @@
|
||||
<style>
|
||||
.delete {
|
||||
text-decoration-line: line-through;
|
||||
}
|
||||
</style>
|
||||
<div ui-view>
|
||||
<section class="content-header">
|
||||
<h1>Bill QR Code</h1>
|
||||
<ol class="breadcrumb">
|
||||
<li>
|
||||
<i class="fa fa-sitemap"></i> Bill QR Code
|
||||
</li>
|
||||
<li class="active">Bill QR Code Management</li>
|
||||
</ol>
|
||||
</section>
|
||||
|
||||
<div class="content">
|
||||
<div class="row">
|
||||
<div class="col-sm-12">
|
||||
<div class="box-solid">
|
||||
<div class="box box-warning">
|
||||
<div class="box-header">
|
||||
<div class="form-horizontal col-sm-8">
|
||||
<div class="form-group">
|
||||
<label class="control-label col-xs-4 col-sm-3">* Order Amount</label>
|
||||
<div class="col-xs-8 col-sm-6">
|
||||
<input type="number" step="0.01" name="order_amount" class="form-control"
|
||||
ng-model="new_bill.order_amount" required
|
||||
onkeyup="this.value=this.value.replace(/^(\-)*(\d+)\.(\d\d).*$/,'$1$2.$3')">
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label class="control-label col-xs-4 col-sm-3">* Expire</label>
|
||||
<div class="col-xs-8 col-sm-6">
|
||||
<input class="form-control" ng-model="new_bill.cancle_time"
|
||||
uib-datepicker-popup size="10" is-open="ctrl.dateInput"
|
||||
ng-click="ctrl.dateInput=true"
|
||||
datepicker-options="{minDate:today}" name="cancle_time" required>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label class="control-label col-xs-4 col-sm-3">Client Order Id</label>
|
||||
<div class="col-xs-8 col-sm-6">
|
||||
<input class="form-control" ng-model="new_bill.client_order_id">
|
||||
<p class="small text-info">Client Order Id 为空时,系统自动生成</p>
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label class="control-label col-xs-4 col-sm-3">Remark</label>
|
||||
<div class="col-xs-8 col-sm-6">
|
||||
<textarea class="form-control" ng-model="new_bill.remark"></textarea>
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label class="control-label col-xs-4 col-sm-3"></label>
|
||||
<div class="col-sm-6">
|
||||
<button class="btn btn-success" ng-click="generateBill()">generate
|
||||
</button>
|
||||
<button class="btn btn-primary" ng-click="clearBill()">clear</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-4" style="text-align: center" ng-if="code_url">
|
||||
<a class="thumbnail" download ng-href="{{code_url}}" uib-tooltip="Download">
|
||||
<img ng-src="{{code_url}}">
|
||||
</a>
|
||||
<p>
|
||||
<a ng-href="{{code_url}}" download><i class="fa fa-download"></i> Download Bill QR
|
||||
Code Image</a>
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="box">
|
||||
<div class="box-header">
|
||||
<h3 class="box-title">Bill List
|
||||
<a role="button" ng-class="{'bg-primary':params.status==null}"
|
||||
ng-click="params.status=null;loadBills(1)">All</a> |
|
||||
<a role="button" ng-class="{'bg-primary':params.status==3}"
|
||||
ng-click="params.status=3;loadBills(1)">Success</a> |
|
||||
<a role="button" ng-class="{'bg-primary':params.status==2}"
|
||||
ng-click="params.status=2;loadBills(1)">Disabled</a>
|
||||
|
||||
</h3>
|
||||
</div>
|
||||
<div class="box-body table-responsive">
|
||||
<table class="table table-striped">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Price</th>
|
||||
<th>Client Order Id</th>
|
||||
<th>Create time</th>
|
||||
<th>Expire time</th>
|
||||
<th>Remark</th>
|
||||
<th>Order Status</th>
|
||||
<th>Bill Status</th>
|
||||
<th>Payer</th>
|
||||
<th>Operation</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr ng-repeat="bill in bills"
|
||||
ng-class="{'delete':bill.status==2}">
|
||||
<td ng-bind="bill.order_amount"></td>
|
||||
<th ng-bind="bill.client_order_id"></th>
|
||||
<td ng-bind="bill.create_time"></td>
|
||||
<td ng-bind="bill.cancle_time"></td>
|
||||
<td ng-bind="bill.remark"></td>
|
||||
<th ng-bind="bill.order_status|tradeStatus"></th>
|
||||
<th ng-bind="bill.status|billStatus"></th>
|
||||
<td ng-bind="bill.nickname"></td>
|
||||
<td>
|
||||
<a ng-if="bill.order_id" class="text-primary" role="button" title="Detail"
|
||||
ng-click="showTradeDetail(bill)">
|
||||
<i class="fa fa-search"></i>
|
||||
</a>
|
||||
<a ng-if="bill.code_url&&bill.status==1" ng-href="{{bill.code_url}}" download><i
|
||||
class="fa fa-download"
|
||||
uib-tooltip="Download Bill QR Code Image"></i></a>
|
||||
<a ng-if="!bill.order_id&&bill.status==1"
|
||||
class="text-bold text-danger" role="button" ng-click="disableBill(bill)">Disable</a>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
<div class="box-footer" ng-if="bills.length">
|
||||
<uib-pagination class="pagination"
|
||||
total-items="pagination.totalCount"
|
||||
boundary-links="true"
|
||||
ng-model="pagination.page"
|
||||
items-per-page="pagination.limit"
|
||||
max-size="10"
|
||||
ng-change="loadBills(1)"
|
||||
previous-text="‹"
|
||||
next-text="›"
|
||||
first-text="«"
|
||||
last-text="»"></uib-pagination>
|
||||
<div class="row">
|
||||
<div class="col-xs-12">Total Records:{{pagination.totalCount}};Total
|
||||
Pages:{{pagination.totalPages}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
Loading…
Reference in new issue