statistics_products.php 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149
  1. <?php
  2. /**
  3. * 产品统计分析模块
  4. *
  5. * 包含与产品相关的数据分析功能
  6. */
  7. require_once 'statistics_utils.php';
  8. /**
  9. * 获取热门产品数据
  10. *
  11. * @param mysqli $conn 数据库连接
  12. * @param string $start_date 开始日期
  13. * @param string $end_date 结束日期
  14. * @param int $limit 限制返回的产品数量
  15. * @return mysqli_result 热门产品数据结果集
  16. */
  17. function getTopProducts($conn, $start_date, $end_date, $limit = 5) {
  18. $sql = "SELECT
  19. p.ProductName,
  20. SUM(oi.quantity) as total_quantity,
  21. SUM(oi.total_price) as total_revenue
  22. FROM order_items oi
  23. JOIN products p ON oi.product_id = p.id
  24. JOIN orders o ON oi.order_id = o.id
  25. WHERE o.order_date BETWEEN ? AND ?
  26. GROUP BY oi.product_id
  27. ORDER BY total_revenue DESC
  28. LIMIT ?";
  29. $stmt = $conn->prepare($sql);
  30. $stmt->bind_param("ssi", $start_date, $end_date, $limit);
  31. $stmt->execute();
  32. return $stmt->get_result();
  33. }
  34. /**
  35. * 获取产品销售趋势
  36. *
  37. * @param mysqli $conn 数据库连接
  38. * @param string $start_date 开始日期
  39. * @param string $end_date 结束日期
  40. * @param int $product_id 产品ID,为0时获取所有产品的总体趋势
  41. * @param string $period 时间粒度 (day/week/month)
  42. * @return mysqli_result 产品销售趋势数据结果集
  43. */
  44. function getProductSalesTrend($conn, $start_date, $end_date, $product_id = 0, $period = 'month') {
  45. $groupFormat = '%Y-%m-%d';
  46. if ($period == 'week') {
  47. $groupFormat = '%x-W%v'; // ISO year and week number
  48. } else if ($period == 'month') {
  49. $groupFormat = '%Y-%m';
  50. }
  51. $sql = "SELECT
  52. DATE_FORMAT(o.order_date, '$groupFormat') as time_period,
  53. SUM(oi.quantity) as total_quantity,
  54. SUM(oi.total_price) as total_revenue,
  55. COUNT(DISTINCT o.id) as order_count
  56. FROM order_items oi
  57. JOIN orders o ON oi.order_id = o.id";
  58. if ($product_id > 0) {
  59. $sql .= " WHERE o.order_date BETWEEN ? AND ? AND oi.product_id = ?";
  60. } else {
  61. $sql .= " WHERE o.order_date BETWEEN ? AND ?";
  62. }
  63. $sql .= " GROUP BY time_period
  64. ORDER BY MIN(o.order_date)";
  65. $stmt = $conn->prepare($sql);
  66. if ($product_id > 0) {
  67. $stmt->bind_param("ssi", $start_date, $end_date, $product_id);
  68. } else {
  69. $stmt->bind_param("ss", $start_date, $end_date);
  70. }
  71. $stmt->execute();
  72. return $stmt->get_result();
  73. }
  74. /**
  75. * 获取产品类别销售分布
  76. *
  77. * @param mysqli $conn 数据库连接
  78. * @param string $start_date 开始日期
  79. * @param string $end_date 结束日期
  80. * @return mysqli_result 产品类别销售分布数据结果集
  81. */
  82. function getProductCategorySales($conn, $start_date, $end_date) {
  83. $sql = "SELECT
  84. pc.name as category_name,
  85. SUM(oi.quantity) as total_quantity,
  86. SUM(oi.total_price) as total_revenue,
  87. COUNT(DISTINCT o.id) as order_count
  88. FROM order_items oi
  89. JOIN products p ON oi.product_id = p.id
  90. JOIN product_categories pc ON p.category_id = pc.id
  91. JOIN orders o ON oi.order_id = o.id
  92. WHERE o.order_date BETWEEN ? AND ?
  93. GROUP BY p.category_id
  94. ORDER BY total_revenue DESC";
  95. $stmt = $conn->prepare($sql);
  96. $stmt->bind_param("ss", $start_date, $end_date);
  97. $stmt->execute();
  98. return $stmt->get_result();
  99. }
  100. /**
  101. * 获取产品与地区关联分析
  102. *
  103. * @param mysqli $conn 数据库连接
  104. * @param string $start_date 开始日期
  105. * @param string $end_date 结束日期
  106. * @param int $limit 限制返回的产品-地区组合数量
  107. * @return mysqli_result 产品与地区关联分析数据结果集
  108. */
  109. function getProductRegionAnalysis($conn, $start_date, $end_date, $limit = 10) {
  110. $sql = "SELECT
  111. p.ProductName,
  112. c.countryName,
  113. SUM(oi.quantity) as total_quantity,
  114. SUM(oi.total_price) as total_revenue
  115. FROM order_items oi
  116. JOIN products p ON oi.product_id = p.id
  117. JOIN orders o ON oi.order_id = o.id
  118. JOIN customer cu ON o.customer_id = cu.id
  119. JOIN country c ON cu.cs_country = c.id
  120. WHERE o.order_date BETWEEN ? AND ?
  121. GROUP BY oi.product_id, cu.cs_country
  122. ORDER BY total_revenue DESC
  123. LIMIT ?";
  124. $stmt = $conn->prepare($sql);
  125. $stmt->bind_param("ssi", $start_date, $end_date, $limit);
  126. $stmt->execute();
  127. return $stmt->get_result();
  128. }
  129. /**
  130. * 获取产品销售概览数据
  131. */
  132. function getProductSalesOverview($conn, $start_date, $end_date, $category_filter = 0) {
  133. $where_clause = "WHERE o.order_date BETWEEN ? AND ?";
  134. $params = [$start_date, $end_date];
  135. if ($category_filter > 0) {
  136. $where_clause .= " AND p.category_id = ?";
  137. $params[] = $category_filter;
  138. }
  139. $sql = "SELECT
  140. COUNT(DISTINCT oi.product_id) as total_products,
  141. SUM(oi.quantity) as total_quantity,
  142. SUM(oi.total_price) as total_revenue,
  143. AVG(oi.unit_price) as avg_unit_price,
  144. COUNT(DISTINCT o.id) as total_orders,
  145. SUM(oi.total_price) / COUNT(DISTINCT o.id) as avg_order_value,
  146. COUNT(DISTINCT o.customer_id) as total_customers
  147. FROM order_items oi
  148. JOIN orders o ON oi.order_id = o.id
  149. JOIN products p ON oi.product_id = p.id
  150. $where_clause";
  151. $stmt = $conn->prepare($sql);
  152. $stmt->bind_param(str_repeat('s', count($params)), ...$params);
  153. $stmt->execute();
  154. return $stmt->get_result()->fetch_assoc();
  155. }
  156. /**
  157. * 获取产品价格趋势分析
  158. */
  159. function getProductPriceTrendAnalysis($conn, $start_date, $end_date, $product_id = 0, $period = 'month') {
  160. $groupFormat = getPeriodFormat($period);
  161. $sql = "SELECT
  162. DATE_FORMAT(o.order_date, '$groupFormat') as time_period,
  163. AVG(oi.unit_price) as avg_price,
  164. MIN(oi.unit_price) as min_price,
  165. MAX(oi.unit_price) as max_price
  166. FROM order_items oi
  167. JOIN orders o ON oi.order_id = o.id";
  168. if ($product_id > 0) {
  169. $sql .= " WHERE o.order_date BETWEEN ? AND ? AND oi.product_id = ?";
  170. } else {
  171. $sql .= " WHERE o.order_date BETWEEN ? AND ?";
  172. }
  173. $sql .= " GROUP BY time_period ORDER BY MIN(o.order_date)";
  174. $stmt = $conn->prepare($sql);
  175. if ($product_id > 0) {
  176. $stmt->bind_param("ssi", $start_date, $end_date, $product_id);
  177. } else {
  178. $stmt->bind_param("ss", $start_date, $end_date);
  179. }
  180. $stmt->execute();
  181. return $stmt->get_result();
  182. }
  183. /**
  184. * 获取产品季节性分析
  185. */
  186. function getProductSeasonalityAnalysis($conn, $start_date, $end_date, $product_id = 0) {
  187. $sql = "SELECT
  188. MONTH(o.order_date) as month,
  189. SUM(oi.quantity) as total_quantity,
  190. SUM(oi.total_price) as total_revenue,
  191. COUNT(DISTINCT o.id) as order_count
  192. FROM order_items oi
  193. JOIN orders o ON oi.order_id = o.id";
  194. if ($product_id > 0) {
  195. $sql .= " WHERE oi.product_id = ? AND o.order_date BETWEEN ? AND ?";
  196. } else {
  197. $sql .= " WHERE o.order_date BETWEEN ? AND ?";
  198. }
  199. $sql .= " GROUP BY MONTH(o.order_date)
  200. ORDER BY MONTH(o.order_date)";
  201. $stmt = $conn->prepare($sql);
  202. if ($product_id > 0) {
  203. $stmt->bind_param("iss", $product_id, $start_date, $end_date);
  204. } else {
  205. $stmt->bind_param("ss", $start_date, $end_date);
  206. }
  207. $stmt->execute();
  208. return $stmt->get_result();
  209. }
  210. /**
  211. * 获取产品客户细分分析
  212. */
  213. function getProductCustomerSegmentAnalysis($conn, $start_date, $end_date, $product_id = 0) {
  214. $sql = "SELECT
  215. ct.businessType as segment_name,
  216. COUNT(DISTINCT o.customer_id) as customer_count,
  217. SUM(oi.quantity) as total_quantity,
  218. SUM(oi.total_price) as total_revenue,
  219. AVG(oi.unit_price) as avg_unit_price
  220. FROM order_items oi
  221. JOIN orders o ON oi.order_id = o.id
  222. JOIN customer c ON o.customer_id = c.id
  223. JOIN clienttype ct ON c.cs_type = ct.id";
  224. if ($product_id > 0) {
  225. $sql .= " WHERE oi.product_id = ? AND o.order_date BETWEEN ? AND ?";
  226. } else {
  227. $sql .= " WHERE o.order_date BETWEEN ? AND ?";
  228. }
  229. $sql .= " GROUP BY ct.id";
  230. $stmt = $conn->prepare($sql);
  231. if ($product_id > 0) {
  232. $stmt->bind_param("iss", $product_id, $start_date, $end_date);
  233. } else {
  234. $stmt->bind_param("ss", $start_date, $end_date);
  235. }
  236. $stmt->execute();
  237. return $stmt->get_result();
  238. }
  239. /**
  240. * 获取产品分类列表
  241. *
  242. * @param mysqli $conn 数据库连接
  243. * @return mysqli_result 产品分类数据结果集
  244. */
  245. function getProductCategories($conn) {
  246. $sql = "SELECT
  247. id,
  248. parent_id,
  249. name,
  250. description,
  251. sort_order
  252. FROM product_categories
  253. WHERE status = 1
  254. ORDER BY sort_order ASC, id ASC";
  255. $stmt = $conn->prepare($sql);
  256. $stmt->execute();
  257. return $stmt->get_result();
  258. }
  259. /**
  260. * 渲染热门产品表格
  261. *
  262. * @param mysqli_result $top_products 热门产品数据
  263. * @return void
  264. */
  265. function renderTopProductsTable($top_products) {
  266. ?>
  267. <div class="chart-container">
  268. <div class="chart-header">
  269. <h2 class="chart-title">热门产品</h2>
  270. </div>
  271. <table class="data-table">
  272. <thead>
  273. <tr>
  274. <th>产品名称</th>
  275. <th>销售数量</th>
  276. <th>销售收入</th>
  277. </tr>
  278. </thead>
  279. <tbody>
  280. <?php while ($row = $top_products->fetch_assoc()): ?>
  281. <tr>
  282. <td><?php echo htmlspecialchars($row['ProductName']); ?></td>
  283. <td><?php echo number_format($row['total_quantity']); ?></td>
  284. <td>¥<?php echo number_format($row['total_revenue'], 2); ?></td>
  285. </tr>
  286. <?php endwhile; ?>
  287. </tbody>
  288. </table>
  289. </div>
  290. <?php
  291. }
  292. /**
  293. * 渲染产品销售趋势图
  294. *
  295. * @param array $time_labels 时间标签
  296. * @param array $quantities 产品销售数量
  297. * @param array $revenues 产品销售收入
  298. * @return void
  299. */
  300. function renderProductSalesTrendChart($time_labels, $quantities, $revenues) {
  301. ?>
  302. <div class="chart-container">
  303. <div class="chart-header">
  304. <h2 class="chart-title">产品销售趋势</h2>
  305. </div>
  306. <canvas id="productSalesTrendChart"></canvas>
  307. </div>
  308. <script>
  309. // 产品销售趋势图
  310. var productSalesTrendCtx = document.getElementById('productSalesTrendChart').getContext('2d');
  311. var productSalesTrendChart = new Chart(productSalesTrendCtx, {
  312. type: 'line',
  313. data: {
  314. labels: <?php echo json_encode($time_labels); ?>,
  315. datasets: [
  316. {
  317. label: '销售数量',
  318. data: <?php echo json_encode($quantities); ?>,
  319. backgroundColor: 'rgba(54, 162, 235, 0.2)',
  320. borderColor: 'rgba(54, 162, 235, 1)',
  321. borderWidth: 2,
  322. yAxisID: 'y-quantity',
  323. tension: 0.1
  324. },
  325. {
  326. label: '销售收入',
  327. data: <?php echo json_encode($revenues); ?>,
  328. backgroundColor: 'rgba(255, 99, 132, 0.2)',
  329. borderColor: 'rgba(255, 99, 132, 1)',
  330. borderWidth: 2,
  331. yAxisID: 'y-revenue',
  332. tension: 0.1
  333. }
  334. ]
  335. },
  336. options: {
  337. responsive: true,
  338. scales: {
  339. 'y-quantity': {
  340. type: 'linear',
  341. position: 'left',
  342. title: {
  343. display: true,
  344. text: '销售数量'
  345. },
  346. beginAtZero: true
  347. },
  348. 'y-revenue': {
  349. type: 'linear',
  350. position: 'right',
  351. title: {
  352. display: true,
  353. text: '销售收入'
  354. },
  355. beginAtZero: true,
  356. grid: {
  357. drawOnChartArea: false
  358. }
  359. }
  360. }
  361. }
  362. });
  363. </script>
  364. <?php
  365. }
  366. /**
  367. * 渲染产品类别销售分布图
  368. *
  369. * @param array $categories 类别名称
  370. * @param array $quantities 类别销售数量
  371. * @param array $revenues 类别销售收入
  372. * @return void
  373. */
  374. function renderProductCategorySalesChart($categories, $quantities, $revenues) {
  375. ?>
  376. <div class="chart-container">
  377. <div class="chart-header">
  378. <h2 class="chart-title">产品类别销售分布</h2>
  379. </div>
  380. <style>
  381. .pie-charts-container {
  382. display: flex;
  383. flex-direction: row;
  384. justify-content: space-between;
  385. margin-bottom: 20px;
  386. }
  387. .pie-chart-wrapper {
  388. flex: 0 0 48%;
  389. max-width: 48%;
  390. }
  391. </style>
  392. <div class="pie-charts-container">
  393. <div class="pie-chart-wrapper">
  394. <h3 style="text-align: center; margin-bottom: 15px;">产品类别销售数量分布</h3>
  395. <canvas id="categoryQuantityChart"></canvas>
  396. </div>
  397. <div class="pie-chart-wrapper">
  398. <h3 style="text-align: center; margin-bottom: 15px;">产品类别销售收入分布</h3>
  399. <canvas id="categoryRevenueChart"></canvas>
  400. </div>
  401. </div>
  402. </div>
  403. <script>
  404. // 产品类别数量分布图
  405. var categoryQuantityCtx = document.getElementById('categoryQuantityChart').getContext('2d');
  406. var categoryQuantityChart = new Chart(categoryQuantityCtx, {
  407. type: 'pie',
  408. data: {
  409. labels: <?php echo json_encode($categories); ?>,
  410. datasets: [{
  411. data: <?php echo json_encode($quantities); ?>,
  412. backgroundColor: [
  413. 'rgba(255, 99, 132, 0.7)',
  414. 'rgba(54, 162, 235, 0.7)',
  415. 'rgba(255, 206, 86, 0.7)',
  416. 'rgba(75, 192, 192, 0.7)',
  417. 'rgba(153, 102, 255, 0.7)',
  418. 'rgba(255, 159, 64, 0.7)'
  419. ],
  420. borderWidth: 1
  421. }]
  422. },
  423. options: {
  424. responsive: true,
  425. maintainAspectRatio: true,
  426. plugins: {
  427. legend: {
  428. position: 'bottom',
  429. }
  430. }
  431. }
  432. });
  433. // 产品类别收入分布图
  434. var categoryRevenueCtx = document.getElementById('categoryRevenueChart').getContext('2d');
  435. var categoryRevenueChart = new Chart(categoryRevenueCtx, {
  436. type: 'pie',
  437. data: {
  438. labels: <?php echo json_encode($categories); ?>,
  439. datasets: [{
  440. data: <?php echo json_encode($revenues); ?>,
  441. backgroundColor: [
  442. 'rgba(255, 99, 132, 0.7)',
  443. 'rgba(54, 162, 235, 0.7)',
  444. 'rgba(255, 206, 86, 0.7)',
  445. 'rgba(75, 192, 192, 0.7)',
  446. 'rgba(153, 102, 255, 0.7)',
  447. 'rgba(255, 159, 64, 0.7)'
  448. ],
  449. borderWidth: 1
  450. }]
  451. },
  452. options: {
  453. responsive: true,
  454. maintainAspectRatio: true,
  455. plugins: {
  456. legend: {
  457. position: 'bottom',
  458. }
  459. }
  460. }
  461. });
  462. </script>
  463. <?php
  464. }
  465. /**
  466. * 渲染产品与地区关联分析表格
  467. *
  468. * @param mysqli_result $product_region_data 产品与地区关联数据
  469. * @return void
  470. */
  471. function renderProductRegionAnalysisTable($product_region_data) {
  472. ?>
  473. <div class="chart-container">
  474. <div class="chart-header">
  475. <h2 class="chart-title">产品地区关联分析</h2>
  476. </div>
  477. <table class="data-table">
  478. <thead>
  479. <tr>
  480. <th>产品名称</th>
  481. <th>国家/地区</th>
  482. <th>销售数量</th>
  483. <th>销售收入</th>
  484. </tr>
  485. </thead>
  486. <tbody>
  487. <?php while ($row = $product_region_data->fetch_assoc()): ?>
  488. <tr>
  489. <td><?php echo htmlspecialchars($row['ProductName']); ?></td>
  490. <td><?php echo htmlspecialchars($row['countryName']); ?></td>
  491. <td><?php echo number_format($row['total_quantity']); ?></td>
  492. <td>¥<?php echo number_format($row['total_revenue'], 2); ?></td>
  493. </tr>
  494. <?php endwhile; ?>
  495. </tbody>
  496. </table>
  497. </div>
  498. <?php
  499. }
  500. /**
  501. * 渲染产品销售概览
  502. */
  503. function renderProductSalesOverview($overview) {
  504. // 处理可能为null的值
  505. $total_products = isset($overview['total_products']) ? $overview['total_products'] : 0;
  506. $total_quantity = isset($overview['total_quantity']) ? $overview['total_quantity'] : 0;
  507. $total_revenue = isset($overview['total_revenue']) ? $overview['total_revenue'] : 0;
  508. $avg_unit_price = isset($overview['avg_unit_price']) ? $overview['avg_unit_price'] : 0;
  509. $total_orders = isset($overview['total_orders']) ? $overview['total_orders'] : 0;
  510. $avg_order_value = isset($overview['avg_order_value']) ? $overview['avg_order_value'] : 0;
  511. ?>
  512. <div class="stats-card-container">
  513. <div class="stats-card">
  514. <div class="stats-card-header">
  515. <h3>总销售产品数</h3>
  516. </div>
  517. <div class="stats-card-body">
  518. <div class="stats-card-value"><?php echo number_format($total_products); ?></div>
  519. <div class="stats-card-subtitle">种类</div>
  520. </div>
  521. </div>
  522. <div class="stats-card">
  523. <div class="stats-card-header">
  524. <h3>总销售数量</h3>
  525. </div>
  526. <div class="stats-card-body">
  527. <div class="stats-card-value"><?php echo number_format($total_quantity); ?></div>
  528. <div class="stats-card-subtitle">件</div>
  529. </div>
  530. </div>
  531. <div class="stats-card">
  532. <div class="stats-card-header">
  533. <h3>总销售收入</h3>
  534. </div>
  535. <div class="stats-card-body">
  536. <div class="stats-card-value">¥<?php echo number_format($total_revenue, 2); ?></div>
  537. <div class="stats-card-subtitle">元</div>
  538. </div>
  539. </div>
  540. <div class="stats-card">
  541. <div class="stats-card-header">
  542. <h3>平均单价</h3>
  543. </div>
  544. <div class="stats-card-body">
  545. <div class="stats-card-value">¥<?php echo number_format($avg_unit_price, 2); ?></div>
  546. <div class="stats-card-subtitle">元/件</div>
  547. </div>
  548. </div>
  549. <div class="stats-card">
  550. <div class="stats-card-header">
  551. <h3>订单数量</h3>
  552. </div>
  553. <div class="stats-card-body">
  554. <div class="stats-card-value"><?php echo number_format($total_orders); ?></div>
  555. <div class="stats-card-subtitle">笔</div>
  556. </div>
  557. </div>
  558. <div class="stats-card">
  559. <div class="stats-card-header">
  560. <h3>平均订单金额</h3>
  561. </div>
  562. <div class="stats-card-body">
  563. <div class="stats-card-value">¥<?php echo number_format($avg_order_value, 2); ?></div>
  564. <div class="stats-card-subtitle">元/订单</div>
  565. </div>
  566. </div>
  567. </div>
  568. <?php
  569. }
  570. /**
  571. * 渲染产品价格趋势图表
  572. */
  573. function renderProductPriceTrendChart($price_trend_data) {
  574. $time_periods = [];
  575. $avg_prices = [];
  576. $min_prices = [];
  577. $max_prices = [];
  578. while ($row = $price_trend_data->fetch_assoc()) {
  579. $time_periods[] = $row['time_period'];
  580. $avg_prices[] = round($row['avg_price'], 2);
  581. $min_prices[] = round($row['min_price'], 2);
  582. $max_prices[] = round($row['max_price'], 2);
  583. }
  584. ?>
  585. <div class="chart-container">
  586. <div class="chart-header">
  587. <h2 class="chart-title">产品价格趋势分析</h2>
  588. </div>
  589. <canvas id="priceTrendChart"></canvas>
  590. </div>
  591. <script>
  592. var priceTrendCtx = document.getElementById('priceTrendChart').getContext('2d');
  593. new Chart(priceTrendCtx, {
  594. type: 'line',
  595. data: {
  596. labels: <?php echo json_encode($time_periods); ?>,
  597. datasets: [
  598. {
  599. label: '平均价格',
  600. data: <?php echo json_encode($avg_prices); ?>,
  601. borderColor: 'rgb(54, 162, 235)',
  602. backgroundColor: 'rgba(54, 162, 235, 0.1)',
  603. borderWidth: 2,
  604. fill: false
  605. },
  606. {
  607. label: '最低价格',
  608. data: <?php echo json_encode($min_prices); ?>,
  609. borderColor: 'rgb(75, 192, 192)',
  610. backgroundColor: 'rgba(75, 192, 192, 0.1)',
  611. borderWidth: 2,
  612. fill: false
  613. },
  614. {
  615. label: '最高价格',
  616. data: <?php echo json_encode($max_prices); ?>,
  617. borderColor: 'rgb(255, 99, 132)',
  618. backgroundColor: 'rgba(255, 99, 132, 0.1)',
  619. borderWidth: 2,
  620. fill: false
  621. }
  622. ]
  623. },
  624. options: {
  625. responsive: true,
  626. scales: {
  627. y: {
  628. beginAtZero: true,
  629. title: {
  630. display: true,
  631. text: '价格 (元)'
  632. }
  633. }
  634. },
  635. plugins: {
  636. title: {
  637. display: true,
  638. text: '产品价格变化趋势'
  639. }
  640. }
  641. }
  642. });
  643. </script>
  644. <?php
  645. }
  646. /**
  647. * 渲染产品季节性分析图表
  648. */
  649. function renderProductSeasonalityChart($seasonality_data) {
  650. $months = [];
  651. $quantities = [];
  652. $revenues = [];
  653. $order_counts = [];
  654. while ($row = $seasonality_data->fetch_assoc()) {
  655. $months[] = date('n月', mktime(0, 0, 0, $row['month'], 1));
  656. $quantities[] = (int)$row['total_quantity'];
  657. $revenues[] = round($row['total_revenue'], 2);
  658. $order_counts[] = (int)$row['order_count'];
  659. }
  660. ?>
  661. <div class="chart-container">
  662. <div class="chart-header">
  663. <h2 class="chart-title">产品季节性分析</h2>
  664. </div>
  665. <canvas id="seasonalityChart"></canvas>
  666. </div>
  667. <script>
  668. var seasonalityCtx = document.getElementById('seasonalityChart').getContext('2d');
  669. new Chart(seasonalityCtx, {
  670. type: 'bar',
  671. data: {
  672. labels: <?php echo json_encode($months); ?>,
  673. datasets: [
  674. {
  675. label: '销售数量',
  676. data: <?php echo json_encode($quantities); ?>,
  677. backgroundColor: 'rgba(54, 162, 235, 0.5)',
  678. borderColor: 'rgb(54, 162, 235)',
  679. borderWidth: 1,
  680. yAxisID: 'y-quantity'
  681. },
  682. {
  683. label: '销售收入',
  684. data: <?php echo json_encode($revenues); ?>,
  685. backgroundColor: 'rgba(255, 99, 132, 0.5)',
  686. borderColor: 'rgb(255, 99, 132)',
  687. borderWidth: 1,
  688. yAxisID: 'y-revenue'
  689. },
  690. {
  691. label: '订单数',
  692. data: <?php echo json_encode($order_counts); ?>,
  693. type: 'line',
  694. fill: false,
  695. borderColor: 'rgb(75, 192, 192)',
  696. tension: 0.1,
  697. yAxisID: 'y-orders'
  698. }
  699. ]
  700. },
  701. options: {
  702. responsive: true,
  703. scales: {
  704. 'y-quantity': {
  705. type: 'linear',
  706. position: 'left',
  707. title: {
  708. display: true,
  709. text: '销售数量'
  710. }
  711. },
  712. 'y-revenue': {
  713. type: 'linear',
  714. position: 'right',
  715. title: {
  716. display: true,
  717. text: '销售收入 (元)'
  718. },
  719. grid: {
  720. drawOnChartArea: false
  721. }
  722. },
  723. 'y-orders': {
  724. type: 'linear',
  725. position: 'right',
  726. title: {
  727. display: true,
  728. text: '订单数'
  729. },
  730. grid: {
  731. drawOnChartArea: false
  732. }
  733. }
  734. },
  735. plugins: {
  736. title: {
  737. display: true,
  738. text: '产品销售季节性分布'
  739. }
  740. }
  741. }
  742. });
  743. </script>
  744. <?php
  745. }
  746. /**
  747. * 渲染产品客户细分分析图表
  748. */
  749. function renderProductCustomerSegmentChart($segment_data) {
  750. $segments = [];
  751. $customer_counts = [];
  752. $revenues = [];
  753. $avg_prices = [];
  754. while ($row = $segment_data->fetch_assoc()) {
  755. $segments[] = $row['segment_name'];
  756. $customer_counts[] = (int)$row['customer_count'];
  757. $revenues[] = round($row['total_revenue'], 2);
  758. $avg_prices[] = round($row['avg_unit_price'], 2);
  759. }
  760. ?>
  761. <div class="chart-container">
  762. <div class="chart-header">
  763. <h2 class="chart-title">产品客户细分分析</h2>
  764. </div>
  765. <div class="chart-row">
  766. <div class="chart-column">
  767. <canvas id="customerSegmentChart1"></canvas>
  768. </div>
  769. <div class="chart-column">
  770. <canvas id="customerSegmentChart2"></canvas>
  771. </div>
  772. </div>
  773. </div>
  774. <script>
  775. // 客户数量和收入分布
  776. var segmentCtx1 = document.getElementById('customerSegmentChart1').getContext('2d');
  777. new Chart(segmentCtx1, {
  778. type: 'bar',
  779. data: {
  780. labels: <?php echo json_encode($segments); ?>,
  781. datasets: [
  782. {
  783. label: '客户数量',
  784. data: <?php echo json_encode($customer_counts); ?>,
  785. backgroundColor: 'rgba(54, 162, 235, 0.5)',
  786. borderColor: 'rgb(54, 162, 235)',
  787. borderWidth: 1,
  788. yAxisID: 'y-customers'
  789. },
  790. {
  791. label: '销售收入',
  792. data: <?php echo json_encode($revenues); ?>,
  793. backgroundColor: 'rgba(255, 99, 132, 0.5)',
  794. borderColor: 'rgb(255, 99, 132)',
  795. borderWidth: 1,
  796. yAxisID: 'y-revenue'
  797. }
  798. ]
  799. },
  800. options: {
  801. responsive: true,
  802. scales: {
  803. 'y-customers': {
  804. type: 'linear',
  805. position: 'left',
  806. title: {
  807. display: true,
  808. text: '客户数量'
  809. }
  810. },
  811. 'y-revenue': {
  812. type: 'linear',
  813. position: 'right',
  814. title: {
  815. display: true,
  816. text: '销售收入 (元)'
  817. },
  818. grid: {
  819. drawOnChartArea: false
  820. }
  821. }
  822. },
  823. plugins: {
  824. title: {
  825. display: true,
  826. text: '客户细分分布'
  827. }
  828. }
  829. }
  830. });
  831. // 平均单价分布
  832. var segmentCtx2 = document.getElementById('customerSegmentChart2').getContext('2d');
  833. new Chart(segmentCtx2, {
  834. type: 'radar',
  835. data: {
  836. labels: <?php echo json_encode($segments); ?>,
  837. datasets: [{
  838. label: '平均单价',
  839. data: <?php echo json_encode($avg_prices); ?>,
  840. backgroundColor: 'rgba(75, 192, 192, 0.2)',
  841. borderColor: 'rgb(75, 192, 192)',
  842. pointBackgroundColor: 'rgb(75, 192, 192)',
  843. pointBorderColor: '#fff',
  844. pointHoverBackgroundColor: '#fff',
  845. pointHoverBorderColor: 'rgb(75, 192, 192)'
  846. }]
  847. },
  848. options: {
  849. responsive: true,
  850. plugins: {
  851. title: {
  852. display: true,
  853. text: '客户细分平均单价分布'
  854. }
  855. }
  856. }
  857. });
  858. </script>
  859. <?php
  860. }
  861. /**
  862. * 获取产品增长率分析
  863. */
  864. function getProductGrowthAnalysis($conn, $start_date, $end_date, $period = 'month') {
  865. $groupFormat = getPeriodFormat($period);
  866. // 获取当前期间的数据
  867. $sql = "SELECT
  868. p.ProductName,
  869. SUM(oi.total_price) as current_revenue,
  870. SUM(oi.quantity) as current_quantity,
  871. COUNT(DISTINCT o.id) as current_orders
  872. FROM order_items oi
  873. JOIN products p ON oi.product_id = p.id
  874. JOIN orders o ON oi.order_id = o.id
  875. WHERE o.order_date BETWEEN ? AND ?
  876. GROUP BY oi.product_id
  877. HAVING current_revenue > 0
  878. ORDER BY current_revenue DESC
  879. LIMIT 10";
  880. $stmt = $conn->prepare($sql);
  881. $stmt->bind_param("ss", $start_date, $end_date);
  882. $stmt->execute();
  883. $current_data = $stmt->get_result();
  884. // 计算上一个时间段
  885. $date1 = new DateTime($start_date);
  886. $date2 = new DateTime($end_date);
  887. $interval = $date1->diff($date2);
  888. $days_diff = $interval->days;
  889. $prev_end = $date1->format('Y-m-d');
  890. $prev_start = $date1->modify("-{$days_diff} days")->format('Y-m-d');
  891. // 获取上一期间的数据
  892. $sql = "SELECT
  893. p.ProductName,
  894. SUM(oi.total_price) as prev_revenue,
  895. SUM(oi.quantity) as prev_quantity,
  896. COUNT(DISTINCT o.id) as prev_orders
  897. FROM order_items oi
  898. JOIN products p ON oi.product_id = p.id
  899. JOIN orders o ON oi.order_id = o.id
  900. WHERE o.order_date BETWEEN ? AND ?
  901. GROUP BY oi.product_id";
  902. $stmt = $conn->prepare($sql);
  903. $stmt->bind_param("ss", $prev_start, $prev_end);
  904. $stmt->execute();
  905. $prev_result = $stmt->get_result();
  906. $prev_data = [];
  907. while ($row = $prev_result->fetch_assoc()) {
  908. $prev_data[$row['ProductName']] = $row;
  909. }
  910. $growth_data = [];
  911. while ($current = $current_data->fetch_assoc()) {
  912. $product_name = $current['ProductName'];
  913. $prev = isset($prev_data[$product_name]) ? $prev_data[$product_name] : [
  914. 'prev_revenue' => 0,
  915. 'prev_quantity' => 0,
  916. 'prev_orders' => 0
  917. ];
  918. $growth_data[] = [
  919. 'product_name' => $product_name,
  920. 'current_revenue' => $current['current_revenue'],
  921. 'current_quantity' => $current['current_quantity'],
  922. 'current_orders' => $current['current_orders'],
  923. 'prev_revenue' => $prev['prev_revenue'],
  924. 'prev_quantity' => $prev['prev_quantity'],
  925. 'prev_orders' => $prev['prev_orders'],
  926. 'revenue_growth' => calculateGrowthRate($current['current_revenue'], $prev['prev_revenue']),
  927. 'quantity_growth' => calculateGrowthRate($current['current_quantity'], $prev['prev_quantity']),
  928. 'orders_growth' => calculateGrowthRate($current['current_orders'], $prev['prev_orders'])
  929. ];
  930. }
  931. return $growth_data;
  932. }
  933. /**
  934. * 计算增长率
  935. */
  936. function calculateGrowthRate($current, $previous) {
  937. if ($previous == 0) {
  938. return $current > 0 ? 100 : 0;
  939. }
  940. return round((($current - $previous) / $previous) * 100, 2);
  941. }
  942. /**
  943. * 渲染产品增长率分析
  944. */
  945. function renderProductGrowthAnalysis($growth_data) {
  946. ?>
  947. <div class="chart-container">
  948. <div class="chart-header">
  949. <h2 class="chart-title">产品增长率分析</h2>
  950. <div class="chart-subtitle">与上一时期相比</div>
  951. </div>
  952. <table class="data-table">
  953. <thead>
  954. <tr>
  955. <th>产品名称</th>
  956. <th>当期收入</th>
  957. <th>收入增长率</th>
  958. <th>当期销量</th>
  959. <th>销量增长率</th>
  960. <th>当期订单数</th>
  961. <th>订单增长率</th>
  962. </tr>
  963. </thead>
  964. <tbody>
  965. <?php foreach ($growth_data as $row): ?>
  966. <tr>
  967. <td><?php echo htmlspecialchars($row['product_name']); ?></td>
  968. <td>¥<?php echo number_format($row['current_revenue'], 2); ?></td>
  969. <td class="<?php echo $row['revenue_growth'] >= 0 ? 'positive' : 'negative'; ?>">
  970. <?php echo ($row['revenue_growth'] >= 0 ? '+' : '') . $row['revenue_growth']; ?>%
  971. </td>
  972. <td><?php echo number_format($row['current_quantity']); ?></td>
  973. <td class="<?php echo $row['quantity_growth'] >= 0 ? 'positive' : 'negative'; ?>">
  974. <?php echo ($row['quantity_growth'] >= 0 ? '+' : '') . $row['quantity_growth']; ?>%
  975. </td>
  976. <td><?php echo number_format($row['current_orders']); ?></td>
  977. <td class="<?php echo $row['orders_growth'] >= 0 ? 'positive' : 'negative'; ?>">
  978. <?php echo ($row['orders_growth'] >= 0 ? '+' : '') . $row['orders_growth']; ?>%
  979. </td>
  980. </tr>
  981. <?php endforeach; ?>
  982. </tbody>
  983. </table>
  984. </div>
  985. <style>
  986. .positive {
  987. color: #4CAF50;
  988. font-weight: bold;
  989. }
  990. .negative {
  991. color: #f44336;
  992. font-weight: bold;
  993. }
  994. .chart-subtitle {
  995. font-size: 14px;
  996. color: #666;
  997. margin-top: 5px;
  998. }
  999. </style>
  1000. <?php
  1001. }
  1002. /**
  1003. * 获取产品购买频率分析
  1004. */
  1005. function getProductPurchaseFrequency($conn, $start_date, $end_date) {
  1006. $sql = "SELECT
  1007. p.ProductName,
  1008. COUNT(DISTINCT o.id) as order_count,
  1009. COUNT(DISTINCT o.customer_id) as customer_count,
  1010. COUNT(DISTINCT o.id) / COUNT(DISTINCT o.customer_id) as purchase_frequency,
  1011. AVG(
  1012. CASE
  1013. WHEN next_order.next_date IS NOT NULL
  1014. THEN DATEDIFF(next_order.next_date, o.order_date)
  1015. ELSE NULL
  1016. END
  1017. ) as avg_days_between_orders
  1018. FROM order_items oi
  1019. JOIN products p ON oi.product_id = p.id
  1020. JOIN orders o ON oi.order_id = o.id
  1021. LEFT JOIN (
  1022. SELECT
  1023. o1.customer_id,
  1024. o1.order_date,
  1025. MIN(o2.order_date) as next_date
  1026. FROM orders o1
  1027. LEFT JOIN orders o2 ON o1.customer_id = o2.customer_id
  1028. AND o2.order_date > o1.order_date
  1029. WHERE o1.order_date BETWEEN ? AND ?
  1030. GROUP BY o1.customer_id, o1.order_date
  1031. ) next_order ON o.customer_id = next_order.customer_id
  1032. AND o.order_date = next_order.order_date
  1033. WHERE o.order_date BETWEEN ? AND ?
  1034. GROUP BY p.id
  1035. HAVING order_count > 1
  1036. ORDER BY purchase_frequency DESC
  1037. LIMIT 10";
  1038. $stmt = $conn->prepare($sql);
  1039. $stmt->bind_param("ssss", $start_date, $end_date, $start_date, $end_date);
  1040. $stmt->execute();
  1041. return $stmt->get_result();
  1042. }
  1043. /**
  1044. * 渲染产品购买频率分析
  1045. */
  1046. function renderProductPurchaseFrequency($frequency_data) {
  1047. ?>
  1048. <div class="chart-container">
  1049. <div class="chart-header">
  1050. <h2 class="chart-title">产品购买频率分析</h2>
  1051. </div>
  1052. <table class="data-table">
  1053. <thead>
  1054. <tr>
  1055. <th>产品名称</th>
  1056. <th>订单总数</th>
  1057. <th>购买客户数</th>
  1058. <th>平均购买频率</th>
  1059. <th>平均购买间隔(天)</th>
  1060. </tr>
  1061. </thead>
  1062. <tbody>
  1063. <?php while ($row = $frequency_data->fetch_assoc()): ?>
  1064. <tr>
  1065. <td><?php echo htmlspecialchars($row['ProductName']); ?></td>
  1066. <td><?php echo number_format($row['order_count']); ?></td>
  1067. <td><?php echo number_format($row['customer_count']); ?></td>
  1068. <td><?php echo number_format($row['purchase_frequency'], 2); ?>次/客户</td>
  1069. <td><?php echo $row['avg_days_between_orders'] ? number_format($row['avg_days_between_orders'], 1) : '-'; ?></td>
  1070. </tr>
  1071. <?php endwhile; ?>
  1072. </tbody>
  1073. </table>
  1074. </div>
  1075. <?php
  1076. }