Springboot集成支付宝沙箱支付(退款功能)

x33g5p2x  于2022-07-10 转载在 Spring  
字(25.5k)|赞(0)|评价(0)|浏览(609)

包括:
支付宝沙箱 支付 + 异步通知 + 退款功能

正式版本的sdk

通用版本SDK文档:https://opendocs.alipay.com/open/02np94

  1. <dependency>
  2. <groupId>com.alipay.sdk</groupId>
  3. <artifactId>alipay-sdk-java</artifactId>
  4. <version>4.22.110.ALL</version>
  5. </dependency>

AlipayConfig.java 只需要获取配置alipay的参数即可

  1. import lombok.Data;
  2. import org.springframework.boot.context.properties.ConfigurationProperties;
  3. import org.springframework.stereotype.Component;
  4. @Data
  5. @Component
  6. @ConfigurationProperties(prefix = "alipay")
  7. public class AliPayConfig {
  8. private String appId;
  9. private String appPrivateKey;
  10. private String alipayPublicKey;
  11. private String notifyUrl;
  12. }

参数更新:

  1. import lombok.Data;
  2. @Data
  3. public class AliPay {
  4. private String traceNo;
  5. private double totalAmount;
  6. private String subject;
  7. private String alipayTraceNo;
  8. }

完整的支付宝通用版本支付代码:

  1. import cn.hutool.core.date.DateUnit;
  2. import cn.hutool.core.date.DateUtil;
  3. import cn.hutool.json.JSONObject;
  4. import com.alipay.api.AlipayApiException;
  5. import com.alipay.api.AlipayClient;
  6. import com.alipay.api.DefaultAlipayClient;
  7. import com.alipay.api.internal.util.AlipaySignature;
  8. import com.alipay.api.request.AlipayTradePagePayRequest;
  9. import com.alipay.api.request.AlipayTradeRefundRequest;
  10. import com.alipay.api.response.AlipayTradeRefundResponse;
  11. import com.example.springboot.common.Result;
  12. import com.example.springboot.config.AliPayConfig;
  13. import com.example.springboot.controller.dto.AliPay;
  14. import com.example.springboot.entity.Orders;
  15. import com.example.springboot.mapper.OrdersMapper;
  16. import org.springframework.web.bind.annotation.GetMapping;
  17. import org.springframework.web.bind.annotation.PostMapping;
  18. import org.springframework.web.bind.annotation.RequestMapping;
  19. import org.springframework.web.bind.annotation.RestController;
  20. import javax.annotation.Resource;
  21. import javax.servlet.http.HttpServletRequest;
  22. import javax.servlet.http.HttpServletResponse;
  23. import java.util.HashMap;
  24. import java.util.Map;
  25. // xjlugv6874@sandbox.com
  26. // 9428521.24 - 30 = 9428491.24 + 30 = 9428521.24
  27. @RestController
  28. @RequestMapping("/alipay")
  29. public class AliPayController {
  30. private static final String GATEWAY_URL = "https://openapi.alipaydev.com/gateway.do";
  31. private static final String FORMAT = "JSON";
  32. private static final String CHARSET = "UTF-8";
  33. //签名方式
  34. private static final String SIGN_TYPE = "RSA2";
  35. @Resource
  36. private AliPayConfig aliPayConfig;
  37. @Resource
  38. private OrdersMapper ordersMapper;
  39. @GetMapping("/pay") // &subject=xxx&traceNo=xxx&totalAmount=xxx
  40. public void pay(AliPay aliPay, HttpServletResponse httpResponse) throws Exception {
  41. // 1. 创建Client,通用SDK提供的Client,负责调用支付宝的API
  42. AlipayClient alipayClient = new DefaultAlipayClient(GATEWAY_URL, aliPayConfig.getAppId(),
  43. aliPayConfig.getAppPrivateKey(), FORMAT, CHARSET, aliPayConfig.getAlipayPublicKey(), SIGN_TYPE);
  44. // 2. 创建 Request并设置Request参数
  45. AlipayTradePagePayRequest request = new AlipayTradePagePayRequest(); // 发送请求的 Request类
  46. request.setNotifyUrl(aliPayConfig.getNotifyUrl());
  47. JSONObject bizContent = new JSONObject();
  48. bizContent.set("out_trade_no", aliPay.getTraceNo()); // 我们自己生成的订单编号
  49. bizContent.set("total_amount", aliPay.getTotalAmount()); // 订单的总金额
  50. bizContent.set("subject", aliPay.getSubject()); // 支付的名称
  51. bizContent.set("product_code", "FAST_INSTANT_TRADE_PAY"); // 固定配置
  52. request.setBizContent(bizContent.toString());
  53. // 执行请求,拿到响应的结果,返回给浏览器
  54. String form = "";
  55. try {
  56. form = alipayClient.pageExecute(request).getBody(); // 调用SDK生成表单
  57. } catch (AlipayApiException e) {
  58. e.printStackTrace();
  59. }
  60. httpResponse.setContentType("text/html;charset=" + CHARSET);
  61. httpResponse.getWriter().write(form);// 直接将完整的表单html输出到页面
  62. httpResponse.getWriter().flush();
  63. httpResponse.getWriter().close();
  64. }
  65. @PostMapping("/notify") // 注意这里必须是POST接口
  66. public String payNotify(HttpServletRequest request) throws Exception {
  67. if (request.getParameter("trade_status").equals("TRADE_SUCCESS")) {
  68. System.out.println("=========支付宝异步回调========");
  69. Map<String, String> params = new HashMap<>();
  70. Map<String, String[]> requestParams = request.getParameterMap();
  71. for (String name : requestParams.keySet()) {
  72. params.put(name, request.getParameter(name));
  73. // System.out.println(name + " = " + request.getParameter(name));
  74. }
  75. String tradeNo = params.get("out_trade_no");
  76. String gmtPayment = params.get("gmt_payment");
  77. String alipayTradeNo = params.get("trade_no");
  78. String sign = params.get("sign");
  79. String content = AlipaySignature.getSignCheckContentV1(params);
  80. boolean checkSignature = AlipaySignature.rsa256CheckContent(content, sign, aliPayConfig.getAlipayPublicKey(), "UTF-8"); // 验证签名
  81. // 支付宝验签
  82. if (checkSignature) {
  83. // 验签通过
  84. System.out.println("交易名称: " + params.get("subject"));
  85. System.out.println("交易状态: " + params.get("trade_status"));
  86. System.out.println("支付宝交易凭证号: " + params.get("trade_no"));
  87. System.out.println("商户订单号: " + params.get("out_trade_no"));
  88. System.out.println("交易金额: " + params.get("total_amount"));
  89. System.out.println("买家在支付宝唯一id: " + params.get("buyer_id"));
  90. System.out.println("买家付款时间: " + params.get("gmt_payment"));
  91. System.out.println("买家付款金额: " + params.get("buyer_pay_amount"));
  92. // 更新订单未已支付
  93. ordersMapper.updateState(tradeNo, "已支付", gmtPayment, alipayTradeNo);
  94. }
  95. }
  96. return "success";
  97. }
  98. @GetMapping("/return")
  99. public Result returnPay(AliPay aliPay) throws AlipayApiException {
  100. // 7天无理由退款
  101. String now = DateUtil.now();
  102. Orders orders = ordersMapper.getByNo(aliPay.getTraceNo());
  103. if (orders != null) {
  104. // hutool工具类,判断时间间隔
  105. long between = DateUtil.between(DateUtil.parseDateTime(orders.getPaymentTime()), DateUtil.parseDateTime(now), DateUnit.DAY);
  106. if (between > 7) {
  107. return Result.error("-1", "该订单已超过7天,不支持退款");
  108. }
  109. }
  110. // 1. 创建Client,通用SDK提供的Client,负责调用支付宝的API
  111. AlipayClient alipayClient = new DefaultAlipayClient(GATEWAY_URL,
  112. aliPayConfig.getAppId(), aliPayConfig.getAppPrivateKey(), FORMAT, CHARSET,
  113. aliPayConfig.getAlipayPublicKey(), SIGN_TYPE);
  114. // 2. 创建 Request,设置参数
  115. AlipayTradeRefundRequest request = new AlipayTradeRefundRequest();
  116. JSONObject bizContent = new JSONObject();
  117. bizContent.set("trade_no", aliPay.getAlipayTraceNo()); // 支付宝回调的订单流水号
  118. bizContent.set("refund_amount", aliPay.getTotalAmount()); // 订单的总金额
  119. bizContent.set("out_request_no", aliPay.getTraceNo()); // 我的订单编号
  120. // 返回参数选项,按需传入
  121. //JSONArray queryOptions = new JSONArray();
  122. //queryOptions.add("refund_detail_item_list");
  123. //bizContent.put("query_options", queryOptions);
  124. request.setBizContent(bizContent.toString());
  125. // 3. 执行请求
  126. AlipayTradeRefundResponse response = alipayClient.execute(request);
  127. if (response.isSuccess()) { // 退款成功,isSuccess 为true
  128. System.out.println("调用成功");
  129. // 4. 更新数据库状态
  130. ordersMapper.updatePayState(aliPay.getTraceNo(), "已退款", now);
  131. return Result.success();
  132. } else { // 退款失败,isSuccess 为false
  133. System.out.println(response.getBody());
  134. return Result.error(response.getCode(), response.getBody());
  135. }
  136. }
  137. }

