statistics_customers.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. <?php
  2. /**
  3. * 客户统计分析模块
  4. *
  5. * 包含与客户相关的数据分析功能
  6. */
  7. require_once 'statistics_utils.php';
  8. /**
  9. * 获取客户类型分布
  10. *
  11. * @param mysqli $conn 数据库连接
  12. * @return mysqli_result 客户类型分布数据结果集
  13. */
  14. function getCustomerTypeDistribution($conn) {
  15. $sql = "SELECT
  16. ct.businessType,
  17. COUNT(c.id) as customer_count
  18. FROM customer c
  19. JOIN clienttype ct ON c.cs_type = ct.id
  20. GROUP BY c.cs_type";
  21. return $conn->query($sql);
  22. }
  23. /**
  24. * 获取成交阶段分布
  25. *
  26. * @param mysqli $conn 数据库连接
  27. * @return mysqli_result 成交阶段分布数据结果集
  28. */
  29. function getDealStageDistribution($conn) {
  30. $sql = "SELECT
  31. cs_deal,
  32. CASE
  33. WHEN cs_deal = 1 THEN '背景调查'
  34. WHEN cs_deal = 2 THEN '明确需求'
  35. WHEN cs_deal = 3 THEN '已成交'
  36. ELSE '其他'
  37. END as stage_name,
  38. COUNT(id) as customer_count
  39. FROM customer
  40. GROUP BY cs_deal";
  41. return $conn->query($sql);
  42. }
  43. /**
  44. * 获取客户增长趋势
  45. *
  46. * @param mysqli $conn 数据库连接
  47. * @param int $months 获取多少个月的数据,默认12个月
  48. * @return mysqli_result 客户增长趋势数据结果集
  49. */
  50. function getCustomerGrowthTrend($conn, $months = 12) {
  51. $sql = "SELECT
  52. DATE_FORMAT(cs_addtime, '%Y-%m') as month,
  53. COUNT(id) as new_customers
  54. FROM customer
  55. WHERE cs_addtime >= DATE_SUB(CURDATE(), INTERVAL ? MONTH)
  56. GROUP BY DATE_FORMAT(cs_addtime, '%Y-%m')
  57. ORDER BY month";
  58. $stmt = $conn->prepare($sql);
  59. $stmt->bind_param("i", $months);
  60. $stmt->execute();
  61. return $stmt->get_result();
  62. }
  63. /**
  64. * 获取新老客户订单分析
  65. *
  66. * @param mysqli $conn 数据库连接
  67. * @param string $start_date 开始日期
  68. * @param string $end_date 结束日期
  69. * @return array 新老客户订单分析数据
  70. */
  71. function getNewVsReturningCustomerOrders($conn, $start_date, $end_date) {
  72. // 获取选定日期范围内的订单
  73. $sql = "SELECT
  74. o.customer_id,
  75. COUNT(o.id) as order_count,
  76. SUM(o.total_amount) as total_amount,
  77. MIN(o.order_date) as first_order_date,
  78. MAX(c.cs_addtime) as customer_addtime
  79. FROM orders o
  80. JOIN customer c ON o.customer_id = c.id
  81. WHERE o.order_date BETWEEN ? AND ?
  82. GROUP BY o.customer_id";
  83. $stmt = $conn->prepare($sql);
  84. $stmt->bind_param("ss", $start_date, $end_date);
  85. $stmt->execute();
  86. $result = $stmt->get_result();
  87. $new_customers = 0;
  88. $returning_customers = 0;
  89. $new_customer_amount = 0;
  90. $returning_customer_amount = 0;
  91. while ($row = $result->fetch_assoc()) {
  92. // 查找之前是否有订单
  93. $prev_sql = "SELECT id FROM orders
  94. WHERE customer_id = ?
  95. AND order_date < ?
  96. LIMIT 1";
  97. $prev_stmt = $conn->prepare($prev_sql);
  98. $prev_stmt->bind_param("is", $row['customer_id'], $start_date);
  99. $prev_stmt->execute();
  100. $prev_result = $prev_stmt->get_result();
  101. if ($prev_result->num_rows > 0) {
  102. // 老客户
  103. $returning_customers++;
  104. $returning_customer_amount += $row['total_amount'];
  105. } else {
  106. // 新客户
  107. $new_customers++;
  108. $new_customer_amount += $row['total_amount'];
  109. }
  110. }
  111. return [
  112. 'new_customers' => $new_customers,
  113. 'returning_customers' => $returning_customers,
  114. 'new_customer_amount' => $new_customer_amount,
  115. 'returning_customer_amount' => $returning_customer_amount,
  116. 'total_customers' => $new_customers + $returning_customers,
  117. 'total_amount' => $new_customer_amount + $returning_customer_amount
  118. ];
  119. }
  120. /**
  121. * 渲染客户类型分布图
  122. *
  123. * @param array $type_labels 类型标签
  124. * @param array $type_data 类型数据
  125. * @return void
  126. */
  127. function renderCustomerTypeChart($type_labels, $type_data) {
  128. ?>
  129. <div class="chart-container">
  130. <div class="chart-header">
  131. <h2 class="chart-title">客户类型分布</h2>
  132. </div>
  133. <canvas id="customerTypeChart"></canvas>
  134. </div>
  135. <script>
  136. // 客户类型分布图
  137. var customerTypeCtx = document.getElementById('customerTypeChart').getContext('2d');
  138. var customerTypeChart = new Chart(customerTypeCtx, {
  139. type: 'doughnut',
  140. data: {
  141. labels: <?php echo json_encode($type_labels); ?>,
  142. datasets: [{
  143. data: <?php echo json_encode($type_data); ?>,
  144. backgroundColor: [
  145. 'rgba(54, 162, 235, 0.7)',
  146. 'rgba(255, 99, 132, 0.7)',
  147. 'rgba(255, 206, 86, 0.7)',
  148. 'rgba(75, 192, 192, 0.7)',
  149. 'rgba(153, 102, 255, 0.7)'
  150. ],
  151. borderWidth: 1
  152. }]
  153. },
  154. options: {
  155. responsive: true,
  156. plugins: {
  157. legend: {
  158. position: 'right',
  159. }
  160. }
  161. }
  162. });
  163. </script>
  164. <?php
  165. }
  166. /**
  167. * 渲染成交阶段分布图
  168. *
  169. * @param array $stage_labels 阶段标签
  170. * @param array $stage_data 阶段数据
  171. * @return void
  172. */
  173. function renderDealStageChart($stage_labels, $stage_data) {
  174. ?>
  175. <div class="chart-container">
  176. <div class="chart-header">
  177. <h2 class="chart-title">成交阶段分布</h2>
  178. </div>
  179. <canvas id="dealStageChart"></canvas>
  180. </div>
  181. <script>
  182. // 成交阶段分布图
  183. var dealStageCtx = document.getElementById('dealStageChart').getContext('2d');
  184. var dealStageChart = new Chart(dealStageCtx, {
  185. type: 'bar',
  186. data: {
  187. labels: <?php echo json_encode($stage_labels); ?>,
  188. datasets: [{
  189. label: '客户数量',
  190. data: <?php echo json_encode($stage_data); ?>,
  191. backgroundColor: [
  192. 'rgba(255, 206, 86, 0.7)',
  193. 'rgba(54, 162, 235, 0.7)',
  194. 'rgba(255, 99, 132, 0.7)'
  195. ],
  196. borderWidth: 1
  197. }]
  198. },
  199. options: {
  200. responsive: true,
  201. scales: {
  202. y: {
  203. beginAtZero: true,
  204. title: {
  205. display: true,
  206. text: '客户数量'
  207. }
  208. }
  209. }
  210. }
  211. });
  212. </script>
  213. <?php
  214. }
  215. /**
  216. * 渲染客户增长趋势图
  217. *
  218. * @param array $growth_labels 增长标签
  219. * @param array $growth_data 增长数据
  220. * @return void
  221. */
  222. function renderCustomerGrowthChart($growth_labels, $growth_data) {
  223. ?>
  224. <div class="chart-container">
  225. <div class="chart-header">
  226. <h2 class="chart-title">客户增长趋势</h2>
  227. </div>
  228. <canvas id="customerGrowthChart"></canvas>
  229. </div>
  230. <script>
  231. // 客户增长趋势图
  232. var customerGrowthCtx = document.getElementById('customerGrowthChart').getContext('2d');
  233. var customerGrowthChart = new Chart(customerGrowthCtx, {
  234. type: 'line',
  235. data: {
  236. labels: <?php echo json_encode($growth_labels); ?>,
  237. datasets: [{
  238. label: '新增客户',
  239. data: <?php echo json_encode($growth_data); ?>,
  240. backgroundColor: 'rgba(75, 192, 192, 0.2)',
  241. borderColor: 'rgba(75, 192, 192, 1)',
  242. borderWidth: 2,
  243. tension: 0.1
  244. }]
  245. },
  246. options: {
  247. responsive: true,
  248. scales: {
  249. y: {
  250. beginAtZero: true,
  251. title: {
  252. display: true,
  253. text: '客户数量'
  254. }
  255. }
  256. }
  257. }
  258. });
  259. </script>
  260. <?php
  261. }
  262. /**
  263. * 渲染新老客户分析图
  264. *
  265. * @param array $new_vs_returning 新老客户数据
  266. * @return void
  267. */
  268. function renderNewVsReturningCustomersChart($new_vs_returning) {
  269. ?>
  270. <div class="chart-container">
  271. <div class="chart-header">
  272. <h2 class="chart-title">新老客户分析</h2>
  273. </div>
  274. <div class="chart-row">
  275. <div class="chart-column">
  276. <canvas id="newVsReturningCustomersChart"></canvas>
  277. </div>
  278. <div class="chart-column">
  279. <canvas id="newVsReturningAmountChart"></canvas>
  280. </div>
  281. </div>
  282. <div class="customer-stats-summary">
  283. <div class="stats-row">
  284. <div class="stat-item">
  285. <span class="stat-label">总客户数:</span>
  286. <span class="stat-value"><?php echo number_format($new_vs_returning['total_customers']); ?></span>
  287. </div>
  288. <div class="stat-item">
  289. <span class="stat-label">新客户:</span>
  290. <span class="stat-value"><?php echo number_format($new_vs_returning['new_customers']); ?>
  291. (<?php echo number_format(($new_vs_returning['new_customers'] / $new_vs_returning['total_customers']) * 100, 1); ?>%)</span>
  292. </div>
  293. <div class="stat-item">
  294. <span class="stat-label">老客户:</span>
  295. <span class="stat-value"><?php echo number_format($new_vs_returning['returning_customers']); ?>
  296. (<?php echo number_format(($new_vs_returning['returning_customers'] / $new_vs_returning['total_customers']) * 100, 1); ?>%)</span>
  297. </div>
  298. </div>
  299. <div class="stats-row">
  300. <div class="stat-item">
  301. <span class="stat-label">总销售额:</span>
  302. <span class="stat-value">¥<?php echo number_format($new_vs_returning['total_amount'], 2); ?></span>
  303. </div>
  304. <div class="stat-item">
  305. <span class="stat-label">新客户销售额:</span>
  306. <span class="stat-value">¥<?php echo number_format($new_vs_returning['new_customer_amount'], 2); ?>
  307. (<?php echo number_format(($new_vs_returning['new_customer_amount'] / $new_vs_returning['total_amount']) * 100, 1); ?>%)</span>
  308. </div>
  309. <div class="stat-item">
  310. <span class="stat-label">老客户销售额:</span>
  311. <span class="stat-value">¥<?php echo number_format($new_vs_returning['returning_customer_amount'], 2); ?>
  312. (<?php echo number_format(($new_vs_returning['returning_customer_amount'] / $new_vs_returning['total_amount']) * 100, 1); ?>%)</span>
  313. </div>
  314. </div>
  315. </div>
  316. </div>
  317. <script>
  318. // 新老客户数量图
  319. var newVsReturningCtx = document.getElementById('newVsReturningCustomersChart').getContext('2d');
  320. var newVsReturningChart = new Chart(newVsReturningCtx, {
  321. type: 'pie',
  322. data: {
  323. labels: ['新客户', '老客户'],
  324. datasets: [{
  325. data: [
  326. <?php echo $new_vs_returning['new_customers']; ?>,
  327. <?php echo $new_vs_returning['returning_customers']; ?>
  328. ],
  329. backgroundColor: [
  330. 'rgba(54, 162, 235, 0.7)',
  331. 'rgba(255, 99, 132, 0.7)'
  332. ],
  333. borderWidth: 1
  334. }]
  335. },
  336. options: {
  337. responsive: true,
  338. plugins: {
  339. legend: {
  340. position: 'bottom',
  341. },
  342. title: {
  343. display: true,
  344. text: '客户数量分布'
  345. }
  346. }
  347. }
  348. });
  349. // 新老客户销售额图
  350. var amountCtx = document.getElementById('newVsReturningAmountChart').getContext('2d');
  351. var amountChart = new Chart(amountCtx, {
  352. type: 'pie',
  353. data: {
  354. labels: ['新客户销售额', '老客户销售额'],
  355. datasets: [{
  356. data: [
  357. <?php echo $new_vs_returning['new_customer_amount']; ?>,
  358. <?php echo $new_vs_returning['returning_customer_amount']; ?>
  359. ],
  360. backgroundColor: [
  361. 'rgba(54, 162, 235, 0.7)',
  362. 'rgba(255, 99, 132, 0.7)'
  363. ],
  364. borderWidth: 1
  365. }]
  366. },
  367. options: {
  368. responsive: true,
  369. plugins: {
  370. legend: {
  371. position: 'bottom',
  372. },
  373. title: {
  374. display: true,
  375. text: '销售额分布'
  376. }
  377. }
  378. }
  379. });
  380. </script>
  381. <style>
  382. .chart-row {
  383. display: flex;
  384. flex-wrap: wrap;
  385. }
  386. .chart-column {
  387. flex: 0 0 50%;
  388. max-width: 50%;
  389. }
  390. .customer-stats-summary {
  391. margin-top: 20px;
  392. padding: 15px;
  393. background-color: #f9f9f9;
  394. border-radius: 5px;
  395. }
  396. .stats-row {
  397. display: flex;
  398. margin-bottom: 15px;
  399. }
  400. .stat-item {
  401. flex: 1;
  402. padding: 0 10px;
  403. }
  404. .stat-label {
  405. display: block;
  406. font-weight: bold;
  407. margin-bottom: 5px;
  408. color: #555;
  409. }
  410. .stat-value {
  411. font-size: 16px;
  412. color: #333;
  413. }
  414. </style>
  415. <?php
  416. }