statistics_customers.php 46 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315
  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. <style>
  275. .pie-charts-container {
  276. display: flex;
  277. flex-direction: row;
  278. justify-content: space-between;
  279. margin-bottom: 20px;
  280. }
  281. .pie-chart-wrapper {
  282. flex: 0 0 48%;
  283. max-width: 48%;
  284. }
  285. .customer-stats-summary {
  286. margin-top: 20px;
  287. padding: 15px;
  288. background-color: #f9f9f9;
  289. border-radius: 5px;
  290. }
  291. .stats-row {
  292. display: flex;
  293. margin-bottom: 15px;
  294. }
  295. .stat-item {
  296. flex: 1;
  297. padding: 0 10px;
  298. }
  299. .stat-label {
  300. display: block;
  301. font-weight: bold;
  302. margin-bottom: 5px;
  303. color: #555;
  304. }
  305. .stat-value {
  306. font-size: 16px;
  307. color: #333;
  308. }
  309. </style>
  310. <div class="pie-charts-container">
  311. <div class="pie-chart-wrapper">
  312. <h3 style="text-align: center; margin-bottom: 15px;">客户数量分布</h3>
  313. <canvas id="newVsReturningCustomersChart"></canvas>
  314. </div>
  315. <div class="pie-chart-wrapper">
  316. <h3 style="text-align: center; margin-bottom: 15px;">销售额分布</h3>
  317. <canvas id="newVsReturningAmountChart"></canvas>
  318. </div>
  319. </div>
  320. <div class="customer-stats-summary">
  321. <div class="stats-row">
  322. <div class="stat-item">
  323. <span class="stat-label">总客户数:</span>
  324. <span class="stat-value"><?php echo number_format($new_vs_returning['total_customers']); ?></span>
  325. </div>
  326. <div class="stat-item">
  327. <span class="stat-label">新客户:</span>
  328. <span class="stat-value"><?php echo number_format($new_vs_returning['new_customers']); ?>
  329. (<?php echo ($new_vs_returning['total_customers'] > 0) ? number_format(($new_vs_returning['new_customers'] / $new_vs_returning['total_customers']) * 100, 1) : '0'; ?>%)</span>
  330. </div>
  331. <div class="stat-item">
  332. <span class="stat-label">老客户:</span>
  333. <span class="stat-value"><?php echo number_format($new_vs_returning['returning_customers']); ?>
  334. (<?php echo ($new_vs_returning['total_customers'] > 0) ? number_format(($new_vs_returning['returning_customers'] / $new_vs_returning['total_customers']) * 100, 1) : '0'; ?>%)</span>
  335. </div>
  336. </div>
  337. <div class="stats-row">
  338. <div class="stat-item">
  339. <span class="stat-label">总销售额:</span>
  340. <span class="stat-value">¥<?php echo number_format($new_vs_returning['total_amount'], 2); ?></span>
  341. </div>
  342. <div class="stat-item">
  343. <span class="stat-label">新客户销售额:</span>
  344. <span class="stat-value">¥<?php echo number_format($new_vs_returning['new_customer_amount'], 2); ?>
  345. (<?php echo ($new_vs_returning['total_amount'] > 0) ? number_format(($new_vs_returning['new_customer_amount'] / $new_vs_returning['total_amount']) * 100, 1) : '0'; ?>%)</span>
  346. </div>
  347. <div class="stat-item">
  348. <span class="stat-label">老客户销售额:</span>
  349. <span class="stat-value">¥<?php echo number_format($new_vs_returning['returning_customer_amount'], 2); ?>
  350. (<?php echo ($new_vs_returning['total_amount'] > 0) ? number_format(($new_vs_returning['returning_customer_amount'] / $new_vs_returning['total_amount']) * 100, 1) : '0'; ?>%)</span>
  351. </div>
  352. </div>
  353. </div>
  354. </div>
  355. <script>
  356. // 新老客户数量图
  357. var newVsReturningCtx = document.getElementById('newVsReturningCustomersChart').getContext('2d');
  358. var newVsReturningChart = new Chart(newVsReturningCtx, {
  359. type: 'pie',
  360. data: {
  361. labels: ['新客户', '老客户'],
  362. datasets: [{
  363. data: [
  364. <?php echo $new_vs_returning['new_customers']; ?>,
  365. <?php echo $new_vs_returning['returning_customers']; ?>
  366. ],
  367. backgroundColor: [
  368. 'rgba(54, 162, 235, 0.7)',
  369. 'rgba(255, 99, 132, 0.7)'
  370. ],
  371. borderWidth: 1
  372. }]
  373. },
  374. options: {
  375. responsive: true,
  376. maintainAspectRatio: true,
  377. plugins: {
  378. legend: {
  379. position: 'bottom',
  380. }
  381. }
  382. }
  383. });
  384. // 新老客户销售额图
  385. var amountCtx = document.getElementById('newVsReturningAmountChart').getContext('2d');
  386. var amountChart = new Chart(amountCtx, {
  387. type: 'pie',
  388. data: {
  389. labels: ['新客户销售额', '老客户销售额'],
  390. datasets: [{
  391. data: [
  392. <?php echo $new_vs_returning['new_customer_amount']; ?>,
  393. <?php echo $new_vs_returning['returning_customer_amount']; ?>
  394. ],
  395. backgroundColor: [
  396. 'rgba(54, 162, 235, 0.7)',
  397. 'rgba(255, 99, 132, 0.7)'
  398. ],
  399. borderWidth: 1
  400. }]
  401. },
  402. options: {
  403. responsive: true,
  404. maintainAspectRatio: true,
  405. plugins: {
  406. legend: {
  407. position: 'bottom',
  408. }
  409. }
  410. }
  411. });
  412. </script>
  413. <?php
  414. }
  415. /**
  416. * 获取客户总数
  417. *
  418. * @param mysqli $conn 数据库连接
  419. * @return int 客户总数
  420. */
  421. function getTotalCustomers($conn) {
  422. $sql = "SELECT COUNT(id) as total FROM customer";
  423. $result = $conn->query($sql);
  424. $row = $result->fetch_assoc();
  425. return $row['total'];
  426. }
  427. /**
  428. * 获取指定时间段内新增客户数
  429. *
  430. * @param mysqli $conn 数据库连接
  431. * @param string $start_date 开始日期
  432. * @param string $end_date 结束日期
  433. * @return int 新增客户数
  434. */
  435. function getNewCustomers($conn, $start_date, $end_date) {
  436. $sql = "SELECT COUNT(id) as new_count
  437. FROM customer
  438. WHERE cs_addtime BETWEEN ? AND ?";
  439. $stmt = $conn->prepare($sql);
  440. $stmt->bind_param("ss", $start_date, $end_date);
  441. $stmt->execute();
  442. $result = $stmt->get_result();
  443. $row = $result->fetch_assoc();
  444. return $row['new_count'];
  445. }
  446. /**
  447. * 计算平均客户价值(客户平均订单金额)
  448. *
  449. * @param mysqli $conn 数据库连接
  450. * @param string $start_date 开始日期
  451. * @param string $end_date 结束日期
  452. * @return float 平均客户价值
  453. */
  454. function getAverageCustomerValue($conn, $start_date, $end_date) {
  455. $sql = "SELECT AVG(customer_value) as avg_value FROM (
  456. SELECT
  457. o.customer_id,
  458. SUM(o.total_amount) as customer_value
  459. FROM orders o
  460. WHERE o.order_date BETWEEN ? AND ?
  461. GROUP BY o.customer_id
  462. ) as customer_values";
  463. $stmt = $conn->prepare($sql);
  464. $stmt->bind_param("ss", $start_date, $end_date);
  465. $stmt->execute();
  466. $result = $stmt->get_result();
  467. $row = $result->fetch_assoc();
  468. return $row['avg_value'] ? $row['avg_value'] : 0;
  469. }
  470. /**
  471. * 计算客户留存率
  472. *
  473. * @param mysqli $conn 数据库连接
  474. * @param string $start_date 开始日期
  475. * @param string $end_date 结束日期
  476. * @return array 客户留存率数据
  477. */
  478. function getCustomerRetentionRate($conn, $start_date, $end_date) {
  479. // 获取之前时间段的客户
  480. $previous_start = date('Y-m-d', strtotime('-1 year', strtotime($start_date)));
  481. $previous_end = date('Y-m-d', strtotime('-1 day', strtotime($start_date)));
  482. // 之前时间段的客户ID
  483. $prev_sql = "SELECT DISTINCT customer_id
  484. FROM orders
  485. WHERE order_date BETWEEN ? AND ?";
  486. $prev_stmt = $conn->prepare($prev_sql);
  487. $prev_stmt->bind_param("ss", $previous_start, $previous_end);
  488. $prev_stmt->execute();
  489. $prev_result = $prev_stmt->get_result();
  490. $previous_customers = [];
  491. while ($row = $prev_result->fetch_assoc()) {
  492. $previous_customers[] = $row['customer_id'];
  493. }
  494. $previous_count = count($previous_customers);
  495. // 如果没有之前的客户,返回0
  496. if ($previous_count == 0) {
  497. return [
  498. 'retained_count' => 0,
  499. 'total_previous' => 0,
  500. 'retention_rate' => 0
  501. ];
  502. }
  503. // 查询当前时间段内,之前客户中再次购买的客户数
  504. $current_sql = "SELECT COUNT(DISTINCT customer_id) as retained_count
  505. FROM orders
  506. WHERE order_date BETWEEN ? AND ?
  507. AND customer_id IN (" . implode(',', $previous_customers) . ")";
  508. $current_stmt = $conn->prepare($current_sql);
  509. $current_stmt->bind_param("ss", $start_date, $end_date);
  510. $current_stmt->execute();
  511. $current_result = $current_stmt->get_result();
  512. $row = $current_result->fetch_assoc();
  513. $retained_count = $row['retained_count'];
  514. $retention_rate = ($retained_count / $previous_count) * 100;
  515. return [
  516. 'retained_count' => $retained_count,
  517. 'total_previous' => $previous_count,
  518. 'retention_rate' => $retention_rate
  519. ];
  520. }
  521. /**
  522. * 计算下单转换率
  523. *
  524. * @param mysqli $conn 数据库连接
  525. * @param string $start_date 开始日期
  526. * @param string $end_date 结束日期
  527. * @return array 下单转换率数据
  528. */
  529. function getOrderConversionRate($conn, $start_date, $end_date) {
  530. // 获取指定时间段内总客户数
  531. $total_sql = "SELECT COUNT(DISTINCT id) as total_count FROM customer WHERE cs_addtime <= ?";
  532. $total_stmt = $conn->prepare($total_sql);
  533. $total_stmt->bind_param("s", $end_date);
  534. $total_stmt->execute();
  535. $total_result = $total_stmt->get_result();
  536. $total_row = $total_result->fetch_assoc();
  537. $total_customers = $total_row['total_count'];
  538. // 获取有订单的客户数
  539. $order_sql = "SELECT COUNT(DISTINCT customer_id) as order_count
  540. FROM orders
  541. WHERE order_date BETWEEN ? AND ?";
  542. $order_stmt = $conn->prepare($order_sql);
  543. $order_stmt->bind_param("ss", $start_date, $end_date);
  544. $order_stmt->execute();
  545. $order_result = $order_stmt->get_result();
  546. $order_row = $order_result->fetch_assoc();
  547. $customers_with_orders = $order_row['order_count'];
  548. // 计算转换率
  549. $conversion_rate = ($total_customers > 0) ? ($customers_with_orders / $total_customers) * 100 : 0;
  550. return [
  551. 'total_customers' => $total_customers,
  552. 'customers_with_orders' => $customers_with_orders,
  553. 'conversion_rate' => $conversion_rate
  554. ];
  555. }
  556. /**
  557. * 渲染关键指标仪表板
  558. *
  559. * @param array $kpi_data 关键指标数据
  560. * @return void
  561. */
  562. function renderKeyMetricsCard($kpi_data) {
  563. ?>
  564. <div class="stats-card-container">
  565. <div class="stats-card">
  566. <div class="stats-card-header">
  567. <h3>客户总数</h3>
  568. </div>
  569. <div class="stats-card-body">
  570. <div class="stats-card-value"><?php echo number_format($kpi_data['total_customers']); ?></div>
  571. </div>
  572. </div>
  573. <div class="stats-card">
  574. <div class="stats-card-header">
  575. <h3>新增客户</h3>
  576. </div>
  577. <div class="stats-card-body">
  578. <div class="stats-card-value"><?php echo number_format($kpi_data['new_customers']); ?></div>
  579. </div>
  580. </div>
  581. <div class="stats-card">
  582. <div class="stats-card-header">
  583. <h3>平均客户价值</h3>
  584. </div>
  585. <div class="stats-card-body">
  586. <div class="stats-card-value">¥<?php echo number_format($kpi_data['avg_customer_value'], 2); ?></div>
  587. </div>
  588. </div>
  589. <div class="stats-card">
  590. <div class="stats-card-header">
  591. <h3>客户留存率</h3>
  592. </div>
  593. <div class="stats-card-body">
  594. <div class="stats-card-value"><?php echo number_format($kpi_data['retention_rate'], 1); ?>%</div>
  595. <div class="stats-card-subtitle"><?php echo number_format($kpi_data['retained_count']); ?> / <?php echo number_format($kpi_data['total_previous']); ?></div>
  596. </div>
  597. </div>
  598. <div class="stats-card">
  599. <div class="stats-card-header">
  600. <h3>下单转换率</h3>
  601. </div>
  602. <div class="stats-card-body">
  603. <div class="stats-card-value"><?php echo number_format($kpi_data['conversion_rate'], 1); ?>%</div>
  604. <div class="stats-card-subtitle"><?php echo number_format($kpi_data['customers_with_orders']); ?> / <?php echo number_format($kpi_data['total_customers']); ?></div>
  605. </div>
  606. </div>
  607. </div>
  608. <?php
  609. }
  610. /**
  611. * 获取客户价值分布数据
  612. *
  613. * @param mysqli $conn 数据库连接
  614. * @param string $start_date 开始日期
  615. * @param string $end_date 结束日期
  616. * @return array 客户价值分布数据
  617. */
  618. function getCustomerValueDistribution($conn, $start_date, $end_date) {
  619. $sql = "SELECT
  620. value_segment,
  621. COUNT(customer_id) as customer_count,
  622. SUM(total_amount) as total_amount
  623. FROM (
  624. SELECT
  625. o.customer_id,
  626. SUM(o.total_amount) as total_amount,
  627. CASE
  628. WHEN SUM(o.total_amount) > 100000 THEN '高价值客户(>10万)'
  629. WHEN SUM(o.total_amount) > 50000 THEN '中高价值客户(5-10万)'
  630. WHEN SUM(o.total_amount) > 10000 THEN '中价值客户(1-5万)'
  631. WHEN SUM(o.total_amount) > 5000 THEN '低价值客户(5千-1万)'
  632. ELSE '微价值客户(<5千)'
  633. END as value_segment
  634. FROM orders o
  635. WHERE o.order_date BETWEEN ? AND ?
  636. GROUP BY o.customer_id
  637. ) as customer_value
  638. GROUP BY value_segment
  639. ORDER BY
  640. CASE value_segment
  641. WHEN '高价值客户(>10万)' THEN 1
  642. WHEN '中高价值客户(5-10万)' THEN 2
  643. WHEN '中价值客户(1-5万)' THEN 3
  644. WHEN '低价值客户(5千-1万)' THEN 4
  645. ELSE 5
  646. END";
  647. $stmt = $conn->prepare($sql);
  648. $stmt->bind_param("ss", $start_date, $end_date);
  649. $stmt->execute();
  650. $result = $stmt->get_result();
  651. $value_segments = [];
  652. $customer_counts = [];
  653. $total_amounts = [];
  654. $total_customers = 0;
  655. while ($row = $result->fetch_assoc()) {
  656. $value_segments[] = $row['value_segment'];
  657. $customer_counts[] = $row['customer_count'];
  658. $total_amounts[] = $row['total_amount'];
  659. $total_customers += $row['customer_count'];
  660. }
  661. return [
  662. 'segments' => $value_segments,
  663. 'counts' => $customer_counts,
  664. 'amounts' => $total_amounts,
  665. 'total_customers' => $total_customers
  666. ];
  667. }
  668. /**
  669. * 获取客户活跃度分析数据
  670. *
  671. * @param mysqli $conn 数据库连接
  672. * @param string $end_date 截止日期
  673. * @return array 客户活跃度分析数据
  674. */
  675. function getCustomerActivityAnalysis($conn, $end_date) {
  676. $sql = "SELECT
  677. activity_level,
  678. COUNT(*) as customer_count
  679. FROM (
  680. SELECT
  681. o.customer_id,
  682. CASE
  683. WHEN DATEDIFF(?, MAX(o.order_date)) <= 30 THEN '活跃客户(30天内)'
  684. WHEN DATEDIFF(?, MAX(o.order_date)) <= 90 THEN '一般活跃(90天内)'
  685. WHEN DATEDIFF(?, MAX(o.order_date)) <= 180 THEN '低活跃(180天内)'
  686. WHEN DATEDIFF(?, MAX(o.order_date)) <= 365 THEN '沉睡客户(1年内)'
  687. ELSE '流失客户(超过1年)'
  688. END as activity_level
  689. FROM orders o
  690. GROUP BY o.customer_id
  691. ) as customer_activity
  692. GROUP BY activity_level
  693. ORDER BY
  694. CASE activity_level
  695. WHEN '活跃客户(30天内)' THEN 1
  696. WHEN '一般活跃(90天内)' THEN 2
  697. WHEN '低活跃(180天内)' THEN 3
  698. WHEN '沉睡客户(1年内)' THEN 4
  699. ELSE 5
  700. END";
  701. $stmt = $conn->prepare($sql);
  702. $end_date_formatted = date('Y-m-d', strtotime($end_date));
  703. $stmt->bind_param("ssss", $end_date_formatted, $end_date_formatted, $end_date_formatted, $end_date_formatted);
  704. $stmt->execute();
  705. $result = $stmt->get_result();
  706. $activity_levels = [];
  707. $customer_counts = [];
  708. while ($row = $result->fetch_assoc()) {
  709. $activity_levels[] = $row['activity_level'];
  710. $customer_counts[] = $row['customer_count'];
  711. }
  712. return [
  713. 'levels' => $activity_levels,
  714. 'counts' => $customer_counts
  715. ];
  716. }
  717. /**
  718. * 获取客户流失风险分析数据
  719. *
  720. * @param mysqli $conn 数据库连接
  721. * @param string $end_date 截止日期
  722. * @return array 客户流失风险分析数据
  723. */
  724. function getCustomerChurnRiskAnalysis($conn, $end_date) {
  725. $sql = "SELECT
  726. risk_level,
  727. COUNT(*) as customer_count
  728. FROM (
  729. SELECT
  730. c.id,
  731. CASE
  732. WHEN last_order_date IS NULL THEN '从未购买'
  733. WHEN DATEDIFF(?, last_order_date) <= 90 THEN '低风险(90天内)'
  734. WHEN DATEDIFF(?, last_order_date) <= 180 THEN '中风险(90-180天)'
  735. WHEN DATEDIFF(?, last_order_date) <= 365 THEN '高风险(180-365天)'
  736. ELSE '极高风险(超过1年)'
  737. END as risk_level
  738. FROM customer c
  739. LEFT JOIN (
  740. SELECT customer_id, MAX(order_date) as last_order_date
  741. FROM orders
  742. GROUP BY customer_id
  743. ) o ON c.id = o.customer_id
  744. ) as customer_risk
  745. GROUP BY risk_level
  746. ORDER BY
  747. CASE risk_level
  748. WHEN '低风险(90天内)' THEN 1
  749. WHEN '中风险(90-180天)' THEN 2
  750. WHEN '高风险(180-365天)' THEN 3
  751. WHEN '极高风险(超过1年)' THEN 4
  752. WHEN '从未购买' THEN 5
  753. END";
  754. $stmt = $conn->prepare($sql);
  755. $end_date_formatted = date('Y-m-d', strtotime($end_date));
  756. $stmt->bind_param("sss", $end_date_formatted, $end_date_formatted, $end_date_formatted);
  757. $stmt->execute();
  758. $result = $stmt->get_result();
  759. $risk_levels = [];
  760. $customer_counts = [];
  761. while ($row = $result->fetch_assoc()) {
  762. $risk_levels[] = $row['risk_level'];
  763. $customer_counts[] = $row['customer_count'];
  764. }
  765. return [
  766. 'levels' => $risk_levels,
  767. 'counts' => $customer_counts
  768. ];
  769. }
  770. /**
  771. * 获取客户来源分析数据
  772. *
  773. * @param mysqli $conn 数据库连接
  774. * @return array 客户来源分析数据
  775. */
  776. function getCustomerSourceAnalysis($conn) {
  777. // 假设cs_from字段代表客户来源,需要根据实际情况调整SQL
  778. $sql = "SELECT
  779. source,
  780. COUNT(*) as customer_count
  781. FROM (
  782. SELECT
  783. id,
  784. CASE
  785. WHEN cs_from = 1 THEN '网站注册'
  786. WHEN cs_from = 2 THEN '销售开发'
  787. WHEN cs_from = 3 THEN '广告引流'
  788. WHEN cs_from = 4 THEN '展会获取'
  789. WHEN cs_from = 5 THEN '客户推荐'
  790. ELSE '其他来源'
  791. END as source
  792. FROM customer
  793. ) as customer_source
  794. GROUP BY source
  795. ORDER BY customer_count DESC";
  796. $result = $conn->query($sql);
  797. $sources = [];
  798. $counts = [];
  799. while ($row = $result->fetch_assoc()) {
  800. $sources[] = $row['source'];
  801. $counts[] = $row['customer_count'];
  802. }
  803. return [
  804. 'sources' => $sources,
  805. 'counts' => $counts
  806. ];
  807. }
  808. /**
  809. * 渲染客户价值分布图表
  810. *
  811. * @param array $value_data 客户价值分布数据
  812. * @return void
  813. */
  814. function renderCustomerValueCharts($value_data) {
  815. ?>
  816. <div class="chart-container">
  817. <div class="chart-header">
  818. <h2 class="chart-title">客户价值分布</h2>
  819. </div>
  820. <div class="chart-row" style="display: flex; width: 100%; margin: 0 -10px;">
  821. <div class="chart-column" style="flex: 0 0 50%; max-width: 50%; padding: 0 10px; box-sizing: border-box;">
  822. <h3 style="text-align: center; margin-bottom: 15px;">客户价值分布(柱状图)</h3>
  823. <canvas id="customerValueBarChart"></canvas>
  824. </div>
  825. <div class="chart-column" style="flex: 0 0 50%; max-width: 50%; padding: 0 10px; box-sizing: border-box;">
  826. <h3 style="text-align: center; margin-bottom: 15px;">客户价值分布(饼图)</h3>
  827. <canvas id="customerValuePieChart"></canvas>
  828. </div>
  829. </div>
  830. <div class="customer-stats-summary">
  831. <div class="stats-row">
  832. <?php foreach ($value_data['segments'] as $index => $segment): ?>
  833. <div class="stat-item">
  834. <span class="stat-label"><?php echo $segment; ?>:</span>
  835. <span class="stat-value"><?php echo number_format($value_data['counts'][$index]); ?>
  836. (<?php echo ($value_data['total_customers'] > 0) ?
  837. number_format(($value_data['counts'][$index] / $value_data['total_customers']) * 100, 1) : '0'; ?>%)</span>
  838. <span class="stat-sub-value">¥<?php echo number_format($value_data['amounts'][$index], 2); ?></span>
  839. </div>
  840. <?php endforeach; ?>
  841. </div>
  842. </div>
  843. </div>
  844. <script>
  845. // 客户价值分布柱状图
  846. var valueBarCtx = document.getElementById('customerValueBarChart').getContext('2d');
  847. var valueBarChart = new Chart(valueBarCtx, {
  848. type: 'bar',
  849. data: {
  850. labels: <?php echo json_encode($value_data['segments']); ?>,
  851. datasets: [{
  852. label: '客户数量',
  853. data: <?php echo json_encode($value_data['counts']); ?>,
  854. backgroundColor: [
  855. 'rgba(54, 162, 235, 0.7)',
  856. 'rgba(75, 192, 192, 0.7)',
  857. 'rgba(255, 206, 86, 0.7)',
  858. 'rgba(255, 99, 132, 0.7)',
  859. 'rgba(153, 102, 255, 0.7)'
  860. ],
  861. borderWidth: 1
  862. }]
  863. },
  864. options: {
  865. responsive: true,
  866. scales: {
  867. y: {
  868. beginAtZero: true,
  869. title: {
  870. display: true,
  871. text: '客户数量'
  872. }
  873. }
  874. }
  875. }
  876. });
  877. // 客户价值分布饼图
  878. var valuePieCtx = document.getElementById('customerValuePieChart').getContext('2d');
  879. var valuePieChart = new Chart(valuePieCtx, {
  880. type: 'pie',
  881. data: {
  882. labels: <?php echo json_encode($value_data['segments']); ?>,
  883. datasets: [{
  884. data: <?php echo json_encode($value_data['counts']); ?>,
  885. backgroundColor: [
  886. 'rgba(54, 162, 235, 0.7)',
  887. 'rgba(75, 192, 192, 0.7)',
  888. 'rgba(255, 206, 86, 0.7)',
  889. 'rgba(255, 99, 132, 0.7)',
  890. 'rgba(153, 102, 255, 0.7)'
  891. ],
  892. borderWidth: 1
  893. }]
  894. },
  895. options: {
  896. responsive: true
  897. }
  898. });
  899. </script>
  900. <?php
  901. }
  902. /**
  903. * 渲染客户活跃度分析图表
  904. *
  905. * @param array $activity_data 客户活跃度数据
  906. * @return void
  907. */
  908. function renderCustomerActivityChart($activity_data) {
  909. ?>
  910. <div class="chart-container">
  911. <div class="chart-header">
  912. <h2 class="chart-title">客户活跃度分析</h2>
  913. </div>
  914. <canvas id="customerActivityChart"></canvas>
  915. <div class="customer-stats-summary">
  916. <div class="stats-row">
  917. <?php
  918. $total_customers = array_sum($activity_data['counts']);
  919. foreach ($activity_data['levels'] as $index => $level):
  920. ?>
  921. <div class="stat-item">
  922. <span class="stat-label"><?php echo $level; ?>:</span>
  923. <span class="stat-value"><?php echo number_format($activity_data['counts'][$index]); ?>
  924. (<?php echo ($total_customers > 0) ?
  925. number_format(($activity_data['counts'][$index] / $total_customers) * 100, 1) : '0'; ?>%)</span>
  926. </div>
  927. <?php endforeach; ?>
  928. </div>
  929. </div>
  930. </div>
  931. <script>
  932. // 客户活跃度分析图
  933. var activityCtx = document.getElementById('customerActivityChart').getContext('2d');
  934. var activityChart = new Chart(activityCtx, {
  935. type: 'bar',
  936. data: {
  937. labels: <?php echo json_encode($activity_data['levels']); ?>,
  938. datasets: [{
  939. label: '客户数量',
  940. data: <?php echo json_encode($activity_data['counts']); ?>,
  941. backgroundColor: [
  942. 'rgba(54, 162, 235, 0.7)',
  943. 'rgba(75, 192, 192, 0.7)',
  944. 'rgba(255, 206, 86, 0.7)',
  945. 'rgba(255, 99, 132, 0.7)',
  946. 'rgba(153, 102, 255, 0.7)'
  947. ],
  948. borderWidth: 1
  949. }]
  950. },
  951. options: {
  952. responsive: true,
  953. scales: {
  954. y: {
  955. beginAtZero: true,
  956. title: {
  957. display: true,
  958. text: '客户数量'
  959. }
  960. }
  961. }
  962. }
  963. });
  964. </script>
  965. <?php
  966. }
  967. /**
  968. * 渲染客户流失风险分析图表
  969. *
  970. * @param array $risk_data 客户流失风险数据
  971. * @return void
  972. */
  973. function renderCustomerChurnRiskChart($risk_data) {
  974. ?>
  975. <div class="chart-container">
  976. <div class="chart-header">
  977. <h2 class="chart-title">客户流失风险分析</h2>
  978. </div>
  979. <canvas id="customerRiskChart"></canvas>
  980. <div class="customer-stats-summary">
  981. <div class="stats-row">
  982. <?php
  983. $total_customers = array_sum($risk_data['counts']);
  984. foreach ($risk_data['levels'] as $index => $level):
  985. ?>
  986. <div class="stat-item">
  987. <span class="stat-label"><?php echo $level; ?>:</span>
  988. <span class="stat-value"><?php echo number_format($risk_data['counts'][$index]); ?>
  989. (<?php echo ($total_customers > 0) ?
  990. number_format(($risk_data['counts'][$index] / $total_customers) * 100, 1) : '0'; ?>%)</span>
  991. </div>
  992. <?php endforeach; ?>
  993. </div>
  994. </div>
  995. </div>
  996. <script>
  997. // 客户流失风险分析图
  998. var riskCtx = document.getElementById('customerRiskChart').getContext('2d');
  999. var riskChart = new Chart(riskCtx, {
  1000. type: 'doughnut',
  1001. data: {
  1002. labels: <?php echo json_encode($risk_data['levels']); ?>,
  1003. datasets: [{
  1004. data: <?php echo json_encode($risk_data['counts']); ?>,
  1005. backgroundColor: [
  1006. 'rgba(54, 162, 235, 0.7)',
  1007. 'rgba(75, 192, 192, 0.7)',
  1008. 'rgba(255, 206, 86, 0.7)',
  1009. 'rgba(255, 99, 132, 0.7)',
  1010. 'rgba(153, 102, 255, 0.7)'
  1011. ],
  1012. borderWidth: 1
  1013. }]
  1014. },
  1015. options: {
  1016. responsive: true,
  1017. plugins: {
  1018. legend: {
  1019. position: 'right',
  1020. }
  1021. }
  1022. }
  1023. });
  1024. </script>
  1025. <?php
  1026. }
  1027. /**
  1028. * 渲染客户来源分析图表
  1029. *
  1030. * @param array $source_data 客户来源数据
  1031. * @return void
  1032. */
  1033. function renderCustomerSourceChart($source_data) {
  1034. ?>
  1035. <div class="chart-container">
  1036. <div class="chart-header">
  1037. <h2 class="chart-title">客户来源分析</h2>
  1038. </div>
  1039. <canvas id="customerSourceChart"></canvas>
  1040. <div class="customer-stats-summary">
  1041. <div class="stats-row">
  1042. <?php
  1043. $total_customers = array_sum($source_data['counts']);
  1044. foreach ($source_data['sources'] as $index => $source):
  1045. ?>
  1046. <div class="stat-item">
  1047. <span class="stat-label"><?php echo $source; ?>:</span>
  1048. <span class="stat-value"><?php echo number_format($source_data['counts'][$index]); ?>
  1049. (<?php echo ($total_customers > 0) ?
  1050. number_format(($source_data['counts'][$index] / $total_customers) * 100, 1) : '0'; ?>%)</span>
  1051. </div>
  1052. <?php endforeach; ?>
  1053. </div>
  1054. </div>
  1055. </div>
  1056. <script>
  1057. // 客户来源分析图
  1058. var sourceCtx = document.getElementById('customerSourceChart').getContext('2d');
  1059. var sourceChart = new Chart(sourceCtx, {
  1060. type: 'pie',
  1061. data: {
  1062. labels: <?php echo json_encode($source_data['sources']); ?>,
  1063. datasets: [{
  1064. data: <?php echo json_encode($source_data['counts']); ?>,
  1065. backgroundColor: [
  1066. 'rgba(54, 162, 235, 0.7)',
  1067. 'rgba(75, 192, 192, 0.7)',
  1068. 'rgba(255, 206, 86, 0.7)',
  1069. 'rgba(255, 99, 132, 0.7)',
  1070. 'rgba(153, 102, 255, 0.7)',
  1071. 'rgba(255, 159, 64, 0.7)'
  1072. ],
  1073. borderWidth: 1
  1074. }]
  1075. },
  1076. options: {
  1077. responsive: true,
  1078. plugins: {
  1079. legend: {
  1080. position: 'right',
  1081. }
  1082. }
  1083. }
  1084. });
  1085. </script>
  1086. <?php
  1087. }
  1088. /**
  1089. * 获取客户转化漏斗数据
  1090. *
  1091. * @param mysqli $conn 数据库连接
  1092. * @param string $start_date 开始日期
  1093. * @param string $end_date 结束日期
  1094. * @return array 客户转化漏斗数据
  1095. */
  1096. function getCustomerConversionFunnel($conn, $start_date, $end_date) {
  1097. // 获取总客户数(潜在客户)
  1098. $total_sql = "SELECT COUNT(id) as total FROM customer";
  1099. $total_result = $conn->query($total_sql);
  1100. $total_row = $total_result->fetch_assoc();
  1101. $total_customers = $total_row['total'];
  1102. // 获取明确需求的客户数
  1103. $needs_sql = "SELECT COUNT(id) as needs_count FROM customer WHERE cs_deal = 2";
  1104. $needs_result = $conn->query($needs_sql);
  1105. $needs_row = $needs_result->fetch_assoc();
  1106. $needs_customers = $needs_row['needs_count'];
  1107. // 获取已成交客户数
  1108. $deal_sql = "SELECT COUNT(id) as deal_count FROM customer WHERE cs_deal = 3";
  1109. $deal_result = $conn->query($deal_sql);
  1110. $deal_row = $deal_result->fetch_assoc();
  1111. $deal_customers = $deal_row['deal_count'];
  1112. // 获取有订单的客户数
  1113. $order_sql = "SELECT COUNT(DISTINCT customer_id) as order_count FROM orders WHERE order_date BETWEEN ? AND ?";
  1114. $order_stmt = $conn->prepare($order_sql);
  1115. $order_stmt->bind_param("ss", $start_date, $end_date);
  1116. $order_stmt->execute();
  1117. $order_result = $order_stmt->get_result();
  1118. $order_row = $order_result->fetch_assoc();
  1119. $order_customers = $order_row['order_count'];
  1120. // 获取复购客户数(多次下单)
  1121. $repeat_sql = "SELECT COUNT(customer_id) as repeat_count FROM (
  1122. SELECT customer_id, COUNT(id) as order_count
  1123. FROM orders
  1124. WHERE order_date BETWEEN ? AND ?
  1125. GROUP BY customer_id
  1126. HAVING order_count > 1
  1127. ) as repeat_customers";
  1128. $repeat_stmt = $conn->prepare($repeat_sql);
  1129. $repeat_stmt->bind_param("ss", $start_date, $end_date);
  1130. $repeat_stmt->execute();
  1131. $repeat_result = $repeat_stmt->get_result();
  1132. $repeat_row = $repeat_result->fetch_assoc();
  1133. $repeat_customers = $repeat_row['repeat_count'];
  1134. return [
  1135. 'stages' => ['潜在客户', '明确需求', '已成交', '有效订单', '复购客户'],
  1136. 'counts' => [$total_customers, $needs_customers, $deal_customers, $order_customers, $repeat_customers]
  1137. ];
  1138. }
  1139. /**
  1140. * 渲染客户转化漏斗图表
  1141. *
  1142. * @param array $funnel_data 客户转化漏斗数据
  1143. * @return void
  1144. */
  1145. function renderCustomerFunnelChart($funnel_data) {
  1146. ?>
  1147. <div class="chart-container">
  1148. <div class="chart-header">
  1149. <h2 class="chart-title">客户转化漏斗</h2>
  1150. </div>
  1151. <canvas id="customerFunnelChart" style="max-height: 400px;"></canvas>
  1152. <div class="customer-stats-summary">
  1153. <div class="stats-row">
  1154. <?php
  1155. foreach ($funnel_data['stages'] as $index => $stage):
  1156. $current_count = $funnel_data['counts'][$index];
  1157. $prev_count = $index > 0 ? $funnel_data['counts'][$index-1] : $current_count;
  1158. $conversion_rate = $prev_count > 0 ? ($current_count / $prev_count) * 100 : 0;
  1159. ?>
  1160. <div class="stat-item">
  1161. <span class="stat-label"><?php echo $stage; ?>:</span>
  1162. <span class="stat-value"><?php echo number_format($current_count); ?></span>
  1163. <?php if ($index > 0): ?>
  1164. <span class="stat-conversion">
  1165. 转化率: <?php echo number_format($conversion_rate, 1); ?>%
  1166. </span>
  1167. <?php endif; ?>
  1168. </div>
  1169. <?php endforeach; ?>
  1170. </div>
  1171. </div>
  1172. </div>
  1173. <script>
  1174. // 客户转化漏斗图
  1175. var funnelCtx = document.getElementById('customerFunnelChart').getContext('2d');
  1176. var funnelChart = new Chart(funnelCtx, {
  1177. type: 'bar',
  1178. data: {
  1179. labels: <?php echo json_encode($funnel_data['stages']); ?>,
  1180. datasets: [{
  1181. label: '客户数量',
  1182. data: <?php echo json_encode($funnel_data['counts']); ?>,
  1183. backgroundColor: [
  1184. 'rgba(54, 162, 235, 0.7)',
  1185. 'rgba(75, 192, 192, 0.7)',
  1186. 'rgba(255, 206, 86, 0.7)',
  1187. 'rgba(255, 99, 132, 0.7)',
  1188. 'rgba(153, 102, 255, 0.7)'
  1189. ],
  1190. borderWidth: 1
  1191. }]
  1192. },
  1193. options: {
  1194. indexAxis: 'y',
  1195. responsive: true,
  1196. scales: {
  1197. x: {
  1198. beginAtZero: true,
  1199. title: {
  1200. display: true,
  1201. text: '客户数量'
  1202. }
  1203. }
  1204. },
  1205. plugins: {
  1206. tooltip: {
  1207. callbacks: {
  1208. afterLabel: function(context) {
  1209. var index = context.dataIndex;
  1210. if (index > 0) {
  1211. var currentValue = context.parsed.x;
  1212. var previousValue = context.dataset.data[index-1];
  1213. var conversionRate = previousValue > 0 ? (currentValue / previousValue * 100).toFixed(1) : 0;
  1214. return '转化率: ' + conversionRate + '%';
  1215. }
  1216. return '';
  1217. }
  1218. }
  1219. }
  1220. }
  1221. }
  1222. });
  1223. </script>
  1224. <?php
  1225. }