至此,后端集成完毕。

注意事项

如果你在支付的过程中出现了这样的一个界面,不用慌,直接使用 隐私窗口打开即可。

参考SQL

商品表:

  1. CREATE TABLE `goods` (
  2. `id` int(11) NOT NULL AUTO_INCREMENT,
  3. `name` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '名称',
  4. `price` decimal(10,2) DEFAULT NULL COMMENT '单价',
  5. `store` int(10) DEFAULT NULL COMMENT '库存',
  6. `unit` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '单位',
  7. `img` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '封面',
  8. PRIMARY KEY (`id`)
  9. ) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

订单表

  1. CREATE TABLE `orders` (
  2. `id` int(11) NOT NULL AUTO_INCREMENT,
  3. `no` varchar(50) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '订单编号',
  4. `name` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '名称',
  5. `time` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '订单时间',
  6. `state` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT '待支付' COMMENT '支付状态',
  7. `total` decimal(10,2) DEFAULT NULL COMMENT '订单总价',
  8. `user_id` int(11) DEFAULT NULL COMMENT '用户id',
  9. `payment_time` varchar(50) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '支付时间',
  10. `alipay_no` varchar(50) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '支付宝交易号',
  11. `return_time` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT '退款时间',
  12. PRIMARY KEY (`id`)
  13. ) ENGINE=InnoDB AUTO_INCREMENT=12 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

