import.html 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. {% extends 'base.html' %}
  2. {% block title %}导入交易记录 - 期货数据管理系统{% endblock %}
  3. {% block content %}
  4. <div class="d-flex justify-content-between align-items-center mb-4">
  5. <h2>导入交易记录</h2>
  6. <a href="{{ url_for('transaction.index') }}" class="btn btn-secondary">返回列表</a>
  7. </div>
  8. <div class="card">
  9. <div class="card-body">
  10. <div class="row mb-4">
  11. <div class="col-md-12">
  12. <p>请按照以下步骤操作:</p>
  13. <ol>
  14. <li>下载导入模板</li>
  15. <li>按照模板格式填写数据</li>
  16. <li>上传Excel文件</li>
  17. </ol>
  18. <div class="alert alert-info">
  19. <strong>提示:</strong> 必填字段为交易ID、合约代码、合约名称、多空仓位、成交价格、成交手数。多空仓位取值:0-开多,1-平多,2-开空,3-平空。
  20. </div>
  21. <div class="alert alert-warning">
  22. <strong>重要说明:</strong>
  23. <ol>
  24. <li>每个<strong>交易ID</strong>最多只能出现两次,一次开仓一次平仓</li>
  25. <li>具有相同交易ID的记录必须组成有效的开平仓配对:
  26. <ul>
  27. <li>开多(0)必须与平多(1)配对</li>
  28. <li>开空(2)必须与平空(3)配对</li>
  29. </ul>
  30. </li>
  31. <li>如果导入失败,请尝试<a href="#" onclick="clearCache()">清除浏览器缓存</a>或使用新的浏览器窗口重试</li>
  32. </ol>
  33. </div>
  34. </div>
  35. </div>
  36. <div class="row mb-4">
  37. <div class="col-md-6">
  38. <button id="download-template" class="btn btn-primary">
  39. <i class="fas fa-download"></i> 下载导入模板
  40. </button>
  41. </div>
  42. </div>
  43. <form id="import-form" enctype="multipart/form-data">
  44. <div class="row mb-4">
  45. <div class="col-md-6">
  46. <div class="form-group">
  47. <label for="file">选择Excel文件</label>
  48. <input type="file" class="form-control" id="file" name="file" accept=".xlsx" required>
  49. </div>
  50. </div>
  51. </div>
  52. <div class="form-group">
  53. <button type="submit" class="btn btn-success">
  54. <i class="fas fa-upload"></i> 导入数据
  55. </button>
  56. </div>
  57. </form>
  58. <div id="result" class="mt-4" style="display: none;">
  59. </div>
  60. </div>
  61. </div>
  62. <script>
  63. // 清除浏览器缓存的函数
  64. function clearCache() {
  65. // 添加一个随机参数来强制刷新页面
  66. const timestamp = new Date().getTime();
  67. window.location.href = window.location.href.split('?')[0] + '?cache_buster=' + timestamp;
  68. }
  69. // 添加切换行数据显示/隐藏的函数
  70. function toggleRowData(btn) {
  71. const rowData = btn.nextElementSibling;
  72. if (rowData.style.display === 'none') {
  73. rowData.style.display = 'block';
  74. btn.textContent = '隐藏行数据';
  75. } else {
  76. rowData.style.display = 'none';
  77. btn.textContent = '显示行数据';
  78. }
  79. }
  80. // 切换元素显示/隐藏的通用函数
  81. function toggleElement(id) {
  82. const element = document.getElementById(id);
  83. if (element) {
  84. if (element.style.display === 'none') {
  85. element.style.display = 'block';
  86. } else {
  87. element.style.display = 'none';
  88. }
  89. }
  90. }
  91. document.addEventListener('DOMContentLoaded', function() {
  92. const form = document.getElementById('import-form');
  93. const resultDiv = document.getElementById('result');
  94. const downloadBtn = document.getElementById('download-template');
  95. // 处理下载模板按钮点击
  96. downloadBtn.addEventListener('click', function(e) {
  97. // 禁用按钮防止重复点击
  98. downloadBtn.disabled = true;
  99. // 创建一个隐藏的iframe来处理下载
  100. const iframe = document.createElement('iframe');
  101. iframe.style.display = 'none';
  102. iframe.src = "{{ url_for('transaction.get_template') }}";
  103. document.body.appendChild(iframe);
  104. // 3秒后重新启用按钮
  105. setTimeout(() => {
  106. downloadBtn.disabled = false;
  107. document.body.removeChild(iframe);
  108. }, 3000);
  109. });
  110. // 处理表单提交(导入文件)
  111. form.addEventListener('submit', function(e) {
  112. e.preventDefault();
  113. const formData = new FormData(form);
  114. // 显示加载状态
  115. resultDiv.innerHTML = '<div class="alert alert-info">正在导入数据,请稍候...</div>';
  116. resultDiv.style.display = 'block';
  117. // 添加随机参数避免缓存
  118. const cacheBuster = new Date().getTime();
  119. fetch(`/transaction/api/import?cache_buster=${cacheBuster}`, {
  120. method: 'POST',
  121. body: formData
  122. })
  123. .then(response => response.json())
  124. .then(data => {
  125. if (data.code === 0) {
  126. // 成功导入
  127. let html = `<div class="alert alert-success">${data.msg}</div>`;
  128. if (data.data.error_count > 0) {
  129. html += '<div class="alert alert-warning">';
  130. html += '<strong>导入过程中出现以下错误:</strong>';
  131. html += '<button class="btn btn-sm btn-link float-right" id="toggle-errors">显示/隐藏详情</button>';
  132. html += '<div id="error-details" style="display: none; margin-top: 10px;">';
  133. if (data.data.error_messages.length > 0) {
  134. html += '<ul class="list-group">';
  135. data.data.error_messages.forEach(msg => {
  136. let errorMsg = msg;
  137. if (typeof msg === 'string') {
  138. // 分离错误消息和行数据
  139. const parts = msg.split('\n');
  140. if (parts.length > 1) {
  141. // 格式化显示
  142. html += `<li class="list-group-item list-group-item-danger">${parts[0]}`;
  143. html += `<button class="btn btn-sm btn-link" onclick="toggleRowData(this)">显示行数据</button>`;
  144. html += `<div class="row-data" style="display:none;margin-top:10px;"><pre>${parts[1]}</pre></div>`;
  145. html += `</li>`;
  146. } else {
  147. html += `<li class="list-group-item list-group-item-danger">${msg}</li>`;
  148. }
  149. } else {
  150. html += `<li class="list-group-item list-group-item-danger">${msg}</li>`;
  151. }
  152. });
  153. html += '</ul>';
  154. } else {
  155. html += '<p>没有详细错误信息</p>';
  156. }
  157. html += '</div></div>';
  158. }
  159. resultDiv.innerHTML = html;
  160. // 添加错误切换按钮的事件监听
  161. const toggleErrorsBtn = document.getElementById('toggle-errors');
  162. if (toggleErrorsBtn) {
  163. toggleErrorsBtn.addEventListener('click', function() {
  164. const errorDetails = document.getElementById('error-details');
  165. if (errorDetails.style.display === 'none') {
  166. errorDetails.style.display = 'block';
  167. this.textContent = '隐藏详情';
  168. } else {
  169. errorDetails.style.display = 'none';
  170. this.textContent = '显示详情';
  171. }
  172. });
  173. }
  174. } else {
  175. // 导入失败
  176. resultDiv.innerHTML = `<div class="alert alert-danger">${data.msg}</div>`;
  177. }
  178. })
  179. .catch(error => {
  180. console.error('Error:', error);
  181. resultDiv.innerHTML = '<div class="alert alert-danger">导入失败,请查看控制台了解详情</div>';
  182. });
  183. });
  184. });
  185. </script>
  186. {% endblock %}