statistics_region.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. <?php
  2. /**
  3. * 地区统计分析模块
  4. *
  5. * 包含与地区相关的数据分析功能
  6. */
  7. require_once 'statistics_utils.php';
  8. /**
  9. * 获取客户国家分布
  10. *
  11. * @param mysqli $conn 数据库连接
  12. * @return mysqli_result 客户国家分布数据结果集
  13. */
  14. function getCustomerCountryDistribution($conn) {
  15. $sql = "SELECT
  16. c.countryName,
  17. COUNT(cu.id) as customer_count
  18. FROM customer cu
  19. JOIN country c ON cu.cs_country = c.id
  20. GROUP BY cu.cs_country
  21. ORDER BY customer_count DESC
  22. LIMIT 10";
  23. return $conn->query($sql);
  24. }
  25. /**
  26. * 获取不同地区的订单数量
  27. *
  28. * @param mysqli $conn 数据库连接
  29. * @param string $start_date 开始日期
  30. * @param string $end_date 结束日期
  31. * @return mysqli_result 地区订单数据结果集
  32. */
  33. function getOrdersByRegion($conn, $start_date, $end_date) {
  34. $sql = "SELECT
  35. c.countryName,
  36. COUNT(o.id) as order_count,
  37. SUM(o.total_amount) as total_amount,
  38. SUM(oi.quantity) as total_quantity
  39. FROM orders o
  40. JOIN customer cu ON o.customer_id = cu.id
  41. JOIN country c ON cu.cs_country = c.id
  42. LEFT JOIN order_items oi ON o.id = oi.order_id
  43. WHERE o.order_date BETWEEN ? AND ?
  44. GROUP BY cu.cs_country
  45. ORDER BY total_quantity DESC
  46. LIMIT 10";
  47. $stmt = $conn->prepare($sql);
  48. $stmt->bind_param("ss", $start_date, $end_date);
  49. $stmt->execute();
  50. return $stmt->get_result();
  51. }
  52. /**
  53. * 获取地区销售同比环比数据
  54. *
  55. * @param mysqli $conn 数据库连接
  56. * @param string $current_start 当前周期开始日期
  57. * @param string $current_end 当前周期结束日期
  58. * @return array 地区销售同比环比数据
  59. */
  60. function getRegionSalesComparison($conn, $current_start, $current_end) {
  61. // 计算上一个相同时长的周期
  62. $current_start_date = new DateTime($current_start);
  63. $current_end_date = new DateTime($current_end);
  64. $interval = $current_start_date->diff($current_end_date);
  65. $prev_end_date = clone $current_start_date;
  66. $prev_end_date->modify('-1 day');
  67. $prev_start_date = clone $prev_end_date;
  68. $prev_start_date->sub($interval);
  69. $prev_start = $prev_start_date->format('Y-m-d');
  70. $prev_end = $prev_end_date->format('Y-m-d') . ' 23:59:59';
  71. // 获取当前周期数据
  72. $sql = "SELECT
  73. c.countryName,
  74. COUNT(o.id) as order_count,
  75. SUM(o.total_amount) as total_amount
  76. FROM orders o
  77. JOIN customer cu ON o.customer_id = cu.id
  78. JOIN country c ON cu.cs_country = c.id
  79. WHERE o.order_date BETWEEN ? AND ?
  80. GROUP BY cu.cs_country
  81. ORDER BY total_amount DESC
  82. LIMIT 5";
  83. $stmt = $conn->prepare($sql);
  84. $stmt->bind_param("ss", $current_start, $current_end);
  85. $stmt->execute();
  86. $current_result = $stmt->get_result();
  87. $current_data = [];
  88. while ($row = $current_result->fetch_assoc()) {
  89. $current_data[$row['countryName']] = [
  90. 'order_count' => $row['order_count'],
  91. 'total_amount' => $row['total_amount']
  92. ];
  93. }
  94. // 获取上一个周期数据
  95. $stmt = $conn->prepare($sql);
  96. $stmt->bind_param("ss", $prev_start, $prev_end);
  97. $stmt->execute();
  98. $prev_result = $stmt->get_result();
  99. $prev_data = [];
  100. while ($row = $prev_result->fetch_assoc()) {
  101. $prev_data[$row['countryName']] = [
  102. 'order_count' => $row['order_count'],
  103. 'total_amount' => $row['total_amount']
  104. ];
  105. }
  106. // 计算同比变化
  107. $comparison_data = [];
  108. foreach ($current_data as $country => $current) {
  109. $prev = $prev_data[$country] ?? ['order_count' => 0, 'total_amount' => 0];
  110. $order_growth = $prev['order_count'] > 0
  111. ? (($current['order_count'] - $prev['order_count']) / $prev['order_count']) * 100
  112. : 100;
  113. $amount_growth = $prev['total_amount'] > 0
  114. ? (($current['total_amount'] - $prev['total_amount']) / $prev['total_amount']) * 100
  115. : 100;
  116. $comparison_data[] = [
  117. 'countryName' => $country,
  118. 'current_orders' => $current['order_count'],
  119. 'prev_orders' => $prev['order_count'],
  120. 'order_growth' => $order_growth,
  121. 'current_amount' => $current['total_amount'],
  122. 'prev_amount' => $prev['total_amount'],
  123. 'amount_growth' => $amount_growth
  124. ];
  125. }
  126. return $comparison_data;
  127. }
  128. /**
  129. * 渲染客户国家分布图
  130. *
  131. * @param array $country_labels 国家标签
  132. * @param array $country_data 国家数据
  133. * @return void
  134. */
  135. function renderCustomerCountryDistributionChart($country_labels, $country_data) {
  136. ?>
  137. <div class="chart-container">
  138. <div class="chart-header">
  139. <h2 class="chart-title">客户国家分布</h2>
  140. </div>
  141. <canvas id="countryDistributionChart"></canvas>
  142. </div>
  143. <script>
  144. // 客户国家分布图
  145. var countryDistributionCtx = document.getElementById('countryDistributionChart').getContext('2d');
  146. var countryDistributionChart = new Chart(countryDistributionCtx, {
  147. type: 'pie',
  148. data: {
  149. labels: <?php echo json_encode($country_labels); ?>,
  150. datasets: [{
  151. data: <?php echo json_encode($country_data); ?>,
  152. backgroundColor: [
  153. 'rgba(255, 99, 132, 0.7)',
  154. 'rgba(54, 162, 235, 0.7)',
  155. 'rgba(255, 206, 86, 0.7)',
  156. 'rgba(75, 192, 192, 0.7)',
  157. 'rgba(153, 102, 255, 0.7)',
  158. 'rgba(255, 159, 64, 0.7)',
  159. 'rgba(199, 199, 199, 0.7)',
  160. 'rgba(83, 102, 255, 0.7)',
  161. 'rgba(40, 159, 64, 0.7)',
  162. 'rgba(210, 199, 199, 0.7)'
  163. ],
  164. borderWidth: 1
  165. }]
  166. },
  167. options: {
  168. responsive: true,
  169. plugins: {
  170. legend: {
  171. position: 'right',
  172. }
  173. }
  174. }
  175. });
  176. </script>
  177. <?php
  178. }
  179. /**
  180. * 渲染地区订单分析图
  181. *
  182. * @param array $region_labels 地区标签
  183. * @param array $region_orders 地区订单数量
  184. * @param array $region_quantities 地区产品数量
  185. * @return void
  186. */
  187. function renderRegionOrdersChart($region_labels, $region_orders, $region_quantities) {
  188. ?>
  189. <div class="chart-container">
  190. <div class="chart-header">
  191. <h2 class="chart-title">地区订单分析</h2>
  192. </div>
  193. <canvas id="regionOrdersChart"></canvas>
  194. </div>
  195. <script>
  196. // 地区订单分析图
  197. var regionOrdersCtx = document.getElementById('regionOrdersChart').getContext('2d');
  198. var regionOrdersChart = new Chart(regionOrdersCtx, {
  199. type: 'bar',
  200. data: {
  201. labels: <?php echo json_encode($region_labels); ?>,
  202. datasets: [
  203. {
  204. label: '订单数量',
  205. data: <?php echo json_encode($region_orders); ?>,
  206. backgroundColor: 'rgba(54, 162, 235, 0.6)',
  207. borderColor: 'rgba(54, 162, 235, 1)',
  208. borderWidth: 1,
  209. yAxisID: 'y-orders'
  210. },
  211. {
  212. label: '产品订购数量',
  213. data: <?php echo json_encode($region_quantities); ?>,
  214. backgroundColor: 'rgba(255, 99, 132, 0.6)',
  215. borderColor: 'rgba(255, 99, 132, 1)',
  216. borderWidth: 1,
  217. yAxisID: 'y-quantity'
  218. }
  219. ]
  220. },
  221. options: {
  222. responsive: true,
  223. scales: {
  224. x: {
  225. title: {
  226. display: true,
  227. text: '地区'
  228. }
  229. },
  230. 'y-orders': {
  231. type: 'linear',
  232. position: 'left',
  233. title: {
  234. display: true,
  235. text: '订单数量'
  236. },
  237. beginAtZero: true
  238. },
  239. 'y-quantity': {
  240. type: 'linear',
  241. position: 'right',
  242. title: {
  243. display: true,
  244. text: '产品订购数量'
  245. },
  246. beginAtZero: true,
  247. grid: {
  248. drawOnChartArea: false
  249. }
  250. }
  251. }
  252. }
  253. });
  254. </script>
  255. <?php
  256. }
  257. /**
  258. * 渲染地区销售同比环比表格
  259. *
  260. * @param array $comparison_data 比较数据
  261. * @return void
  262. */
  263. function renderRegionSalesComparisonTable($comparison_data) {
  264. ?>
  265. <div class="chart-container">
  266. <div class="chart-header">
  267. <h2 class="chart-title">地区销售同比分析</h2>
  268. </div>
  269. <table class="data-table">
  270. <thead>
  271. <tr>
  272. <th>国家/地区</th>
  273. <th>当前订单数</th>
  274. <th>上期订单数</th>
  275. <th>订单增长率</th>
  276. <th>当前销售额</th>
  277. <th>上期销售额</th>
  278. <th>销售额增长率</th>
  279. </tr>
  280. </thead>
  281. <tbody>
  282. <?php foreach ($comparison_data as $row): ?>
  283. <tr>
  284. <td><?php echo htmlspecialchars($row['countryName']); ?></td>
  285. <td><?php echo number_format($row['current_orders']); ?></td>
  286. <td><?php echo number_format($row['prev_orders']); ?></td>
  287. <td class="<?php echo $row['order_growth'] >= 0 ? 'positive' : 'negative'; ?>">
  288. <?php echo number_format($row['order_growth'], 2); ?>%
  289. </td>
  290. <td>¥<?php echo number_format($row['current_amount'], 2); ?></td>
  291. <td>¥<?php echo number_format($row['prev_amount'], 2); ?></td>
  292. <td class="<?php echo $row['amount_growth'] >= 0 ? 'positive' : 'negative'; ?>">
  293. <?php echo number_format($row['amount_growth'], 2); ?>%
  294. </td>
  295. </tr>
  296. <?php endforeach; ?>
  297. </tbody>
  298. </table>
  299. </div>
  300. <?php
  301. }