购买下订单的模拟接口:

  1. @PostMapping("/{goodsId}/{num}")
  2. public Result createOrder(@PathVariable Integer goodsId, @PathVariable Integer num) {
  3. Goods goods = goodsService.getById(goodsId);
  4. if (goods == null) {
  5. throw new ServiceException("-1", "未找到商品");
  6. }
  7. User currentUser = TokenUtils.getCurrentUser();
  8. Integer userId = currentUser.getId();
  9. Orders orders = new Orders();
  10. orders.setName(goods.getName());
  11. orders.setNo(DateUtil.format(new Date(), "yyyyMMdd") + System.currentTimeMillis());
  12. orders.setTime(DateUtil.now());
  13. orders.setUserId(userId);
  14. orders.setTotal(goods.getPrice().multiply(BigDecimal.valueOf(num)));
  15. ordersService.save(orders);
  16. return Result.success();
  17. }

附赠前端调用代码:
Goods.vue

  1. <template>
  2. <div>
  3. <div style="margin: 10px 0">
  4. <el-input style="width: 200px" placeholder="请输入名称" suffix-icon="el-icon-search" v-model="name"></el-input>
  5. <!-- <el-input style="width: 200px" placeholder="请输入" suffix-icon="el-icon-message" class="ml-5" v-model="email"></el-input>-->
  6. <!-- <el-input style="width: 200px" placeholder="请输入" suffix-icon="el-icon-position" class="ml-5" v-model="address"></el-input>-->
  7. <el-button class="ml-5" type="primary" @click="load">搜索</el-button>
  8. <el-button type="warning" @click="reset">重置</el-button>
  9. </div>
  10. <div style="margin: 10px 0">
  11. <el-button type="primary" @click="handleAdd">新增 <i class="el-icon-circle-plus-outline"></i></el-button>
  12. <el-popconfirm
  13. class="ml-5"
  14. confirm-button-text='确定'
  15. cancel-button-text='我再想想'
  16. icon="el-icon-info"
  17. icon-color="red"
  18. title="您确定批量删除这些数据吗?"
  19. @confirm="delBatch"
  20. >
  21. <el-button type="danger" slot="reference">批量删除 <i class="el-icon-remove-outline"></i></el-button>
  22. </el-popconfirm>
  23. <!-- <el-upload action="http://localhost:9090/goods/import" :show-file-list="false" accept="xlsx" :on-success="handleExcelImportSuccess" style="display: inline-block">
  24. <el-button type="primary" class="ml-5">导入 <i class="el-icon-bottom"></i></el-button>
  25. </el-upload>
  26. <el-button type="primary" @click="exp" class="ml-5">导出 <i class="el-icon-top"></i></el-button> -->
  27. </div>
  28. <el-table :data="tableData" border stripe :header-cell-class-name="'headerBg'" @selection-change="handleSelectionChange">
  29. <el-table-column type="selection" width="55"></el-table-column>
  30. <el-table-column prop="id" label="ID" width="80" sortable></el-table-column>
  31. <el-table-column prop="name" label="名称"></el-table-column>
  32. <el-table-column prop="price" label="单价"></el-table-column>
  33. <el-table-column prop="store" label="库存"></el-table-column>
  34. <el-table-column prop="unit" label="单位"></el-table-column>
  35. <el-table-column label="图片"><template slot-scope="scope"><el-image style="width: 100px; height: 100px" :src="scope.row.img" :preview-src-list="[scope.row.img]"></el-image></template></el-table-column>
  36. <el-table-column label="购买" width="180" align="center">
  37. <template v-slot="scope">
  38. <el-button type="primary" @click="buy(scope.row.id)">购 买</el-button>
  39. </template>
  40. </el-table-column>
  41. <el-table-column label="操作" width="180" align="center">
  42. <template slot-scope="scope">
  43. <el-button type="success" @click="handleEdit(scope.row)">编辑 <i class="el-icon-edit"></i></el-button>
  44. <el-popconfirm
  45. class="ml-5"
  46. confirm-button-text='确定'
  47. cancel-button-text='我再想想'
  48. icon="el-icon-info"
  49. icon-color="red"
  50. title="您确定删除吗?"
  51. @confirm="del(scope.row.id)"
  52. >
  53. <el-button type="danger" slot="reference">删除 <i class="el-icon-remove-outline"></i></el-button>
  54. </el-popconfirm>
  55. </template>
  56. </el-table-column>
  57. </el-table>
  58. <div style="padding: 10px 0">
  59. <el-pagination
  60. @size-change="handleSizeChange"
  61. @current-change="handleCurrentChange"
  62. :current-page="pageNum"
  63. :page-sizes="[2, 5, 10, 20]"
  64. :page-size="pageSize"
  65. layout="total, sizes, prev, pager, next, jumper"
  66. :total="total">
  67. </el-pagination>
  68. </div>
  69. <el-dialog title="信息" :visible.sync="dialogFormVisible" width="30%" :close-on-click-modal="false">
  70. <el-form label-width="100px" size="small" style="width: 90%">
  71. <el-form-item label="名称">
  72. <el-input v-model="form.name" autocomplete="off"></el-input>
  73. </el-form-item>
  74. <el-form-item label="单价">
  75. <el-input v-model="form.price" autocomplete="off"></el-input>
  76. </el-form-item>
  77. <el-form-item label="库存">
  78. <el-input v-model="form.store" autocomplete="off"></el-input>
  79. </el-form-item>
  80. <el-form-item label="单位">
  81. <el-input v-model="form.unit" autocomplete="off"></el-input>
  82. </el-form-item>
  83. <el-form-item label="封面">
  84. <el-upload action="http://localhost:9090/file/upload" ref="img" :on-success="handleImgUploadSuccess">
  85. <el-button size="small" type="primary">点击上传</el-button>
  86. </el-upload>
  87. </el-form-item>
  88. </el-form>
  89. <div slot="footer" class="dialog-footer">
  90. <el-button @click="dialogFormVisible = false">取 消</el-button>
  91. <el-button type="primary" @click="save">确 定</el-button>
  92. </div>
  93. </el-dialog>
  94. </div>
  95. </template>
  96. <script>
  97. export default {
  98. name: "Goods",
  99. data() {
  100. return {
  101. tableData: [],
  102. total: 0,
  103. pageNum: 1,
  104. pageSize: 10,
  105. name: "",
  106. form: {},
  107. dialogFormVisible: false,
  108. multipleSelection: [],
  109. user: localStorage.getItem("user") ? JSON.parse(localStorage.getItem("user")) : {}
  110. }
  111. },
  112. created() {
  113. this.load()
  114. },
  115. methods: {
  116. buy(goodsId) {
  117. this.request.post("/orders/" + goodsId + "/" + 1).then(res => {
  118. if (res.code === '200') {
  119. this.$message.success("购买成功,请前往订单页面支付")
  120. } else {
  121. this.$message.error(res.msg)
  122. }
  123. })
  124. },
  125. load() {
  126. this.request.get("/goods/page", {
  127. params: {
  128. pageNum: this.pageNum,
  129. pageSize: this.pageSize,
  130. name: this.name,
  131. }
  132. }).then(res => {
  133. this.tableData = res.data.records
  134. this.total = res.data.total
  135. })
  136. },
  137. save() {
  138. this.request.post("/goods", this.form).then(res => {
  139. if (res.code === '200') {
  140. this.$message.success("保存成功")
  141. this.dialogFormVisible = false
  142. this.load()
  143. } else {
  144. this.$message.error("保存失败")
  145. }
  146. })
  147. },
  148. handleAdd() {
  149. this.dialogFormVisible = true
  150. this.form = {}
  151. this.$nextTick(() => {
  152. if(this.$refs.img) {
  153. this.$refs.img.clearFiles();
  154. }
  155. if(this.$refs.file) {
  156. this.$refs.file.clearFiles();
  157. }
  158. })
  159. },
  160. handleEdit(row) {
  161. this.form = JSON.parse(JSON.stringify(row))
  162. this.dialogFormVisible = true
  163. this.$nextTick(() => {
  164. if(this.$refs.img) {
  165. this.$refs.img.clearFiles();
  166. }
  167. if(this.$refs.file) {
  168. this.$refs.file.clearFiles();
  169. }
  170. })
  171. },
  172. del(id) {
  173. this.request.delete("/goods/" + id).then(res => {
  174. if (res.code === '200') {
  175. this.$message.success("删除成功")
  176. this.load()
  177. } else {
  178. this.$message.error("删除失败")
  179. }
  180. })
  181. },
  182. handleSelectionChange(val) {
  183. console.log(val)
  184. this.multipleSelection = val
  185. },
  186. delBatch() {
  187. if (!this.multipleSelection.length) {
  188. this.$message.error("请选择需要删除的数据")
  189. return
  190. }
  191. let ids = this.multipleSelection.map(v => v.id) // [{}, {}, {}] => [1,2,3]
  192. this.request.post("/goods/del/batch", ids).then(res => {
  193. if (res.code === '200') {
  194. this.$message.success("批量删除成功")
  195. this.load()
  196. } else {
  197. this.$message.error("批量删除失败")
  198. }
  199. })
  200. },
  201. reset() {
  202. this.name = ""
  203. this.load()
  204. },
  205. handleSizeChange(pageSize) {
  206. console.log(pageSize)
  207. this.pageSize = pageSize
  208. this.load()
  209. },
  210. handleCurrentChange(pageNum) {
  211. console.log(pageNum)
  212. this.pageNum = pageNum
  213. this.load()
  214. },
  215. handleFileUploadSuccess(res) {
  216. this.form.file = res
  217. },
  218. handleImgUploadSuccess(res) {
  219. this.form.img = res
  220. },
  221. download(url) {
  222. window.open(url)
  223. },
  224. exp() {
  225. window.open("http://localhost:9090/goods/export")
  226. },
  227. handleExcelImportSuccess() {
  228. this.$message.success("导入成功")
  229. this.load()
  230. }
  231. }
  232. }
  233. </script>
  234. <style>
  235. .headerBg {
  236. background: #eee!important;
  237. }
  238. </style>

