customerAdd.php 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582
  1. <?php
  2. require_once 'conn.php';
  3. checkLogin();
  4. ?>
  5. <!DOCTYPE html>
  6. <html xmlns="http://www.w3.org/1999/xhtml">
  7. <head>
  8. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  9. <title>客户管理</title>
  10. <link rel="stylesheet" href="css/common.css" type="text/css" />
  11. <script src="js/jquery-1.7.2.min.js"></script>
  12. <script src="js/js.js"></script>
  13. <script src="js/xheditor-1.1.9/xheditor-1.1.9-zh-cn.min.js"></script>
  14. <script src="js/Hz2Py-szm-min.js"></script>
  15. <script src="js/ySearchSelect.js"></script>
  16. <script>
  17. $(document).ready(function(){
  18. $('.txt2').xheditor({
  19. tools:'full',
  20. hoverExecDelay:-1,
  21. urlBase:'system',
  22. upLinkUrl:"upload.php",
  23. upLinkExt:"zip,rar,txt,pdf",
  24. upImgUrl:"upload.php",
  25. upImgExt:"jpg,jpeg,gif,png",
  26. upFlashUrl:"upload.php",
  27. upFlashExt:"swf",
  28. upMediaUrl:"upload.php",
  29. upMediaExt:"wmv,avi,wma,mp3,mid"
  30. });
  31. // Add contact form
  32. $('.add-contact-btn').click(function() {
  33. var contactsContainer = $('#contacts-container');
  34. var contactIndex = contactsContainer.children('.contact-form').length;
  35. var contactForm = `
  36. <div class="contact-form" id="contact-form-${contactIndex}">
  37. <div class="contact-header">
  38. <button type="button" class="remove-contact-btn" data-index="${contactIndex}">删除</button>
  39. <h3>联系人 #${contactIndex + 1}</h3>
  40. </div>
  41. <input type="hidden" name="contact[${contactIndex}][id]" value="">
  42. <div class="contact-method-row">
  43. <span style="width:80px;display:inline-block;font-weight:bold;" class="method-select">联系人姓名</span>
  44. <input type="text" name="contact[${contactIndex}][contact_name]" class="txt1 method-input" style="width:60%;" placeholder="联系人姓名"/>
  45. </div>
  46. <div class="contact-methods-container" id="contact-methods-${contactIndex}">
  47. <!-- Contact methods will be added here -->
  48. </div>
  49. <button type="button" class="add-method-btn" data-contact-index="${contactIndex}">添加联系方式</button>
  50. </div>
  51. `;
  52. contactsContainer.append(contactForm);
  53. });
  54. // Add contact method
  55. $(document).on('click', '.add-method-btn', function() {
  56. var contactIndex = $(this).data('contact-index');
  57. var methodsContainer = $('#contact-methods-' + contactIndex);
  58. // Count existing methods by type
  59. var methodCounts = {};
  60. methodsContainer.find('select.method-select').each(function() {
  61. var type = $(this).val();
  62. if (type) {
  63. methodCounts[type] = (methodCounts[type] || 0) + 1;
  64. }
  65. });
  66. var methodRow = `
  67. <div class="contact-method-row">
  68. <select class="method-select" onchange="updateMethodSelectAndPlaceholder(this)">
  69. <option value="">请选择联系方式</option>
  70. <option value="tel" ${(methodCounts.tel || 0) >= 3 ? 'disabled' : ''}>电话</option>
  71. <option value="wechat" ${(methodCounts.wechat || 0) >= 3 ? 'disabled' : ''}>微信</option>
  72. <option value="whatsapp" ${(methodCounts.whatsapp || 0) >= 3 ? 'disabled' : ''}>WhatsApp</option>
  73. <option value="email" ${(methodCounts.email || 0) >= 3 ? 'disabled' : ''}>邮箱</option>
  74. <option value="linkedin" ${(methodCounts.linkedin || 0) >= 3 ? 'disabled' : ''}>领英</option>
  75. <option value="facebook" ${(methodCounts.facebook || 0) >= 3 ? 'disabled' : ''}>Facebook</option>
  76. <option value="alibaba" ${(methodCounts.alibaba || 0) >= 3 ? 'disabled' : ''}>阿里巴巴</option>
  77. </select>
  78. <input type="text" class="txt1 method-input" style="width:60%;" placeholder="请选择联系方式类型">
  79. <button type="button" class="remove-method-btn">删除</button>
  80. </div>
  81. `;
  82. methodsContainer.append(methodRow);
  83. updateMethodFields(methodsContainer.find('.contact-method-row:last-child'));
  84. });
  85. // Remove contact method
  86. $(document).on('click', '.remove-method-btn', function() {
  87. var methodRow = $(this).closest('.contact-method-row');
  88. var contactIndex = methodRow.closest('.contact-form').attr('id').split('-')[2];
  89. var type = methodRow.find('select.method-select').val();
  90. methodRow.remove();
  91. // Update available options in other selects
  92. updateAvailableMethodTypes(contactIndex);
  93. });
  94. // Remove contact
  95. $(document).on('click', '.remove-contact-btn', function() {
  96. var contactForm = $(this).closest('.contact-form');
  97. contactForm.remove();
  98. // Renumber remaining contacts
  99. $('#contacts-container .contact-form').each(function(index) {
  100. $(this).find('h3').text('联系人 #' + (index + 1));
  101. });
  102. });
  103. // Add initial contact form if none exists
  104. if ($('#contacts-container').children().length === 0) {
  105. $('.add-contact-btn').click();
  106. }
  107. });
  108. // Update method fields based on selection
  109. function updateMethodFields(methodRow) {
  110. var select = methodRow.find('select.method-select');
  111. var input = methodRow.find('input.method-input');
  112. var contactForm = methodRow.closest('.contact-form');
  113. var contactIndex = contactForm.attr('id').split('-')[2];
  114. var type = select.val();
  115. if (!type) return;
  116. // Count existing methods of this type
  117. var count = 1;
  118. contactForm.find('select.method-select').each(function() {
  119. if ($(this).val() === type && $(this)[0] !== select[0]) {
  120. count++;
  121. }
  122. });
  123. // Update field names
  124. select.attr('name', `contact[${contactIndex}][${type}_${count}]`);
  125. input.attr('name', `contact[${contactIndex}][${type}_${count}]`);
  126. // Add format field for tel and whatsapp
  127. if (type === 'tel' || type === 'whatsapp') {
  128. if (!methodRow.find('.format-input').length) {
  129. input.after(`<input type="hidden" class="format-input" name="contact[${contactIndex}][${type}_${count}_format]">`);
  130. }
  131. }
  132. // Add backup field
  133. if (!methodRow.find('.backup-input').length) {
  134. methodRow.append(`<input type="hidden" class="backup-input" name="contact[${contactIndex}][${type}_${count}_bu]">`);
  135. }
  136. }
  137. // Update available method types for a contact
  138. function updateAvailableMethodTypes(contactIndex) {
  139. var methodsContainer = $('#contact-methods-' + contactIndex);
  140. // Count methods by type
  141. var methodCounts = {};
  142. methodsContainer.find('select.method-select').each(function() {
  143. var type = $(this).val();
  144. if (type) {
  145. methodCounts[type] = (methodCounts[type] || 0) + 1;
  146. }
  147. });
  148. // Update all selects in this contact
  149. methodsContainer.find('select.method-select').each(function() {
  150. var currentValue = $(this).val();
  151. $(this).find('option').each(function() {
  152. var optionValue = $(this).val();
  153. if (optionValue && optionValue !== currentValue) {
  154. $(this).prop('disabled', (methodCounts[optionValue] || 0) >= 3);
  155. }
  156. });
  157. });
  158. }
  159. // Update placeholder and handle method fields
  160. function updateMethodSelectAndPlaceholder(selectElement) {
  161. var methodRow = $(selectElement).closest('.contact-method-row');
  162. updateMethodPlaceholder(selectElement);
  163. updateMethodFields(methodRow);
  164. var contactIndex = methodRow.closest('.contact-form').attr('id').split('-')[2];
  165. updateAvailableMethodTypes(contactIndex);
  166. }
  167. // Update method placeholder based on selected type
  168. function updateMethodPlaceholder(selectElement) {
  169. var placeholder = "";
  170. var value = $(selectElement).val();
  171. switch(value) {
  172. case "tel":
  173. placeholder = "电话格式必须为:区号+号码 如:+86 15012345678";
  174. break;
  175. case "wechat":
  176. placeholder = "微信";
  177. break;
  178. case "whatsapp":
  179. placeholder = "Whatsapp 格式必须为:区号+号码 如:+86 15012345678";
  180. break;
  181. case "email":
  182. placeholder = "邮箱格式必须正确,如: example@domain.com";
  183. break;
  184. case "linkedin":
  185. placeholder = "领英链接";
  186. break;
  187. case "facebook":
  188. placeholder = "Facebook";
  189. break;
  190. case "alibaba":
  191. placeholder = "阿里巴巴";
  192. break;
  193. default:
  194. placeholder = "请选择联系方式类型";
  195. }
  196. $(selectElement).next('.method-input').attr('placeholder', placeholder);
  197. }
  198. // Custom validation function for multiple contacts form with contact methods
  199. function validateMultipleContactsForm() {
  200. var clientCode = $("#cs_code").val();
  201. var clientCompany = $("#cs_company").val();
  202. var clientFrom = $("#cs_from").val();
  203. var clientCountry = $("#cs_country").val();
  204. // Validate basic customer info
  205. if (clientCode == "" || clientCode == null) {
  206. alert("客户代码不能为空!");
  207. $("#cs_code").focus();
  208. return false;
  209. }
  210. if (clientCountry == 0 || !clientCountry) {
  211. alert("这是哪个国家的客户?");
  212. $("#cs_country").focus();
  213. return false;
  214. }
  215. if (clientFrom == "0") {
  216. alert("请填写客户来源!");
  217. $("#cs_from").focus();
  218. return false;
  219. }
  220. // Validate that at least one business type is selected
  221. if (!$('input[name="cs_type[]"]:checked').length) {
  222. alert("请至少选择一种业务类型!");
  223. return false;
  224. }
  225. // Get source text to check if it's from Alibaba platforms
  226. var clientFromText = $("#cs_from option:selected").text();
  227. var isAlibabaSource = clientFromText.indexOf("1688") >= 0 ||
  228. clientFromText.indexOf("阿里") >= 0 ||
  229. clientFromText.indexOf("alibaba") >= 0;
  230. // Validate that at least one contact has at least one contact method
  231. var hasContactMethod = false;
  232. var hasAlibabaContact = false;
  233. var allContactsValid = true;
  234. var phoneRegex = /^\+\d{1,4}\s\d{5,}$/; // Regex to validate phone format: +[country code] [number]
  235. var emailRegex = /^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$/; // Regex to validate email format
  236. $('.contact-form').each(function(contactIndex) {
  237. var $form = $(this);
  238. var contactName = $form.find('input[name*="[contact_name]"]').val();
  239. var hasMethodInThisContact = false;
  240. // Check if this contact has methods
  241. $form.find('.contact-method-row').each(function() {
  242. var methodType = $(this).find('select.method-select').val();
  243. var methodValue = $(this).find('input.method-input').val();
  244. if (methodValue) {
  245. hasMethodInThisContact = true;
  246. hasContactMethod = true;
  247. // Check if there's an Alibaba contact method
  248. if (methodType === 'alibaba') {
  249. hasAlibabaContact = true;
  250. }
  251. }
  252. // Check if method type is selected but value is empty
  253. if (methodType && !methodValue) {
  254. alert("联系方式类型已选择但值为空");
  255. allContactsValid = false;
  256. return false;
  257. }
  258. // Validate phone number format for tel and whatsapp
  259. if ((methodType === 'tel' || methodType === 'whatsapp') && methodValue) {
  260. if (!phoneRegex.test(methodValue)) {
  261. alert("电话格式不正确,请使用以下格式:区号+号码,如 +86 15012345678");
  262. $(this).find('input.method-input').focus();
  263. allContactsValid = false;
  264. return false;
  265. }
  266. }
  267. // Validate email format
  268. if (methodType === 'email' && methodValue) {
  269. if (!emailRegex.test(methodValue)) {
  270. alert("邮箱格式不正确,请输入有效的邮箱地址");
  271. $(this).find('input.method-input').focus();
  272. allContactsValid = false;
  273. return false;
  274. }
  275. }
  276. });
  277. // If contact has a name but no methods, or has methods but no name
  278. if ((contactName && !hasMethodInThisContact) || (!contactName && hasMethodInThisContact)) {
  279. alert("联系人 #" + (contactIndex + 1) + " 缺少联系人姓名或联系方式");
  280. allContactsValid = false;
  281. return false;
  282. }
  283. // If contact has neither name nor methods, it's an empty contact
  284. if (!contactName && !hasMethodInThisContact) {
  285. alert("联系人 #" + (contactIndex + 1) + " 是空的,请填写信息或删除此联系人");
  286. allContactsValid = false;
  287. return false;
  288. }
  289. });
  290. if (!allContactsValid) {
  291. return false;
  292. }
  293. if (!hasContactMethod) {
  294. alert("至少需要添加一个联系人,且联系人至少需要一种联系方式!");
  295. return false;
  296. }
  297. // If source is from Alibaba platforms, must have Alibaba contact method
  298. if (isAlibabaSource && !hasAlibabaContact) {
  299. alert("客户来源为1688或阿里国际站时,必须添加至少一个阿里巴巴联系方式!");
  300. return false;
  301. }
  302. // Set tag values
  303. $("input#mytag").val($(".taglist").html());
  304. return true;
  305. }
  306. // Modified submission function
  307. function subform() {
  308. if (validateMultipleContactsForm()) {
  309. $("#form1").submit();
  310. }
  311. }
  312. </script>
  313. <style>
  314. body {
  315. margin: 0;
  316. padding: 20px;
  317. background: #fff;
  318. }
  319. #man_zone {
  320. margin-left: 0;
  321. }
  322. .contact-form {
  323. margin-bottom: 10px;
  324. /*border: 1px solid #ddd;*/
  325. padding: 8px;
  326. background-color: #FFFFFF;
  327. }
  328. .contact-header {
  329. display: flex;
  330. align-items: center;
  331. margin-bottom: 8px;
  332. gap: 10px;
  333. }
  334. .contact-header h3 {
  335. margin: 0;
  336. order: 2;
  337. flex-grow: 1;
  338. }
  339. .remove-contact-btn {
  340. background-color: #f44336;
  341. color: white;
  342. border: none;
  343. padding: 4px 8px;
  344. cursor: pointer;
  345. order: 1;
  346. }
  347. .add-contact-btn {
  348. background-color: #4CAF50;
  349. color: white;
  350. border: none;
  351. padding: 6px 12px;
  352. margin-bottom: 10px;
  353. cursor: pointer;
  354. }
  355. .contact-methods-container {
  356. margin-top: 8px;
  357. }
  358. .contact-method-row {
  359. margin-bottom: 6px;
  360. padding: 6px;
  361. /*border: 1px solid #eee;*/
  362. /*background-color: #f5f5f5;*/
  363. display: flex;
  364. align-items: center;
  365. gap: 8px;
  366. }
  367. .add-method-btn {
  368. background-color: #2196F3;
  369. color: white;
  370. border: none;
  371. padding: 4px 8px;
  372. margin-top: 4px;
  373. cursor: pointer;
  374. }
  375. .remove-method-btn {
  376. background-color: #f44336;
  377. color: white;
  378. border: none;
  379. padding: 2px 4px;
  380. cursor: pointer;
  381. }
  382. .method-select {
  383. margin-right: 8px;
  384. padding: 3px;
  385. }
  386. .contact-table {
  387. margin-bottom: 6px;
  388. }
  389. .contact-table td, .contact-table th {
  390. padding: 4px 6px;
  391. }
  392. </style>
  393. </head>
  394. <body class="clear">
  395. <?php // require_once 'panel.php'; ?>
  396. <div id="man_zone">
  397. <form name="form1" id="form1" method="post" action="customerSave.php<?= $hrefstr ?? '' ?>">
  398. <table width="100%" border="0" cellpadding="3" cellspacing="1" class="table1">
  399. <tbody>
  400. <tr>
  401. <th width="8%">客户编号</th>
  402. <td>
  403. <input type="text" id="cs_code" name="cs_code" value="" class="txt1" />
  404. <input type="hidden" name="cs_addtime" value="<?= date('Y-m-d H:i:s') ?>" />
  405. </td>
  406. </tr>
  407. <tr>
  408. <th width="8%">公司名称</th>
  409. <td><input type="text" id="cs_company" name="cs_company" value="" class="txt1" /></td>
  410. </tr>
  411. <tr>
  412. <th width="8%">地区/国家</th>
  413. <td>
  414. <div class="layui-input-inline">
  415. <div class="layui-form-select ySearchSelect y1">
  416. <div class="layui-input">请选择客户区域</div>
  417. <input name="cs_country" id="cs_country" type="hidden">
  418. <i class="layui-edge"></i>
  419. <ul>
  420. <?php
  421. $result = $conn->query("SELECT id, countryCode, countryName FROM country");
  422. while ($row = $result->fetch_assoc()) {
  423. echo "<li class=\"on\" data-c=\"{$row['id']}\">(+{$row['countryCode']}){$row['countryName']}</li>";
  424. }
  425. ?>
  426. <p>无匹配项</p>
  427. </ul>
  428. </div>
  429. </div>
  430. <script>
  431. $(function () {
  432. $(".y1").ySearchSelect();
  433. })
  434. </script>
  435. </td>
  436. </tr>
  437. <tr>
  438. <th width="8%">来源</th>
  439. <td>
  440. <select id="cs_from" name="cs_from">
  441. <option value="0">请选择来源</option>
  442. <?php
  443. $result = $conn->query("SELECT id, ch_name FROM qudao");
  444. while ($row = $result->fetch_assoc()) {
  445. echo "<option value=\"{$row['id']}\">{$row['ch_name']}</option>";
  446. }
  447. ?>
  448. </select>
  449. </td>
  450. </tr>
  451. <tr>
  452. <th width="8%" valign="top">联系人信息</th>
  453. <td>
  454. <button type="button" class="add-contact-btn">添加联系人</button>
  455. <div id="contacts-container">
  456. <!-- Contact forms will be added here -->
  457. </div>
  458. </td>
  459. </tr>
  460. <tr>
  461. <th>地址</th>
  462. <td><input type="text" id="cs_address" name="cs_address" value="" class="txt1" /></td>
  463. </tr>
  464. <tr>
  465. <th>业务类型</th>
  466. <td>
  467. <?php
  468. $result = $conn->query("SELECT id, businessType FROM clienttype");
  469. while ($row = $result->fetch_assoc()) {
  470. echo "<input type=\"checkbox\" name=\"cs_type[]\" value=\"{$row['id']}\" id=\"fortype{$row['id']}\">
  471. <label for=\"fortype{$row['id']}\">{$row['businessType']}</label>";
  472. }
  473. ?>
  474. </td>
  475. </tr>
  476. <tr>
  477. <th>跟进阶段</th>
  478. <td>
  479. <input type="radio" id="fordeal1" class="cs_deal" name="cs_deal" value="0"><label for="fordeal1">无响应</label>
  480. <input type="radio" id="fordeal2" class="cs_deal" name="cs_deal" value="1" checked="checked"><label for="fordeal2">背景调查</label>
  481. <input type="radio" id="fordeal3" class="cs_deal" name="cs_deal" value="2"><label for="fordeal3">明确需求</label>
  482. <input type="radio" id="fordeal4" class="cs_deal" name="cs_deal" value="3"><label for="fordeal4">已成交</label>
  483. </td>
  484. </tr>
  485. <tr>
  486. <th>其他</th>
  487. <td>
  488. <input type="checkbox" id="belongClient" class="cs_belongClient" name="cs_belongClient" value="1">
  489. <label for="belongClient">客户的客户</label>
  490. </td>
  491. </tr>
  492. <tr>
  493. <th>自定义标签</th>
  494. <td>
  495. <div class="taglist"></div>
  496. <input type="hidden" id="mytag" name="mytag" value="">
  497. <div class="commontag clear">
  498. <i class="tag">美特柏品牌客户</i>,
  499. <i class="tag">OEM定制客户</i>,
  500. <i class="tag">小型B端客户</i>,
  501. <i class="tag">C端客户</i>,
  502. <i class="tag">贸易公司</i>,
  503. <i class="tag">档口客户</i>
  504. <?php
  505. // 将bind_param改为SQL拼接
  506. $employee_id = intval($_SESSION['employee_id']);
  507. $sql = "SELECT DISTINCT tagName FROM tagtable WHERE employeeId = ".$employee_id;
  508. $result = $conn->query($sql);
  509. while ($row = $result->fetch_assoc()) {
  510. echo "<i class=\"tag\">" . htmlspecialcharsFix(textUncode($row['tagName'])) . "</i>,";
  511. }
  512. ?>
  513. </div>
  514. <input type="text" id="tapinput" class="txt-short" placeholder="添加新标签,按Enter添加">
  515. </td>
  516. </tr>
  517. <tr>
  518. <th width="8%">备注</th>
  519. <td><textarea name="cs_note" class="txt2" placeholder=""></textarea></td>
  520. </tr>
  521. <tr>
  522. <th></th>
  523. <td>
  524. <input type="button" name="save" id="save" value="确定" class="btn1" onclick="subform();" />
  525. <input type="button" value="返回" class="btn1" onClick="location.href='customers.php'" />
  526. </td>
  527. </tr>
  528. </tbody>
  529. </table>
  530. </form>
  531. </div>
  532. </body>
  533. </html>