statistics.php 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011
  1. <?php
  2. require_once 'conn.php';
  3. checkLogin();
  4. //检查是否管理员
  5. checkPermissionDie(1,2);
  6. // 计算日期范围
  7. $current_month_start = date('Y-m-01');
  8. $current_month_end = date('Y-m-t');
  9. $last_month_start = date('Y-m-01', strtotime('-1 month'));
  10. $last_month_end = date('Y-m-t', strtotime('-1 month'));
  11. $current_year_start = date('Y-01-01');
  12. $current_year_end = date('Y-12-31');
  13. // 可选的日期范围筛选
  14. $date_range = isset($_GET['date_range']) ? $_GET['date_range'] : 'current_month';
  15. $custom_start = isset($_GET['start_date']) ? $_GET['start_date'] : '';
  16. $custom_end = isset($_GET['end_date']) ? $_GET['end_date'] : '';
  17. // 设置日期范围
  18. if ($date_range == 'custom' && !empty($custom_start) && !empty($custom_end)) {
  19. $start_date = $custom_start;
  20. $end_date = $custom_end;
  21. } else {
  22. switch ($date_range) {
  23. case 'last_month':
  24. $start_date = $last_month_start;
  25. $end_date = $last_month_end;
  26. break;
  27. case 'current_year':
  28. $start_date = $current_year_start;
  29. $end_date = $current_year_end;
  30. break;
  31. case 'last_30_days':
  32. $start_date = date('Y-m-d', strtotime('-30 days'));
  33. $end_date = date('Y-m-d');
  34. break;
  35. case 'last_90_days':
  36. $start_date = date('Y-m-d', strtotime('-90 days'));
  37. $end_date = date('Y-m-d');
  38. break;
  39. case 'current_month':
  40. default:
  41. $start_date = $current_month_start;
  42. $end_date = $current_month_end;
  43. break;
  44. }
  45. }
  46. // 格式化日期用于SQL查询
  47. $start_date_sql = date('Y-m-d', strtotime($start_date));
  48. $end_date_sql = date('Y-m-d', strtotime($end_date)) . ' 23:59:59';
  49. // 函数:获取销售概览数据
  50. function getSalesOverview($conn, $start_date, $end_date) {
  51. $sql = "SELECT
  52. COUNT(id) as total_orders,
  53. SUM(total_amount) as total_revenue,
  54. AVG(total_amount) as avg_order_value
  55. FROM orders
  56. WHERE order_date BETWEEN ? AND ?";
  57. $stmt = $conn->prepare($sql);
  58. $stmt->bind_param("ss", $start_date, $end_date);
  59. $stmt->execute();
  60. $result = $stmt->get_result();
  61. return $result->fetch_assoc();
  62. }
  63. // 函数:获取每月销售趋势
  64. function getMonthlySalesTrend($conn, $start_date, $end_date) {
  65. $sql = "SELECT
  66. DATE_FORMAT(order_date, '%Y-%m') as month,
  67. COUNT(id) as orders,
  68. SUM(total_amount) as revenue
  69. FROM orders
  70. WHERE order_date BETWEEN ? AND ?
  71. GROUP BY DATE_FORMAT(order_date, '%Y-%m')
  72. ORDER BY month";
  73. $stmt = $conn->prepare($sql);
  74. $stmt->bind_param("ss", $start_date, $end_date);
  75. $stmt->execute();
  76. return $stmt->get_result();
  77. }
  78. // 函数:获取客户国家分布
  79. function getCustomerCountryDistribution($conn) {
  80. $sql = "SELECT
  81. c.countryName,
  82. COUNT(cu.id) as customer_count
  83. FROM customer cu
  84. JOIN country c ON cu.cs_country = c.id
  85. GROUP BY cu.cs_country
  86. ORDER BY customer_count DESC
  87. LIMIT 10";
  88. return $conn->query($sql);
  89. }
  90. // 函数:获取客户类型分布
  91. function getCustomerTypeDistribution($conn) {
  92. $sql = "SELECT
  93. ct.businessType,
  94. COUNT(c.id) as customer_count
  95. FROM customer c
  96. JOIN clienttype ct ON c.cs_type = ct.id
  97. GROUP BY c.cs_type";
  98. return $conn->query($sql);
  99. }
  100. // 函数:获取成交阶段分布
  101. function getDealStageDistribution($conn) {
  102. $sql = "SELECT
  103. cs_deal,
  104. CASE
  105. WHEN cs_deal = 1 THEN '背景调查'
  106. WHEN cs_deal = 2 THEN '明确需求'
  107. WHEN cs_deal = 3 THEN '已成交'
  108. ELSE '其他'
  109. END as stage_name,
  110. COUNT(id) as customer_count
  111. FROM customer
  112. GROUP BY cs_deal";
  113. return $conn->query($sql);
  114. }
  115. // 函数:获取热门产品
  116. function getTopProducts($conn, $start_date, $end_date, $limit = 5) {
  117. $sql = "SELECT
  118. p.ProductName,
  119. SUM(oi.quantity) as total_quantity,
  120. SUM(oi.total_price) as total_revenue
  121. FROM order_items oi
  122. JOIN products p ON oi.product_id = p.id
  123. JOIN orders o ON oi.order_id = o.id
  124. WHERE o.order_date BETWEEN ? AND ?
  125. GROUP BY oi.product_id
  126. ORDER BY total_revenue DESC
  127. LIMIT ?";
  128. $stmt = $conn->prepare($sql);
  129. $stmt->bind_param("ssi", $start_date, $end_date, $limit);
  130. $stmt->execute();
  131. return $stmt->get_result();
  132. }
  133. // 函数:获取业务员销售业绩
  134. function getEmployeeSalesPerformance($conn, $start_date, $end_date) {
  135. $sql = "SELECT
  136. e.em_user as employee_name,
  137. COUNT(o.id) as order_count,
  138. SUM(o.total_amount) as total_sales
  139. FROM orders o
  140. JOIN employee e ON o.employee_id = e.id
  141. WHERE o.order_date BETWEEN ? AND ?
  142. GROUP BY o.employee_id
  143. ORDER BY total_sales DESC";
  144. $stmt = $conn->prepare($sql);
  145. $stmt->bind_param("ss", $start_date, $end_date);
  146. $stmt->execute();
  147. return $stmt->get_result();
  148. }
  149. // 函数:获取客户增长趋势
  150. function getCustomerGrowthTrend($conn) {
  151. $sql = "SELECT
  152. DATE_FORMAT(cs_addtime, '%Y-%m') as month,
  153. COUNT(id) as new_customers
  154. FROM customer
  155. WHERE cs_addtime >= DATE_SUB(CURDATE(), INTERVAL 12 MONTH)
  156. GROUP BY DATE_FORMAT(cs_addtime, '%Y-%m')
  157. ORDER BY month";
  158. return $conn->query($sql);
  159. }
  160. // 函数:获取不同地区的订单数量
  161. function getOrdersByRegion($conn, $start_date, $end_date) {
  162. $sql = "SELECT
  163. c.countryName,
  164. COUNT(o.id) as order_count,
  165. SUM(o.total_amount) as total_amount,
  166. SUM(oi.quantity) as total_quantity
  167. FROM orders o
  168. JOIN customer cu ON o.customer_id = cu.id
  169. JOIN country c ON cu.cs_country = c.id
  170. LEFT JOIN order_items oi ON o.id = oi.order_id
  171. WHERE o.order_date BETWEEN ? AND ?
  172. GROUP BY cu.cs_country
  173. ORDER BY total_quantity DESC
  174. LIMIT 10";
  175. $stmt = $conn->prepare($sql);
  176. $stmt->bind_param("ss", $start_date, $end_date);
  177. $stmt->execute();
  178. return $stmt->get_result();
  179. }
  180. // 函数:获取详细时间段订单数量
  181. function getDetailedOrderTrend($conn, $start_date, $end_date, $period = 'day') {
  182. $groupFormat = '%Y-%m-%d';
  183. $intervalUnit = 'DAY';
  184. if ($period == 'week') {
  185. $groupFormat = '%x-W%v'; // ISO year and week number
  186. $intervalUnit = 'WEEK';
  187. } else if ($period == 'month') {
  188. $groupFormat = '%Y-%m';
  189. $intervalUnit = 'MONTH';
  190. }
  191. $sql = "SELECT
  192. DATE_FORMAT(o.order_date, '$groupFormat') as time_period,
  193. COUNT(o.id) as order_count,
  194. SUM(oi.quantity) as total_quantity
  195. FROM orders o
  196. LEFT JOIN order_items oi ON o.id = oi.order_id
  197. WHERE o.order_date BETWEEN ? AND ?
  198. GROUP BY time_period
  199. ORDER BY MIN(o.order_date)";
  200. $stmt = $conn->prepare($sql);
  201. $stmt->bind_param("ss", $start_date, $end_date);
  202. $stmt->execute();
  203. return $stmt->get_result();
  204. }
  205. // 获取统计数据
  206. $sales_overview = getSalesOverview($conn, $start_date_sql, $end_date_sql);
  207. $monthly_sales = getMonthlySalesTrend($conn, $start_date_sql, $end_date_sql);
  208. $country_distribution = getCustomerCountryDistribution($conn);
  209. $customer_types = getCustomerTypeDistribution($conn);
  210. $deal_stages = getDealStageDistribution($conn);
  211. $top_products = getTopProducts($conn, $start_date_sql, $end_date_sql);
  212. $employee_performance = getEmployeeSalesPerformance($conn, $start_date_sql, $end_date_sql);
  213. $customer_growth = getCustomerGrowthTrend($conn);
  214. $orders_by_region = getOrdersByRegion($conn, $start_date_sql, $end_date_sql);
  215. // 获取详细时间段订单趋势 - 默认按日期
  216. $period = isset($_GET['period']) ? $_GET['period'] : 'day';
  217. $detailed_orders = getDetailedOrderTrend($conn, $start_date_sql, $end_date_sql, $period);
  218. // 将月度销售数据转换为图表所需格式
  219. $monthly_labels = [];
  220. $monthly_orders = [];
  221. $monthly_revenue = [];
  222. while ($row = $monthly_sales->fetch_assoc()) {
  223. $monthly_labels[] = $row['month'];
  224. $monthly_orders[] = $row['orders'];
  225. $monthly_revenue[] = $row['revenue'];
  226. }
  227. // 将国家分布数据转换为图表所需格式
  228. $country_labels = [];
  229. $country_data = [];
  230. while ($row = $country_distribution->fetch_assoc()) {
  231. $country_labels[] = $row['countryName'];
  232. $country_data[] = $row['customer_count'];
  233. }
  234. // 将客户类型数据转换为图表所需格式
  235. $type_labels = [];
  236. $type_data = [];
  237. while ($row = $customer_types->fetch_assoc()) {
  238. $type_labels[] = $row['businessType'];
  239. $type_data[] = $row['customer_count'];
  240. }
  241. // 将成交阶段数据转换为图表所需格式
  242. $stage_labels = [];
  243. $stage_data = [];
  244. while ($row = $deal_stages->fetch_assoc()) {
  245. $stage_labels[] = $row['stage_name'];
  246. $stage_data[] = $row['customer_count'];
  247. }
  248. // 将客户增长数据转换为图表所需格式
  249. $growth_labels = [];
  250. $growth_data = [];
  251. while ($row = $customer_growth->fetch_assoc()) {
  252. $growth_labels[] = $row['month'];
  253. $growth_data[] = $row['new_customers'];
  254. }
  255. // 将地区订单数据转换为图表所需格式
  256. $region_labels = [];
  257. $region_orders = [];
  258. $region_quantities = [];
  259. while ($row = $orders_by_region->fetch_assoc()) {
  260. $region_labels[] = $row['countryName'];
  261. $region_orders[] = $row['order_count'];
  262. $region_quantities[] = $row['total_quantity'];
  263. }
  264. // 将详细时间订单数据转换为图表所需格式
  265. $time_labels = [];
  266. $time_orders = [];
  267. $time_quantities = [];
  268. while ($row = $detailed_orders->fetch_assoc()) {
  269. $time_labels[] = $row['time_period'];
  270. $time_orders[] = $row['order_count'];
  271. $time_quantities[] = $row['total_quantity'];
  272. }
  273. ?>
  274. <!DOCTYPE html>
  275. <html xmlns="http://www.w3.org/1999/xhtml">
  276. <head>
  277. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  278. <title>统计分析</title>
  279. <link rel="stylesheet" href="css/common.css" type="text/css" />
  280. <script src="system/js/jquery-1.7.2.min.js"></script>
  281. <script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
  282. <style>
  283. body {
  284. margin: 0;
  285. padding: 20px;
  286. background: #fff;
  287. font-family: Arial, sans-serif;
  288. }
  289. .container {
  290. width: 100%;
  291. max-width: 1200px;
  292. margin: 0 auto;
  293. }
  294. .page-header {
  295. margin-bottom: 20px;
  296. border-bottom: 1px solid #eee;
  297. padding-bottom: 10px;
  298. display: flex;
  299. justify-content: space-between;
  300. align-items: center;
  301. }
  302. .page-title {
  303. font-size: 24px;
  304. font-weight: bold;
  305. color: #333;
  306. margin: 0;
  307. }
  308. .export-btn {
  309. padding: 8px 15px;
  310. background: #4CAF50;
  311. color: white;
  312. border: none;
  313. border-radius: 4px;
  314. cursor: pointer;
  315. text-decoration: none;
  316. display: inline-block;
  317. }
  318. .export-btn:hover {
  319. background: #45a049;
  320. }
  321. .filter-form {
  322. background: #f9f9f9;
  323. padding: 15px;
  324. border-radius: 5px;
  325. margin-bottom: 20px;
  326. display: flex;
  327. flex-wrap: wrap;
  328. gap: 15px;
  329. align-items: center;
  330. }
  331. .form-group {
  332. margin-right: 15px;
  333. }
  334. .form-group label {
  335. display: block;
  336. margin-bottom: 5px;
  337. font-weight: bold;
  338. }
  339. .form-control {
  340. padding: 8px;
  341. border: 1px solid #ddd;
  342. border-radius: 4px;
  343. min-width: 150px;
  344. }
  345. .btn {
  346. padding: 8px 15px;
  347. background: #337ab7;
  348. color: white;
  349. border: none;
  350. border-radius: 4px;
  351. cursor: pointer;
  352. }
  353. .btn:hover {
  354. background: #286090;
  355. }
  356. .stats-grid {
  357. display: grid;
  358. grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
  359. gap: 20px;
  360. margin-bottom: 20px;
  361. }
  362. .stat-card {
  363. background: white;
  364. border-radius: 5px;
  365. box-shadow: 0 2px 4px rgba(0,0,0,0.1);
  366. padding: 20px;
  367. }
  368. .stat-card h3 {
  369. margin-top: 0;
  370. color: #555;
  371. font-size: 16px;
  372. border-bottom: 1px solid #eee;
  373. padding-bottom: 10px;
  374. }
  375. .stat-value {
  376. font-size: 24px;
  377. font-weight: bold;
  378. color: #333;
  379. margin: 15px 0;
  380. }
  381. .chart-container {
  382. margin-bottom: 30px;
  383. background: white;
  384. border-radius: 5px;
  385. box-shadow: 0 2px 4px rgba(0,0,0,0.1);
  386. padding: 20px;
  387. }
  388. .chart-header {
  389. display: flex;
  390. justify-content: space-between;
  391. align-items: center;
  392. margin-bottom: 15px;
  393. }
  394. .chart-title {
  395. font-size: 18px;
  396. font-weight: bold;
  397. color: #333;
  398. margin: 0;
  399. }
  400. .data-table {
  401. width: 100%;
  402. border-collapse: collapse;
  403. margin-top: 20px;
  404. }
  405. .data-table th, .data-table td {
  406. padding: 10px;
  407. text-align: left;
  408. border-bottom: 1px solid #eee;
  409. }
  410. .data-table th {
  411. background: #f5f5f5;
  412. font-weight: bold;
  413. }
  414. .data-table tr:hover {
  415. background: #f9f9f9;
  416. }
  417. .custom-date-inputs {
  418. display: none;
  419. }
  420. #date_range[value="custom"]:checked ~ .custom-date-inputs {
  421. display: block;
  422. }
  423. .download-btn {
  424. padding: 5px 10px;
  425. background: #4CAF50;
  426. color: white;
  427. border: none;
  428. border-radius: 3px;
  429. font-size: 14px;
  430. cursor: pointer;
  431. }
  432. .download-btn:hover {
  433. background: #45a049;
  434. }
  435. /* 确保筛选表单元素在同一行显示 */
  436. .filter-form form {
  437. display: flex;
  438. flex-direction: row;
  439. align-items: flex-end;
  440. gap: 15px;
  441. flex-wrap: nowrap;
  442. width: 100%;
  443. }
  444. .filter-form .form-group {
  445. margin-right: 0;
  446. flex: 0 0 auto;
  447. white-space: nowrap;
  448. }
  449. .filter-form label {
  450. display: block;
  451. margin-bottom: 5px;
  452. font-weight: bold;
  453. }
  454. .filter-form #custom_dates {
  455. display: flex;
  456. flex-direction: row;
  457. gap: 10px;
  458. flex: 0 0 auto;
  459. }
  460. </style>
  461. </head>
  462. <body>
  463. <div class="container">
  464. <div class="page-header">
  465. <h1 class="page-title">统计分析</h1>
  466. <a href="export_statistics.php?date_range=<?php echo $date_range; ?>&start_date=<?php echo $custom_start; ?>&end_date=<?php echo $custom_end; ?>&period=<?php echo $period; ?>" class="export-btn">导出数据</a>
  467. </div>
  468. <div class="filter-form">
  469. <form method="get" action="">
  470. <div class="form-group">
  471. <label>日期范围</label>
  472. <select name="date_range" id="date_range" class="form-control" onchange="toggleCustomDates()">
  473. <option value="current_month" <?php echo $date_range == 'current_month' ? 'selected' : ''; ?>>本月</option>
  474. <option value="last_month" <?php echo $date_range == 'last_month' ? 'selected' : ''; ?>>上月</option>
  475. <option value="last_30_days" <?php echo $date_range == 'last_30_days' ? 'selected' : ''; ?>>过去30天</option>
  476. <option value="last_90_days" <?php echo $date_range == 'last_90_days' ? 'selected' : ''; ?>>过去90天</option>
  477. <option value="current_year" <?php echo $date_range == 'current_year' ? 'selected' : ''; ?>>今年</option>
  478. <option value="custom" <?php echo $date_range == 'custom' ? 'selected' : ''; ?>>自定义</option>
  479. </select>
  480. </div>
  481. <div id="custom_dates" style="display: <?php echo $date_range == 'custom' ? 'flex' : 'none'; ?>;">
  482. <div class="form-group">
  483. <label>开始日期</label>
  484. <input type="date" name="start_date" class="form-control" value="<?php echo $custom_start; ?>">
  485. </div>
  486. <div class="form-group">
  487. <label>结束日期</label>
  488. <input type="date" name="end_date" class="form-control" value="<?php echo $custom_end; ?>">
  489. </div>
  490. </div>
  491. <div class="form-group">
  492. <label>时间粒度</label>
  493. <select name="period" class="form-control">
  494. <option value="day" <?php echo $period == 'day' ? 'selected' : ''; ?>>按日</option>
  495. <option value="week" <?php echo $period == 'week' ? 'selected' : ''; ?>>按周</option>
  496. <option value="month" <?php echo $period == 'month' ? 'selected' : ''; ?>>按月</option>
  497. </select>
  498. </div>
  499. <div class="form-group">
  500. <input type="submit" class="btn" value="应用筛选">
  501. </div>
  502. </form>
  503. </div>
  504. <!-- 销售概览卡片 -->
  505. <div class="stats-grid">
  506. <div class="stat-card">
  507. <h3>总订单数</h3>
  508. <div class="stat-value"><?php echo number_format($sales_overview['total_orders']); ?></div>
  509. </div>
  510. <div class="stat-card">
  511. <h3>总收入</h3>
  512. <div class="stat-value">¥<?php echo number_format($sales_overview['total_revenue'], 2); ?></div>
  513. </div>
  514. <div class="stat-card">
  515. <h3>平均订单金额</h3>
  516. <div class="stat-value">¥<?php echo number_format($sales_overview['avg_order_value'], 2); ?></div>
  517. </div>
  518. </div>
  519. <!-- 月度销售趋势图 -->
  520. <div class="chart-container">
  521. <div class="chart-header">
  522. <h2 class="chart-title">销售趋势</h2>
  523. </div>
  524. <canvas id="salesTrendChart"></canvas>
  525. </div>
  526. <!-- 客户分布图 -->
  527. <div class="stats-grid">
  528. <div class="chart-container">
  529. <div class="chart-header">
  530. <h2 class="chart-title">客户国家分布</h2>
  531. </div>
  532. <canvas id="countryDistributionChart"></canvas>
  533. </div>
  534. <div class="chart-container">
  535. <div class="chart-header">
  536. <h2 class="chart-title">客户类型分布</h2>
  537. </div>
  538. <canvas id="customerTypeChart"></canvas>
  539. </div>
  540. </div>
  541. <div class="stats-grid">
  542. <div class="chart-container">
  543. <div class="chart-header">
  544. <h2 class="chart-title">成交阶段分布</h2>
  545. </div>
  546. <canvas id="dealStageChart"></canvas>
  547. </div>
  548. <div class="chart-container">
  549. <div class="chart-header">
  550. <h2 class="chart-title">客户增长趋势</h2>
  551. </div>
  552. <canvas id="customerGrowthChart"></canvas>
  553. </div>
  554. </div>
  555. <!-- 地区订单分析 -->
  556. <div class="chart-container">
  557. <div class="chart-header">
  558. <h2 class="chart-title">地区订单分析</h2>
  559. </div>
  560. <canvas id="regionOrdersChart"></canvas>
  561. </div>
  562. <!-- 详细时间段订单趋势 -->
  563. <div class="chart-container">
  564. <div class="chart-header">
  565. <h2 class="chart-title">详细订单趋势 (<?php echo $period == 'day' ? '日' : ($period == 'week' ? '周' : '月'); ?>)</h2>
  566. </div>
  567. <canvas id="detailedOrdersChart"></canvas>
  568. </div>
  569. <!-- 热门产品表格 -->
  570. <div class="chart-container">
  571. <div class="chart-header">
  572. <h2 class="chart-title">热门产品</h2>
  573. </div>
  574. <table class="data-table">
  575. <thead>
  576. <tr>
  577. <th>产品名称</th>
  578. <th>销售数量</th>
  579. <th>销售收入</th>
  580. </tr>
  581. </thead>
  582. <tbody>
  583. <?php while ($row = $top_products->fetch_assoc()): ?>
  584. <tr>
  585. <td><?php echo htmlspecialchars($row['ProductName']); ?></td>
  586. <td><?php echo number_format($row['total_quantity']); ?></td>
  587. <td>¥<?php echo number_format($row['total_revenue'], 2); ?></td>
  588. </tr>
  589. <?php endwhile; ?>
  590. </tbody>
  591. </table>
  592. </div>
  593. <!-- 业务员销售业绩表格 -->
  594. <div class="chart-container">
  595. <div class="chart-header">
  596. <h2 class="chart-title">业务员销售业绩</h2>
  597. </div>
  598. <table class="data-table">
  599. <thead>
  600. <tr>
  601. <th>业务员姓名</th>
  602. <th>订单数量</th>
  603. <th>销售总额</th>
  604. </tr>
  605. </thead>
  606. <tbody>
  607. <?php while ($row = $employee_performance->fetch_assoc()): ?>
  608. <tr>
  609. <td><?php echo htmlspecialchars($row['employee_name']); ?></td>
  610. <td><?php echo number_format($row['order_count']); ?></td>
  611. <td>¥<?php echo number_format($row['total_sales'], 2); ?></td>
  612. </tr>
  613. <?php endwhile; ?>
  614. </tbody>
  615. </table>
  616. </div>
  617. </div>
  618. <script>
  619. // 切换自定义日期区域显示
  620. function toggleCustomDates() {
  621. var dateRange = document.getElementById('date_range').value;
  622. var customDates = document.getElementById('custom_dates');
  623. if (dateRange === 'custom') {
  624. customDates.style.display = 'flex';
  625. } else {
  626. customDates.style.display = 'none';
  627. }
  628. }
  629. // 销售趋势图
  630. var salesTrendCtx = document.getElementById('salesTrendChart').getContext('2d');
  631. var salesTrendChart = new Chart(salesTrendCtx, {
  632. type: 'line',
  633. data: {
  634. labels: <?php echo json_encode($monthly_labels); ?>,
  635. datasets: [
  636. {
  637. label: '订单数量',
  638. data: <?php echo json_encode($monthly_orders); ?>,
  639. backgroundColor: 'rgba(54, 162, 235, 0.2)',
  640. borderColor: 'rgba(54, 162, 235, 1)',
  641. borderWidth: 2,
  642. yAxisID: 'y-orders'
  643. },
  644. {
  645. label: '销售收入',
  646. data: <?php echo json_encode($monthly_revenue); ?>,
  647. backgroundColor: 'rgba(255, 99, 132, 0.2)',
  648. borderColor: 'rgba(255, 99, 132, 1)',
  649. borderWidth: 2,
  650. yAxisID: 'y-revenue'
  651. }
  652. ]
  653. },
  654. options: {
  655. responsive: true,
  656. scales: {
  657. 'y-orders': {
  658. type: 'linear',
  659. position: 'left',
  660. title: {
  661. display: true,
  662. text: '订单数量'
  663. }
  664. },
  665. 'y-revenue': {
  666. type: 'linear',
  667. position: 'right',
  668. title: {
  669. display: true,
  670. text: '销售收入'
  671. },
  672. grid: {
  673. drawOnChartArea: false
  674. }
  675. }
  676. }
  677. }
  678. });
  679. // 客户国家分布图
  680. var countryDistributionCtx = document.getElementById('countryDistributionChart').getContext('2d');
  681. var countryDistributionChart = new Chart(countryDistributionCtx, {
  682. type: 'pie',
  683. data: {
  684. labels: <?php echo json_encode($country_labels); ?>,
  685. datasets: [{
  686. data: <?php echo json_encode($country_data); ?>,
  687. backgroundColor: [
  688. 'rgba(255, 99, 132, 0.7)',
  689. 'rgba(54, 162, 235, 0.7)',
  690. 'rgba(255, 206, 86, 0.7)',
  691. 'rgba(75, 192, 192, 0.7)',
  692. 'rgba(153, 102, 255, 0.7)',
  693. 'rgba(255, 159, 64, 0.7)',
  694. 'rgba(199, 199, 199, 0.7)',
  695. 'rgba(83, 102, 255, 0.7)',
  696. 'rgba(40, 159, 64, 0.7)',
  697. 'rgba(210, 199, 199, 0.7)'
  698. ],
  699. borderWidth: 1
  700. }]
  701. },
  702. options: {
  703. responsive: true,
  704. plugins: {
  705. legend: {
  706. position: 'right',
  707. }
  708. }
  709. }
  710. });
  711. // 客户类型分布图
  712. var customerTypeCtx = document.getElementById('customerTypeChart').getContext('2d');
  713. var customerTypeChart = new Chart(customerTypeCtx, {
  714. type: 'doughnut',
  715. data: {
  716. labels: <?php echo json_encode($type_labels); ?>,
  717. datasets: [{
  718. data: <?php echo json_encode($type_data); ?>,
  719. backgroundColor: [
  720. 'rgba(54, 162, 235, 0.7)',
  721. 'rgba(255, 99, 132, 0.7)',
  722. 'rgba(255, 206, 86, 0.7)',
  723. 'rgba(75, 192, 192, 0.7)',
  724. 'rgba(153, 102, 255, 0.7)'
  725. ],
  726. borderWidth: 1
  727. }]
  728. },
  729. options: {
  730. responsive: true,
  731. plugins: {
  732. legend: {
  733. position: 'right',
  734. }
  735. }
  736. }
  737. });
  738. // 成交阶段分布图
  739. var dealStageCtx = document.getElementById('dealStageChart').getContext('2d');
  740. var dealStageChart = new Chart(dealStageCtx, {
  741. type: 'bar',
  742. data: {
  743. labels: <?php echo json_encode($stage_labels); ?>,
  744. datasets: [{
  745. label: '客户数量',
  746. data: <?php echo json_encode($stage_data); ?>,
  747. backgroundColor: [
  748. 'rgba(255, 206, 86, 0.7)',
  749. 'rgba(54, 162, 235, 0.7)',
  750. 'rgba(255, 99, 132, 0.7)'
  751. ],
  752. borderWidth: 1
  753. }]
  754. },
  755. options: {
  756. responsive: true,
  757. scales: {
  758. y: {
  759. beginAtZero: true,
  760. title: {
  761. display: true,
  762. text: '客户数量'
  763. }
  764. }
  765. }
  766. }
  767. });
  768. // 客户增长趋势图
  769. var customerGrowthCtx = document.getElementById('customerGrowthChart').getContext('2d');
  770. var customerGrowthChart = new Chart(customerGrowthCtx, {
  771. type: 'line',
  772. data: {
  773. labels: <?php echo json_encode($growth_labels); ?>,
  774. datasets: [{
  775. label: '新增客户',
  776. data: <?php echo json_encode($growth_data); ?>,
  777. backgroundColor: 'rgba(75, 192, 192, 0.2)',
  778. borderColor: 'rgba(75, 192, 192, 1)',
  779. borderWidth: 2,
  780. tension: 0.1
  781. }]
  782. },
  783. options: {
  784. responsive: true,
  785. scales: {
  786. y: {
  787. beginAtZero: true,
  788. title: {
  789. display: true,
  790. text: '客户数量'
  791. }
  792. }
  793. }
  794. }
  795. });
  796. // 地区订单分析图
  797. var regionOrdersCtx = document.getElementById('regionOrdersChart').getContext('2d');
  798. var regionOrdersChart = new Chart(regionOrdersCtx, {
  799. type: 'bar',
  800. data: {
  801. labels: <?php echo json_encode($region_labels); ?>,
  802. datasets: [
  803. {
  804. label: '订单数量',
  805. data: <?php echo json_encode($region_orders); ?>,
  806. backgroundColor: 'rgba(54, 162, 235, 0.6)',
  807. borderColor: 'rgba(54, 162, 235, 1)',
  808. borderWidth: 1,
  809. yAxisID: 'y-orders'
  810. },
  811. {
  812. label: '产品订购数量',
  813. data: <?php echo json_encode($region_quantities); ?>,
  814. backgroundColor: 'rgba(255, 99, 132, 0.6)',
  815. borderColor: 'rgba(255, 99, 132, 1)',
  816. borderWidth: 1,
  817. yAxisID: 'y-quantity'
  818. }
  819. ]
  820. },
  821. options: {
  822. responsive: true,
  823. scales: {
  824. x: {
  825. title: {
  826. display: true,
  827. text: '地区'
  828. }
  829. },
  830. 'y-orders': {
  831. type: 'linear',
  832. position: 'left',
  833. title: {
  834. display: true,
  835. text: '订单数量'
  836. },
  837. beginAtZero: true
  838. },
  839. 'y-quantity': {
  840. type: 'linear',
  841. position: 'right',
  842. title: {
  843. display: true,
  844. text: '产品订购数量'
  845. },
  846. beginAtZero: true,
  847. grid: {
  848. drawOnChartArea: false
  849. }
  850. }
  851. }
  852. }
  853. });
  854. // 详细时间段订单趋势图
  855. var detailedOrdersCtx = document.getElementById('detailedOrdersChart').getContext('2d');
  856. var detailedOrdersChart = new Chart(detailedOrdersCtx, {
  857. type: 'line',
  858. data: {
  859. labels: <?php echo json_encode($time_labels); ?>,
  860. datasets: [
  861. {
  862. label: '订单数量',
  863. data: <?php echo json_encode($time_orders); ?>,
  864. backgroundColor: 'rgba(75, 192, 192, 0.2)',
  865. borderColor: 'rgba(75, 192, 192, 1)',
  866. borderWidth: 2,
  867. yAxisID: 'y-orders',
  868. tension: 0.1
  869. },
  870. {
  871. label: '产品订购数量',
  872. data: <?php echo json_encode($time_quantities); ?>,
  873. backgroundColor: 'rgba(255, 159, 64, 0.2)',
  874. borderColor: 'rgba(255, 159, 64, 1)',
  875. borderWidth: 2,
  876. yAxisID: 'y-quantity',
  877. tension: 0.1
  878. }
  879. ]
  880. },
  881. options: {
  882. responsive: true,
  883. scales: {
  884. x: {
  885. title: {
  886. display: true,
  887. text: '时间'
  888. }
  889. },
  890. 'y-orders': {
  891. type: 'linear',
  892. position: 'left',
  893. title: {
  894. display: true,
  895. text: '订单数量'
  896. },
  897. beginAtZero: true
  898. },
  899. 'y-quantity': {
  900. type: 'linear',
  901. position: 'right',
  902. title: {
  903. display: true,
  904. text: '产品订购数量'
  905. },
  906. beginAtZero: true,
  907. grid: {
  908. drawOnChartArea: false
  909. }
  910. }
  911. }
  912. }
  913. });
  914. </script>
  915. </body>
  916. </html>