Orders.vue

  1. <template>
  2. <div>
  3. <div style="margin: 10px 0">
  4. <el-input style="width: 200px" placeholder="请输入名称" suffix-icon="el-icon-search" v-model="name"></el-input>
  5. <!-- <el-input style="width: 200px" placeholder="请输入" suffix-icon="el-icon-message" class="ml-5" v-model="email"></el-input>-->
  6. <!-- <el-input style="width: 200px" placeholder="请输入" suffix-icon="el-icon-position" class="ml-5" v-model="address"></el-input>-->
  7. <el-button class="ml-5" type="primary" @click="load">搜索</el-button>
  8. <el-button type="warning" @click="reset">重置</el-button>
  9. </div>
  10. <div style="margin: 10px 0">
  11. <el-button type="primary" @click="handleAdd">新增 <i class="el-icon-circle-plus-outline"></i></el-button>
  12. <el-popconfirm
  13. class="ml-5"
  14. confirm-button-text='确定'
  15. cancel-button-text='我再想想'
  16. icon="el-icon-info"
  17. icon-color="red"
  18. title="您确定批量删除这些数据吗?"
  19. @confirm="delBatch"
  20. >
  21. <el-button type="danger" slot="reference">批量删除 <i class="el-icon-remove-outline"></i></el-button>
  22. </el-popconfirm>
  23. <!-- <el-upload action="http://localhost:9090/orders/import" :show-file-list="false" accept="xlsx" :on-success="handleExcelImportSuccess" style="display: inline-block">
  24. <el-button type="primary" class="ml-5">导入 <i class="el-icon-bottom"></i></el-button>
  25. </el-upload>
  26. <el-button type="primary" @click="exp" class="ml-5">导出 <i class="el-icon-top"></i></el-button> -->
  27. </div>
  28. <el-table :data="tableData" border stripe :header-cell-class-name="'headerBg'" @selection-change="handleSelectionChange">
  29. <el-table-column type="selection" width="55"></el-table-column>
  30. <el-table-column prop="id" label="ID" width="80" sortable></el-table-column>
  31. <el-table-column prop="no" label="订单编号"></el-table-column>
  32. <el-table-column prop="name" label="名称"></el-table-column>
  33. <el-table-column prop="time" label="订单时间"></el-table-column>
  34. <el-table-column prop="state" label="支付状态"></el-table-column>
  35. <el-table-column prop="total" label="订单总价"></el-table-column>
  36. <el-table-column prop="paymentTime" label="支付时间"></el-table-column>
  37. <el-table-column prop="alipayNo" label="支付宝宝流水号"></el-table-column>
  38. <el-table-column prop="returnTime" label="退款时间"></el-table-column>
  39. <el-table-column label="支付" width="180" align="center">
  40. <template v-slot="scope">
  41. <el-button type="primary" @click="pay(scope.row)" :disabled="scope.row.state !== '待支付'">支 付</el-button>
  42. <el-button type="danger" @click="returnPay(scope.row)" :disabled="scope.row.state !== '已支付'">退款</el-button>
  43. </template>
  44. </el-table-column>
  45. <el-table-column label="操作" width="180" align="center">
  46. <template slot-scope="scope">
  47. <el-button type="success" @click="handleEdit(scope.row)">编辑 <i class="el-icon-edit"></i></el-button>
  48. <el-popconfirm
  49. class="ml-5"
  50. confirm-button-text='确定'
  51. cancel-button-text='我再想想'
  52. icon="el-icon-info"
  53. icon-color="red"
  54. title="您确定删除吗?"
  55. @confirm="del(scope.row.id)"
  56. >
  57. <el-button type="danger" slot="reference">删除 <i class="el-icon-remove-outline"></i></el-button>
  58. </el-popconfirm>
  59. </template>
  60. </el-table-column>
  61. </el-table>
  62. <div style="padding: 10px 0">
  63. <el-pagination
  64. @size-change="handleSizeChange"
  65. @current-change="handleCurrentChange"
  66. :current-page="pageNum"
  67. :page-sizes="[2, 5, 10, 20]"
  68. :page-size="pageSize"
  69. layout="total, sizes, prev, pager, next, jumper"
  70. :total="total">
  71. </el-pagination>
  72. </div>
  73. <el-dialog title="信息" :visible.sync="dialogFormVisible" width="30%" :close-on-click-modal="false">
  74. <el-form label-width="100px" size="small" style="width: 90%">
  75. <el-form-item label="订单编号">
  76. <el-input v-model="form.no" autocomplete="off"></el-input>
  77. </el-form-item>
  78. <el-form-item label="名称">
  79. <el-input v-model="form.name" autocomplete="off"></el-input>
  80. </el-form-item>
  81. <el-form-item label="订单时间">
  82. <el-date-picker v-model="form.time" type="datetime" value-format="yyyy-MM-dd HH:mm:ss" placeholder="选择日期时间"></el-date-picker>
  83. </el-form-item>
  84. <el-form-item label="支付状态">
  85. <el-input v-model="form.state" autocomplete="off"></el-input>
  86. </el-form-item>
  87. <el-form-item label="订单总价">
  88. <el-input v-model="form.total" autocomplete="off"></el-input>
  89. </el-form-item>
  90. </el-form>
  91. <div slot="footer" class="dialog-footer">
  92. <el-button @click="dialogFormVisible = false">取 消</el-button>
  93. <el-button type="primary" @click="save">确 定</el-button>
  94. </div>
  95. </el-dialog>
  96. </div>
  97. </template>
  98. <script>
  99. export default {
  100. name: "Orders",
  101. data() {
  102. return {
  103. tableData: [],
  104. total: 0,
  105. pageNum: 1,
  106. pageSize: 10,
  107. name: "",
  108. form: {},
  109. dialogFormVisible: false,
  110. multipleSelection: [],
  111. user: localStorage.getItem("user") ? JSON.parse(localStorage.getItem("user")) : {}
  112. }
  113. },
  114. created() {
  115. this.load()
  116. },
  117. methods: {
  118. pay(row) {
  119. const url = `http://localhost:9090/alipay/pay?subject=${row.name}&traceNo=${row.no}&totalAmount=${row.total}`
  120. window.open(url); // 得到一个url,这个url就是支付宝支付的界面url, 新窗口打开这个url就可以了
  121. },
  122. returnPay(row) {
  123. const url = `http://localhost:9090/alipay/return?totalAmount=${row.total}&alipayTraceNo=${row.alipayNo}&traceNo=${row.no}`
  124. this.request.get(url).then(res => {
  125. if(res.code === '200') {
  126. this.$message.success("退款成功")
  127. } else {
  128. this.$message.error("退款失败")
  129. }
  130. this.load()
  131. })
  132. },
  133. load() {
  134. this.request.get("/orders/page", {
  135. params: {
  136. pageNum: this.pageNum,
  137. pageSize: this.pageSize,
  138. name: this.name,
  139. }
  140. }).then(res => {
  141. this.tableData = res.data.records
  142. this.total = res.data.total
  143. })
  144. },
  145. save() {
  146. this.request.post("/orders", this.form).then(res => {
  147. if (res.code === '200') {
  148. this.$message.success("保存成功")
  149. this.dialogFormVisible = false
  150. this.load()
  151. } else {
  152. this.$message.error("保存失败")
  153. }
  154. })
  155. },
  156. handleAdd() {
  157. this.dialogFormVisible = true
  158. this.form = {}
  159. this.$nextTick(() => {
  160. if(this.$refs.img) {
  161. this.$refs.img.clearFiles();
  162. }
  163. if(this.$refs.file) {
  164. this.$refs.file.clearFiles();
  165. }
  166. })
  167. },
  168. handleEdit(row) {
  169. this.form = JSON.parse(JSON.stringify(row))
  170. this.dialogFormVisible = true
  171. this.$nextTick(() => {
  172. if(this.$refs.img) {
  173. this.$refs.img.clearFiles();
  174. }
  175. if(this.$refs.file) {
  176. this.$refs.file.clearFiles();
  177. }
  178. })
  179. },
  180. del(id) {
  181. this.request.delete("/orders/" + id).then(res => {
  182. if (res.code === '200') {
  183. this.$message.success("删除成功")
  184. this.load()
  185. } else {
  186. this.$message.error("删除失败")
  187. }
  188. })
  189. },
  190. handleSelectionChange(val) {
  191. console.log(val)
  192. this.multipleSelection = val
  193. },
  194. delBatch() {
  195. if (!this.multipleSelection.length) {
  196. this.$message.error("请选择需要删除的数据")
  197. return
  198. }
  199. let ids = this.multipleSelection.map(v => v.id) // [{}, {}, {}] => [1,2,3]
  200. this.request.post("/orders/del/batch", ids).then(res => {
  201. if (res.code === '200') {
  202. this.$message.success("批量删除成功")
  203. this.load()
  204. } else {
  205. this.$message.error("批量删除失败")
  206. }
  207. })
  208. },
  209. reset() {
  210. this.name = ""
  211. this.load()
  212. },
  213. handleSizeChange(pageSize) {
  214. console.log(pageSize)
  215. this.pageSize = pageSize
  216. this.load()
  217. },
  218. handleCurrentChange(pageNum) {
  219. console.log(pageNum)
  220. this.pageNum = pageNum
  221. this.load()
  222. },
  223. handleFileUploadSuccess(res) {
  224. this.form.file = res
  225. },
  226. handleImgUploadSuccess(res) {
  227. this.form.img = res
  228. },
  229. download(url) {
  230. window.open(url)
  231. },
  232. exp() {
  233. window.open("http://localhost:9090/orders/export")
  234. },
  235. handleExcelImportSuccess() {
  236. this.$message.success("导入成功")
  237. this.load()
  238. }
  239. }
  240. }
  241. </script>
  242. <style>
  243. .headerBg {
  244. background: #eee!important;
  245. }
  246. </style>

相关文章