statistics_sales.php 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176
  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. * @return array 销售概览数据
  15. */
  16. function getSalesOverview($conn, $start_date, $end_date) {
  17. $sql = "SELECT
  18. COUNT(DISTINCT o.id) as total_orders,
  19. SUM(o.total_amount) as total_revenue,
  20. AVG(o.total_amount) as avg_order_value,
  21. COUNT(DISTINCT o.customer_id) as unique_customers,
  22. SUM(oi.quantity) as total_items_sold
  23. FROM orders o
  24. LEFT JOIN order_items oi ON o.id = oi.order_id
  25. WHERE o.order_date BETWEEN ? AND ?
  26. AND o.order_status != 0"; // 排除已取消订单
  27. $stmt = $conn->prepare($sql);
  28. $stmt->bind_param("ss", $start_date, $end_date);
  29. $stmt->execute();
  30. return $stmt->get_result()->fetch_assoc();
  31. }
  32. /**
  33. * 获取订单转化率统计
  34. */
  35. function getOrderConversionStats($conn, $start_date, $end_date) {
  36. $sql = "SELECT
  37. order_status,
  38. COUNT(*) as count,
  39. SUM(total_amount) as amount
  40. FROM orders
  41. WHERE order_date BETWEEN ? AND ?
  42. GROUP BY order_status";
  43. $stmt = $conn->prepare($sql);
  44. $stmt->bind_param("ss", $start_date, $end_date);
  45. $stmt->execute();
  46. return $stmt->get_result();
  47. }
  48. /**
  49. * 获取产品类别销售统计
  50. */
  51. function getProductCategorySales($conn, $start_date, $end_date) {
  52. $sql = "SELECT
  53. pc.name as category_name,
  54. COUNT(DISTINCT o.id) as order_count,
  55. SUM(oi.quantity) as total_quantity,
  56. SUM(oi.total_price) as total_revenue
  57. FROM orders o
  58. JOIN order_items oi ON o.id = oi.order_id
  59. JOIN products p ON oi.product_id = p.id
  60. JOIN product_categories pc ON p.category_id = pc.id
  61. WHERE o.order_date BETWEEN ? AND ?
  62. AND o.order_status != 0
  63. GROUP BY pc.id
  64. ORDER BY total_revenue DESC";
  65. $stmt = $conn->prepare($sql);
  66. $stmt->bind_param("ss", $start_date, $end_date);
  67. $stmt->execute();
  68. return $stmt->get_result();
  69. }
  70. /**
  71. * 获取客户地区分布
  72. */
  73. function getCustomerDistribution($conn, $start_date, $end_date) {
  74. $sql = "SELECT
  75. c.countryName as region,
  76. COUNT(DISTINCT o.customer_id) as customer_count,
  77. COUNT(o.id) as order_count,
  78. SUM(o.total_amount) as total_revenue
  79. FROM orders o
  80. JOIN customer cu ON o.customer_id = cu.id
  81. JOIN country c ON cu.cs_country = c.id
  82. WHERE o.order_date BETWEEN ? AND ?
  83. AND o.order_status != 0
  84. GROUP BY c.id
  85. ORDER BY total_revenue DESC";
  86. $stmt = $conn->prepare($sql);
  87. $stmt->bind_param("ss", $start_date, $end_date);
  88. $stmt->execute();
  89. return $stmt->get_result();
  90. }
  91. /**
  92. * 获取销售员业绩统计
  93. */
  94. function getEmployeePerformance($conn, $start_date, $end_date) {
  95. $sql = "SELECT
  96. e.em_user as employee_name,
  97. COUNT(DISTINCT o.id) as order_count,
  98. COUNT(DISTINCT o.customer_id) as customer_count,
  99. SUM(o.total_amount) as total_revenue,
  100. AVG(o.total_amount) as avg_order_value
  101. FROM orders o
  102. JOIN employee e ON o.employee_id = e.id
  103. WHERE o.order_date BETWEEN ? AND ?
  104. AND o.order_status != 0
  105. GROUP BY e.id
  106. ORDER BY total_revenue DESC";
  107. $stmt = $conn->prepare($sql);
  108. $stmt->bind_param("ss", $start_date, $end_date);
  109. $stmt->execute();
  110. return $stmt->get_result();
  111. }
  112. /**
  113. * 获取支付状态统计
  114. */
  115. function getPaymentStatusStats($conn, $start_date, $end_date) {
  116. $sql = "SELECT
  117. payment_status,
  118. COUNT(*) as count,
  119. SUM(total_amount) as amount
  120. FROM orders
  121. WHERE order_date BETWEEN ? AND ?
  122. AND order_status != 0
  123. GROUP BY payment_status";
  124. $stmt = $conn->prepare($sql);
  125. $stmt->bind_param("ss", $start_date, $end_date);
  126. $stmt->execute();
  127. return $stmt->get_result();
  128. }
  129. /**
  130. * 获取每月销售趋势
  131. *
  132. * @param mysqli $conn 数据库连接
  133. * @param string $start_date 开始日期
  134. * @param string $end_date 结束日期
  135. * @return mysqli_result 月度销售数据结果集
  136. */
  137. function getMonthlySalesTrend($conn, $start_date, $end_date) {
  138. $sql = "SELECT
  139. DATE_FORMAT(order_date, '%Y-%m') as month,
  140. COUNT(DISTINCT id) as orders,
  141. SUM(total_amount) as revenue,
  142. COUNT(DISTINCT customer_id) as unique_customers
  143. FROM orders
  144. WHERE order_date BETWEEN ? AND ?
  145. AND order_status != 0
  146. GROUP BY DATE_FORMAT(order_date, '%Y-%m')
  147. ORDER BY month";
  148. $stmt = $conn->prepare($sql);
  149. $stmt->bind_param("ss", $start_date, $end_date);
  150. $stmt->execute();
  151. return $stmt->get_result();
  152. }
  153. /**
  154. * 获取详细时间段订单趋势
  155. *
  156. * @param mysqli $conn 数据库连接
  157. * @param string $start_date 开始日期
  158. * @param string $end_date 结束日期
  159. * @param string $period 时间粒度 (day/week/month)
  160. * @return mysqli_result 订单趋势数据结果集
  161. */
  162. function getDetailedOrderTrend($conn, $start_date, $end_date, $period = 'day') {
  163. $groupFormat = '%Y-%m-%d';
  164. if ($period == 'week') {
  165. $groupFormat = '%x-W%v';
  166. } else if ($period == 'month') {
  167. $groupFormat = '%Y-%m';
  168. }
  169. $sql = "SELECT
  170. DATE_FORMAT(o.order_date, '$groupFormat') as time_period,
  171. COUNT(DISTINCT o.id) as order_count,
  172. SUM(oi.quantity) as total_quantity,
  173. SUM(o.total_amount) as total_amount,
  174. COUNT(DISTINCT o.customer_id) as unique_customers
  175. FROM orders o
  176. LEFT JOIN order_items oi ON o.id = oi.order_id
  177. WHERE o.order_date BETWEEN ? AND ?
  178. AND o.order_status != 0
  179. GROUP BY time_period
  180. ORDER BY MIN(o.order_date)";
  181. $stmt = $conn->prepare($sql);
  182. $stmt->bind_param("ss", $start_date, $end_date);
  183. $stmt->execute();
  184. return $stmt->get_result();
  185. }
  186. /**
  187. * 渲染销售概览卡片
  188. *
  189. * @param array $sales_overview 销售概览数据
  190. * @return void
  191. */
  192. function renderSalesOverviewCards($sales_overview) {
  193. // 添加空值检查函数
  194. ?>
  195. <div class="stats-grid">
  196. <div class="stat-card">
  197. <h3>总订单数</h3>
  198. <div class="stat-value"><?php echo formatNumber($sales_overview['total_orders']); ?></div>
  199. </div>
  200. <div class="stat-card">
  201. <h3>总收入</h3>
  202. <div class="stat-value"><?php echo formatCurrency($sales_overview['total_revenue']); ?></div>
  203. </div>
  204. <div class="stat-card">
  205. <h3>平均订单金额</h3>
  206. <div class="stat-value"><?php echo formatCurrency($sales_overview['avg_order_value']); ?></div>
  207. </div>
  208. <div class="stat-card">
  209. <h3>独立客户数</h3>
  210. <div class="stat-value"><?php echo formatNumber($sales_overview['unique_customers']); ?></div>
  211. </div>
  212. <div class="stat-card">
  213. <h3>总销售数量</h3>
  214. <div class="stat-value"><?php echo formatNumber($sales_overview['total_items_sold']); ?></div>
  215. </div>
  216. </div>
  217. <?php
  218. }
  219. /**
  220. * 渲染订单转化率分析
  221. */
  222. function renderConversionAnalysis($conversion_stats) {
  223. // 添加空值检查函数
  224. $status_names = [
  225. 1 => '待确认',
  226. 2 => '已确认',
  227. 3 => '生产中',
  228. 4 => '已发货',
  229. 5 => '已完成',
  230. 0 => '已取消'
  231. ];
  232. $total_orders = 0;
  233. $data = [];
  234. while ($row = $conversion_stats->fetch_assoc()) {
  235. $total_orders += $row['count'];
  236. $data[$row['order_status']] = $row;
  237. }
  238. ?>
  239. <div class="analysis-grid">
  240. <div>
  241. <canvas id="orderStatusChart"></canvas>
  242. </div>
  243. <div class="table-responsive">
  244. <table class="data-table">
  245. <thead>
  246. <tr>
  247. <th>订单状态</th>
  248. <th>订单数</th>
  249. <th>转化率</th>
  250. <th>金额</th>
  251. </tr>
  252. </thead>
  253. <tbody>
  254. <?php foreach ($status_names as $status_id => $status_name): ?>
  255. <?php if (isset($data[$status_id])): ?>
  256. <tr>
  257. <td><?php echo $status_name; ?></td>
  258. <td><?php echo formatNumber($data[$status_id]['count']); ?></td>
  259. <td><?php echo formatNumber(($data[$status_id]['count'] / ($total_orders ?: 1)) * 100, 1); ?>%</td>
  260. <td><?php echo formatCurrency($data[$status_id]['amount']); ?></td>
  261. </tr>
  262. <?php endif; ?>
  263. <?php endforeach; ?>
  264. </tbody>
  265. </table>
  266. </div>
  267. </div>
  268. <script>
  269. var orderStatusCtx = document.getElementById('orderStatusChart').getContext('2d');
  270. new Chart(orderStatusCtx, {
  271. type: 'doughnut',
  272. data: {
  273. labels: <?php
  274. $labels = [];
  275. $values = [];
  276. foreach ($status_names as $status_id => $status_name) {
  277. if (isset($data[$status_id])) {
  278. $labels[] = $status_name;
  279. $values[] = $data[$status_id]['count'];
  280. }
  281. }
  282. echo json_encode($labels);
  283. ?>,
  284. datasets: [{
  285. data: <?php echo json_encode($values); ?>,
  286. backgroundColor: [
  287. '#FF6384',
  288. '#36A2EB',
  289. '#FFCE56',
  290. '#4BC0C0',
  291. '#9966FF',
  292. '#FF9F40'
  293. ]
  294. }]
  295. },
  296. options: {
  297. responsive: true,
  298. plugins: {
  299. legend: {
  300. position: 'right'
  301. },
  302. title: {
  303. display: true,
  304. text: '订单状态分布'
  305. }
  306. }
  307. }
  308. });
  309. </script>
  310. <?php
  311. }
  312. /**
  313. * 渲染产品类别销售分析图表
  314. */
  315. function renderCategorySalesChart($category_sales) {
  316. $labels = [];
  317. $quantities = [];
  318. $revenues = [];
  319. while ($row = $category_sales->fetch_assoc()) {
  320. $labels[] = $row['category_name'];
  321. $quantities[] = $row['total_quantity'];
  322. $revenues[] = $row['total_revenue'];
  323. }
  324. ?>
  325. <div class="analysis-grid">
  326. <div>
  327. <canvas id="categorySalesChart"></canvas>
  328. </div>
  329. <div class="table-responsive">
  330. <table class="data-table">
  331. <thead>
  332. <tr>
  333. <th>产品类别</th>
  334. <th>订单数</th>
  335. <th>销售数量</th>
  336. <th>销售金额</th>
  337. </tr>
  338. </thead>
  339. <tbody>
  340. <?php
  341. $category_sales->data_seek(0);
  342. while ($row = $category_sales->fetch_assoc()):
  343. ?>
  344. <tr>
  345. <td><?php echo $row['category_name']; ?></td>
  346. <td><?php echo number_format($row['order_count']); ?></td>
  347. <td><?php echo number_format($row['total_quantity']); ?></td>
  348. <td>¥<?php echo number_format($row['total_revenue'], 2); ?></td>
  349. </tr>
  350. <?php endwhile; ?>
  351. </tbody>
  352. </table>
  353. </div>
  354. </div>
  355. <script>
  356. var categorySalesCtx = document.getElementById('categorySalesChart').getContext('2d');
  357. new Chart(categorySalesCtx, {
  358. type: 'bar',
  359. data: {
  360. labels: <?php echo json_encode($labels); ?>,
  361. datasets: [
  362. {
  363. label: '销售数量',
  364. data: <?php echo json_encode($quantities); ?>,
  365. backgroundColor: 'rgba(54, 162, 235, 0.5)',
  366. borderColor: 'rgba(54, 162, 235, 1)',
  367. borderWidth: 1,
  368. yAxisID: 'y-quantity'
  369. },
  370. {
  371. label: '销售金额',
  372. data: <?php echo json_encode($revenues); ?>,
  373. backgroundColor: 'rgba(255, 99, 132, 0.5)',
  374. borderColor: 'rgba(255, 99, 132, 1)',
  375. borderWidth: 1,
  376. yAxisID: 'y-revenue'
  377. }
  378. ]
  379. },
  380. options: {
  381. responsive: true,
  382. scales: {
  383. 'y-quantity': {
  384. type: 'linear',
  385. position: 'left',
  386. title: {
  387. display: true,
  388. text: '销售数量'
  389. }
  390. },
  391. 'y-revenue': {
  392. type: 'linear',
  393. position: 'right',
  394. title: {
  395. display: true,
  396. text: '销售金额'
  397. },
  398. grid: {
  399. drawOnChartArea: false
  400. }
  401. }
  402. }
  403. }
  404. });
  405. </script>
  406. <?php
  407. }
  408. /**
  409. * 渲染客户地区分布图表
  410. */
  411. function renderCustomerDistributionChart($customer_distribution) {
  412. $regions = [];
  413. $customers = [];
  414. $revenues = [];
  415. while ($row = $customer_distribution->fetch_assoc()) {
  416. $regions[] = $row['region'];
  417. $customers[] = $row['customer_count'];
  418. $revenues[] = $row['total_revenue'];
  419. }
  420. ?>
  421. <div class="analysis-grid">
  422. <div>
  423. <canvas id="regionDistributionChart"></canvas>
  424. </div>
  425. <div class="table-responsive">
  426. <table class="data-table">
  427. <thead>
  428. <tr>
  429. <th>地区</th>
  430. <th>客户数</th>
  431. <th>订单数</th>
  432. <th>销售金额</th>
  433. </tr>
  434. </thead>
  435. <tbody>
  436. <?php
  437. $customer_distribution->data_seek(0);
  438. while ($row = $customer_distribution->fetch_assoc()):
  439. ?>
  440. <tr>
  441. <td><?php echo $row['region']; ?></td>
  442. <td><?php echo number_format($row['customer_count']); ?></td>
  443. <td><?php echo number_format($row['order_count']); ?></td>
  444. <td>¥<?php echo number_format($row['total_revenue'], 2); ?></td>
  445. </tr>
  446. <?php endwhile; ?>
  447. </tbody>
  448. </table>
  449. </div>
  450. </div>
  451. <script>
  452. var regionDistributionCtx = document.getElementById('regionDistributionChart').getContext('2d');
  453. new Chart(regionDistributionCtx, {
  454. type: 'bar',
  455. data: {
  456. labels: <?php echo json_encode($regions); ?>,
  457. datasets: [
  458. {
  459. label: '客户数',
  460. data: <?php echo json_encode($customers); ?>,
  461. backgroundColor: 'rgba(75, 192, 192, 0.5)',
  462. borderColor: 'rgba(75, 192, 192, 1)',
  463. borderWidth: 1,
  464. yAxisID: 'y-customers'
  465. },
  466. {
  467. label: '销售金额',
  468. data: <?php echo json_encode($revenues); ?>,
  469. backgroundColor: 'rgba(255, 159, 64, 0.5)',
  470. borderColor: 'rgba(255, 159, 64, 1)',
  471. borderWidth: 1,
  472. yAxisID: 'y-revenue'
  473. }
  474. ]
  475. },
  476. options: {
  477. responsive: true,
  478. scales: {
  479. 'y-customers': {
  480. type: 'linear',
  481. position: 'left',
  482. title: {
  483. display: true,
  484. text: '客户数'
  485. }
  486. },
  487. 'y-revenue': {
  488. type: 'linear',
  489. position: 'right',
  490. title: {
  491. display: true,
  492. text: '销售金额'
  493. },
  494. grid: {
  495. drawOnChartArea: false
  496. }
  497. }
  498. }
  499. }
  500. });
  501. </script>
  502. <?php
  503. }
  504. /**
  505. * 渲染销售员业绩表格
  506. */
  507. function renderEmployeePerformanceTable($employee_performance) {
  508. // 添加空值检查函数
  509. ?>
  510. <div class="table-responsive">
  511. <table class="data-table">
  512. <thead>
  513. <tr>
  514. <th>销售员</th>
  515. <th>订单数</th>
  516. <th>客户数</th>
  517. <th>总销售额</th>
  518. <th>平均订单金额</th>
  519. </tr>
  520. </thead>
  521. <tbody>
  522. <?php while ($row = $employee_performance->fetch_assoc()): ?>
  523. <tr>
  524. <td><?php echo htmlspecialchars($row['employee_name']); ?></td>
  525. <td><?php echo formatNumber($row['order_count']); ?></td>
  526. <td><?php echo formatNumber($row['customer_count']); ?></td>
  527. <td><?php echo formatCurrency($row['total_revenue']); ?></td>
  528. <td><?php echo formatCurrency($row['avg_order_value']); ?></td>
  529. </tr>
  530. <?php endwhile; ?>
  531. </tbody>
  532. </table>
  533. </div>
  534. <?php
  535. }
  536. /**
  537. * 渲染支付状态分析图表
  538. */
  539. function renderPaymentStatusChart($payment_stats) {
  540. $status_names = [
  541. 0 => '未付款',
  542. 1 => '部分付款',
  543. 2 => '已付清'
  544. ];
  545. $data = [];
  546. while ($row = $payment_stats->fetch_assoc()) {
  547. $data[$row['payment_status']] = $row;
  548. }
  549. ?>
  550. <div class="analysis-grid">
  551. <div>
  552. <canvas id="paymentStatusChart"></canvas>
  553. </div>
  554. <div class="table-responsive">
  555. <table class="data-table">
  556. <thead>
  557. <tr>
  558. <th>支付状态</th>
  559. <th>订单数</th>
  560. <th>订单金额</th>
  561. </tr>
  562. </thead>
  563. <tbody>
  564. <?php foreach ($status_names as $status_id => $status_name): ?>
  565. <?php if (isset($data[$status_id])): ?>
  566. <tr>
  567. <td><?php echo $status_name; ?></td>
  568. <td><?php echo number_format($data[$status_id]['count']); ?></td>
  569. <td>¥<?php echo number_format($data[$status_id]['amount'], 2); ?></td>
  570. </tr>
  571. <?php endif; ?>
  572. <?php endforeach; ?>
  573. </tbody>
  574. </table>
  575. </div>
  576. </div>
  577. <script>
  578. var paymentStatusCtx = document.getElementById('paymentStatusChart').getContext('2d');
  579. new Chart(paymentStatusCtx, {
  580. type: 'pie',
  581. data: {
  582. labels: <?php
  583. $labels = [];
  584. $values = [];
  585. foreach ($status_names as $status_id => $status_name) {
  586. if (isset($data[$status_id])) {
  587. $labels[] = $status_name;
  588. $values[] = $data[$status_id]['amount'];
  589. }
  590. }
  591. echo json_encode($labels);
  592. ?>,
  593. datasets: [{
  594. data: <?php echo json_encode($values); ?>,
  595. backgroundColor: [
  596. '#FF6384',
  597. '#36A2EB',
  598. '#4BC0C0'
  599. ]
  600. }]
  601. },
  602. options: {
  603. responsive: true,
  604. plugins: {
  605. legend: {
  606. position: 'right'
  607. },
  608. title: {
  609. display: true,
  610. text: '支付状态分布'
  611. }
  612. }
  613. }
  614. });
  615. </script>
  616. <?php
  617. }
  618. /**
  619. * 渲染月度销售趋势图
  620. *
  621. * @param array $monthly_labels 月份标签
  622. * @param array $monthly_orders 月度订单数量
  623. * @param array $monthly_revenue 月度收入
  624. * @return void
  625. */
  626. function renderMonthlySalesTrendChart($monthly_labels, $monthly_orders, $monthly_revenue) {
  627. ?>
  628. <div class="chart-container">
  629. <div class="chart-header">
  630. <h2 class="chart-title">销售趋势</h2>
  631. </div>
  632. <canvas id="salesTrendChart"></canvas>
  633. </div>
  634. <script>
  635. var salesTrendCtx = document.getElementById('salesTrendChart').getContext('2d');
  636. new Chart(salesTrendCtx, {
  637. type: 'line',
  638. data: {
  639. labels: <?php echo json_encode($monthly_labels); ?>,
  640. datasets: [
  641. {
  642. label: '订单数量',
  643. data: <?php echo json_encode($monthly_orders); ?>,
  644. backgroundColor: 'rgba(54, 162, 235, 0.2)',
  645. borderColor: 'rgba(54, 162, 235, 1)',
  646. borderWidth: 2,
  647. yAxisID: 'y-orders',
  648. tension: 0.1
  649. },
  650. {
  651. label: '销售收入',
  652. data: <?php echo json_encode($monthly_revenue); ?>,
  653. backgroundColor: 'rgba(255, 99, 132, 0.2)',
  654. borderColor: 'rgba(255, 99, 132, 1)',
  655. borderWidth: 2,
  656. yAxisID: 'y-revenue',
  657. tension: 0.1
  658. }
  659. ]
  660. },
  661. options: {
  662. responsive: true,
  663. scales: {
  664. 'y-orders': {
  665. type: 'linear',
  666. position: 'left',
  667. title: {
  668. display: true,
  669. text: '订单数量'
  670. }
  671. },
  672. 'y-revenue': {
  673. type: 'linear',
  674. position: 'right',
  675. title: {
  676. display: true,
  677. text: '销售收入'
  678. },
  679. grid: {
  680. drawOnChartArea: false
  681. }
  682. }
  683. }
  684. }
  685. });
  686. </script>
  687. <?php
  688. }
  689. /**
  690. * 渲染详细订单趋势图
  691. *
  692. * @param array $time_labels 时间标签
  693. * @param array $time_orders 时间段订单数量
  694. * @param array $time_quantities 时间段产品数量
  695. * @param string $period 时间粒度
  696. * @return void
  697. */
  698. function renderDetailedOrderTrendChart($time_labels, $time_orders, $time_quantities, $period = 'day') {
  699. $period_text = $period == 'day' ? '日' : ($period == 'week' ? '周' : '月');
  700. ?>
  701. <div class="chart-container">
  702. <div class="chart-header">
  703. <h2 class="chart-title">详细订单趋势 (<?php echo $period_text; ?>)</h2>
  704. </div>
  705. <canvas id="detailedOrdersChart"></canvas>
  706. </div>
  707. <script>
  708. var detailedOrdersCtx = document.getElementById('detailedOrdersChart').getContext('2d');
  709. new Chart(detailedOrdersCtx, {
  710. type: 'line',
  711. data: {
  712. labels: <?php echo json_encode($time_labels); ?>,
  713. datasets: [
  714. {
  715. label: '订单数量',
  716. data: <?php echo json_encode($time_orders); ?>,
  717. backgroundColor: 'rgba(75, 192, 192, 0.2)',
  718. borderColor: 'rgba(75, 192, 192, 1)',
  719. borderWidth: 2,
  720. yAxisID: 'y-orders',
  721. tension: 0.1
  722. },
  723. {
  724. label: '产品订购数量',
  725. data: <?php echo json_encode($time_quantities); ?>,
  726. backgroundColor: 'rgba(255, 159, 64, 0.2)',
  727. borderColor: 'rgba(255, 159, 64, 1)',
  728. borderWidth: 2,
  729. yAxisID: 'y-quantity',
  730. tension: 0.1
  731. }
  732. ]
  733. },
  734. options: {
  735. responsive: true,
  736. scales: {
  737. x: {
  738. title: {
  739. display: true,
  740. text: '时间'
  741. }
  742. },
  743. 'y-orders': {
  744. type: 'linear',
  745. position: 'left',
  746. title: {
  747. display: true,
  748. text: '订单数量'
  749. },
  750. beginAtZero: true
  751. },
  752. 'y-quantity': {
  753. type: 'linear',
  754. position: 'right',
  755. title: {
  756. display: true,
  757. text: '产品订购数量'
  758. },
  759. beginAtZero: true,
  760. grid: {
  761. drawOnChartArea: false
  762. }
  763. }
  764. }
  765. }
  766. });
  767. </script>
  768. <?php
  769. }
  770. /**
  771. * 获取所有业务员列表
  772. */
  773. function getAllEmployees($conn) {
  774. $sql = "SELECT id, em_user, em_email, em_tel FROM employee ORDER BY em_user";
  775. $result = $conn->query($sql);
  776. return $result->fetch_all(MYSQLI_ASSOC);
  777. }
  778. /**
  779. * 获取业务员详细信息
  780. */
  781. function getEmployeeDetail($conn, $employee_id) {
  782. $sql = "SELECT id, em_user, em_email, em_tel FROM employee WHERE id = ?";
  783. $stmt = $conn->prepare($sql);
  784. $stmt->bind_param("i", $employee_id);
  785. $stmt->execute();
  786. return $stmt->get_result()->fetch_assoc();
  787. }
  788. /**
  789. * 获取业务员统计数据
  790. */
  791. function getEmployeeStats($conn, $employee_id, $start_date, $end_date) {
  792. $sql = "SELECT
  793. COUNT(DISTINCT o.id) as total_orders,
  794. SUM(o.total_amount) as total_revenue,
  795. COUNT(DISTINCT o.customer_id) as customer_count,
  796. AVG(o.total_amount) as avg_order_value,
  797. SUM(CASE WHEN o.order_status = 5 THEN 1 ELSE 0 END) * 100.0 / COUNT(*) as completion_rate
  798. FROM orders o
  799. WHERE o.employee_id = ?
  800. AND o.order_date BETWEEN ? AND ?
  801. AND o.order_status != 0";
  802. $stmt = $conn->prepare($sql);
  803. $stmt->bind_param("iss", $employee_id, $start_date, $end_date);
  804. $stmt->execute();
  805. return $stmt->get_result()->fetch_assoc();
  806. }
  807. /**
  808. * 渲染业务员销售趋势
  809. */
  810. function renderEmployeeSalesTrend($conn, $employee_id, $start_date, $end_date) {
  811. $sql = "SELECT
  812. DATE_FORMAT(order_date, '%Y-%m-%d') as date,
  813. COUNT(DISTINCT id) as orders,
  814. SUM(total_amount) as revenue
  815. FROM orders
  816. WHERE employee_id = ?
  817. AND order_date BETWEEN ? AND ?
  818. AND order_status != 0
  819. GROUP BY DATE_FORMAT(order_date, '%Y-%m-%d')
  820. ORDER BY date";
  821. $stmt = $conn->prepare($sql);
  822. $stmt->bind_param("iss", $employee_id, $start_date, $end_date);
  823. $stmt->execute();
  824. $result = $stmt->get_result();
  825. $dates = [];
  826. $orders = [];
  827. $revenues = [];
  828. while ($row = $result->fetch_assoc()) {
  829. $dates[] = $row['date'];
  830. $orders[] = $row['orders'];
  831. $revenues[] = $row['revenue'];
  832. }
  833. ?>
  834. <div class="chart-container">
  835. <div class="chart-header">
  836. <h2 class="chart-title">销售趋势</h2>
  837. </div>
  838. <canvas id="employeeSalesTrendChart"></canvas>
  839. </div>
  840. <script>
  841. var employeeSalesTrendCtx = document.getElementById('employeeSalesTrendChart').getContext('2d');
  842. new Chart(employeeSalesTrendCtx, {
  843. type: 'line',
  844. data: {
  845. labels: <?php echo json_encode($dates); ?>,
  846. datasets: [
  847. {
  848. label: '订单数量',
  849. data: <?php echo json_encode($orders); ?>,
  850. backgroundColor: 'rgba(54, 162, 235, 0.2)',
  851. borderColor: 'rgba(54, 162, 235, 1)',
  852. borderWidth: 2,
  853. yAxisID: 'y-orders',
  854. tension: 0.1
  855. },
  856. {
  857. label: '销售收入',
  858. data: <?php echo json_encode($revenues); ?>,
  859. backgroundColor: 'rgba(255, 99, 132, 0.2)',
  860. borderColor: 'rgba(255, 99, 132, 1)',
  861. borderWidth: 2,
  862. yAxisID: 'y-revenue',
  863. tension: 0.1
  864. }
  865. ]
  866. },
  867. options: {
  868. responsive: true,
  869. scales: {
  870. 'y-orders': {
  871. type: 'linear',
  872. position: 'left',
  873. title: {
  874. display: true,
  875. text: '订单数量'
  876. }
  877. },
  878. 'y-revenue': {
  879. type: 'linear',
  880. position: 'right',
  881. title: {
  882. display: true,
  883. text: '销售收入'
  884. },
  885. grid: {
  886. drawOnChartArea: false
  887. }
  888. }
  889. }
  890. }
  891. });
  892. </script>
  893. <?php
  894. }
  895. /**
  896. * 渲染业务员客户分布
  897. */
  898. function renderEmployeeCustomerDistribution($conn, $employee_id, $start_date, $end_date) {
  899. $sql = "SELECT
  900. c.countryName as region,
  901. COUNT(DISTINCT o.customer_id) as customer_count,
  902. SUM(o.total_amount) as total_revenue
  903. FROM orders o
  904. JOIN customer cu ON o.customer_id = cu.id
  905. JOIN country c ON cu.cs_country = c.id
  906. WHERE o.employee_id = ?
  907. AND o.order_date BETWEEN ? AND ?
  908. AND o.order_status != 0
  909. GROUP BY c.id
  910. ORDER BY total_revenue DESC
  911. LIMIT 10";
  912. $stmt = $conn->prepare($sql);
  913. $stmt->bind_param("iss", $employee_id, $start_date, $end_date);
  914. $stmt->execute();
  915. $result = $stmt->get_result();
  916. $regions = [];
  917. $customers = [];
  918. $revenues = [];
  919. while ($row = $result->fetch_assoc()) {
  920. $regions[] = $row['region'];
  921. $customers[] = $row['customer_count'];
  922. $revenues[] = $row['total_revenue'];
  923. }
  924. ?>
  925. <div class="chart-container">
  926. <div class="chart-header">
  927. <h2 class="chart-title">客户地区分布 (Top 10)</h2>
  928. </div>
  929. <canvas id="employeeCustomerChart"></canvas>
  930. </div>
  931. <script>
  932. var employeeCustomerCtx = document.getElementById('employeeCustomerChart').getContext('2d');
  933. new Chart(employeeCustomerCtx, {
  934. type: 'bar',
  935. data: {
  936. labels: <?php echo json_encode($regions); ?>,
  937. datasets: [
  938. {
  939. label: '客户数',
  940. data: <?php echo json_encode($customers); ?>,
  941. backgroundColor: 'rgba(75, 192, 192, 0.5)',
  942. borderColor: 'rgba(75, 192, 192, 1)',
  943. borderWidth: 1,
  944. yAxisID: 'y-customers'
  945. },
  946. {
  947. label: '销售金额',
  948. data: <?php echo json_encode($revenues); ?>,
  949. backgroundColor: 'rgba(255, 159, 64, 0.5)',
  950. borderColor: 'rgba(255, 159, 64, 1)',
  951. borderWidth: 1,
  952. yAxisID: 'y-revenue'
  953. }
  954. ]
  955. },
  956. options: {
  957. responsive: true,
  958. scales: {
  959. 'y-customers': {
  960. type: 'linear',
  961. position: 'left',
  962. title: {
  963. display: true,
  964. text: '客户数'
  965. }
  966. },
  967. 'y-revenue': {
  968. type: 'linear',
  969. position: 'right',
  970. title: {
  971. display: true,
  972. text: '销售金额'
  973. },
  974. grid: {
  975. drawOnChartArea: false
  976. }
  977. }
  978. }
  979. }
  980. });
  981. </script>
  982. <?php
  983. }
  984. /**
  985. * 渲染业务员产品销售分析
  986. */
  987. function renderEmployeeProductAnalysis($conn, $employee_id, $start_date, $end_date) {
  988. $sql = "SELECT
  989. pc.name as category_name,
  990. COUNT(DISTINCT o.id) as order_count,
  991. SUM(oi.quantity) as total_quantity,
  992. SUM(oi.total_price) as total_revenue
  993. FROM orders o
  994. JOIN order_items oi ON o.id = oi.order_id
  995. JOIN products p ON oi.product_id = p.id
  996. JOIN product_categories pc ON p.category_id = pc.id
  997. WHERE o.employee_id = ?
  998. AND o.order_date BETWEEN ? AND ?
  999. AND o.order_status != 0
  1000. GROUP BY pc.id
  1001. ORDER BY total_revenue DESC";
  1002. $stmt = $conn->prepare($sql);
  1003. $stmt->bind_param("iss", $employee_id, $start_date, $end_date);
  1004. $stmt->execute();
  1005. $result = $stmt->get_result();
  1006. $categories = [];
  1007. $quantities = [];
  1008. $revenues = [];
  1009. while ($row = $result->fetch_assoc()) {
  1010. $categories[] = $row['category_name'];
  1011. $quantities[] = $row['total_quantity'];
  1012. $revenues[] = $row['total_revenue'];
  1013. }
  1014. ?>
  1015. <div class="chart-container">
  1016. <div class="chart-header">
  1017. <h2 class="chart-title">产品类别销售分析</h2>
  1018. </div>
  1019. <canvas id="employeeProductChart"></canvas>
  1020. </div>
  1021. <script>
  1022. var employeeProductCtx = document.getElementById('employeeProductChart').getContext('2d');
  1023. new Chart(employeeProductCtx, {
  1024. type: 'bar',
  1025. data: {
  1026. labels: <?php echo json_encode($categories); ?>,
  1027. datasets: [
  1028. {
  1029. label: '销售数量',
  1030. data: <?php echo json_encode($quantities); ?>,
  1031. backgroundColor: 'rgba(54, 162, 235, 0.5)',
  1032. borderColor: 'rgba(54, 162, 235, 1)',
  1033. borderWidth: 1,
  1034. yAxisID: 'y-quantity'
  1035. },
  1036. {
  1037. label: '销售金额',
  1038. data: <?php echo json_encode($revenues); ?>,
  1039. backgroundColor: 'rgba(255, 99, 132, 0.5)',
  1040. borderColor: 'rgba(255, 99, 132, 1)',
  1041. borderWidth: 1,
  1042. yAxisID: 'y-revenue'
  1043. }
  1044. ]
  1045. },
  1046. options: {
  1047. responsive: true,
  1048. scales: {
  1049. 'y-quantity': {
  1050. type: 'linear',
  1051. position: 'left',
  1052. title: {
  1053. display: true,
  1054. text: '销售数量'
  1055. }
  1056. },
  1057. 'y-revenue': {
  1058. type: 'linear',
  1059. position: 'right',
  1060. title: {
  1061. display: true,
  1062. text: '销售金额'
  1063. },
  1064. grid: {
  1065. drawOnChartArea: false
  1066. }
  1067. }
  1068. }
  1069. }
  1070. });
  1071. </script>
  1072. <?php
  1073. }
  1074. /**
  1075. * 渲染业务员统计卡片
  1076. */
  1077. function renderEmployeeStats($employee_stats) {
  1078. // 添加空值检查函数
  1079. function formatNumber($value, $decimals = 0) {
  1080. return number_format($value ?? 0, $decimals);
  1081. }
  1082. function formatCurrency($value) {
  1083. return '¥' . number_format($value ?? 0, 2);
  1084. }
  1085. ?>
  1086. <div class="performance-grid">
  1087. <div class="performance-card">
  1088. <div class="performance-label">总订单数</div>
  1089. <div class="performance-value"><?php echo formatNumber($employee_stats['total_orders']); ?></div>
  1090. </div>
  1091. <div class="performance-card">
  1092. <div class="performance-label">总销售额</div>
  1093. <div class="performance-value"><?php echo formatCurrency($employee_stats['total_revenue']); ?></div>
  1094. </div>
  1095. <div class="performance-card">
  1096. <div class="performance-label">客户数量</div>
  1097. <div class="performance-value"><?php echo formatNumber($employee_stats['customer_count']); ?></div>
  1098. </div>
  1099. <div class="performance-card">
  1100. <div class="performance-label">平均订单金额</div>
  1101. <div class="performance-value"><?php echo formatCurrency($employee_stats['avg_order_value']); ?></div>
  1102. </div>
  1103. <div class="performance-card">
  1104. <div class="performance-label">订单完成率</div>
  1105. <div class="performance-value"><?php echo formatNumber($employee_stats['completion_rate'], 1); ?>%</div>
  1106. </div>
  1107. </div>
  1108. <?php
  1109. }