集成说明.html 57 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  1. <!DOCTYPE html><html><head>
  2. <title>集成说明</title>
  3. <meta charset="utf-8">
  4. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  5. <link rel="stylesheet" href="file:////Users/Zhuanz/.cursor/extensions/shd101wyy.markdown-preview-enhanced-0.8.20-universal/crossnote/dependencies/katex/katex.min.css">
  6. <style>
  7. code[class*=language-],pre[class*=language-]{color:#333;background:0 0;font-family:Consolas,"Liberation Mono",Menlo,Courier,monospace;text-align:left;white-space:pre;word-spacing:normal;word-break:normal;word-wrap:normal;line-height:1.4;-moz-tab-size:8;-o-tab-size:8;tab-size:8;-webkit-hyphens:none;-moz-hyphens:none;-ms-hyphens:none;hyphens:none}pre[class*=language-]{padding:.8em;overflow:auto;border-radius:3px;background:#f5f5f5}:not(pre)>code[class*=language-]{padding:.1em;border-radius:.3em;white-space:normal;background:#f5f5f5}.token.blockquote,.token.comment{color:#969896}.token.cdata{color:#183691}.token.doctype,.token.macro.property,.token.punctuation,.token.variable{color:#333}.token.builtin,.token.important,.token.keyword,.token.operator,.token.rule{color:#a71d5d}.token.attr-value,.token.regex,.token.string,.token.url{color:#183691}.token.atrule,.token.boolean,.token.code,.token.command,.token.constant,.token.entity,.token.number,.token.property,.token.symbol{color:#0086b3}.token.prolog,.token.selector,.token.tag{color:#63a35c}.token.attr-name,.token.class,.token.class-name,.token.function,.token.id,.token.namespace,.token.pseudo-class,.token.pseudo-element,.token.url-reference .token.variable{color:#795da3}.token.entity{cursor:help}.token.title,.token.title .token.punctuation{font-weight:700;color:#1d3e81}.token.list{color:#ed6a43}.token.inserted{background-color:#eaffea;color:#55a532}.token.deleted{background-color:#ffecec;color:#bd2c00}.token.bold{font-weight:700}.token.italic{font-style:italic}.language-json .token.property{color:#183691}.language-markup .token.tag .token.punctuation{color:#333}.language-css .token.function,code.language-css{color:#0086b3}.language-yaml .token.atrule{color:#63a35c}code.language-yaml{color:#183691}.language-ruby .token.function{color:#333}.language-markdown .token.url{color:#795da3}.language-makefile .token.symbol{color:#795da3}.language-makefile .token.variable{color:#183691}.language-makefile .token.builtin{color:#0086b3}.language-bash .token.keyword{color:#0086b3}pre[data-line]{position:relative;padding:1em 0 1em 3em}pre[data-line] .line-highlight-wrapper{position:absolute;top:0;left:0;background-color:transparent;display:block;width:100%}pre[data-line] .line-highlight{position:absolute;left:0;right:0;padding:inherit 0;margin-top:1em;background:hsla(24,20%,50%,.08);background:linear-gradient(to right,hsla(24,20%,50%,.1) 70%,hsla(24,20%,50%,0));pointer-events:none;line-height:inherit;white-space:pre}pre[data-line] .line-highlight:before,pre[data-line] .line-highlight[data-end]:after{content:attr(data-start);position:absolute;top:.4em;left:.6em;min-width:1em;padding:0 .5em;background-color:hsla(24,20%,50%,.4);color:#f4f1ef;font:bold 65%/1.5 sans-serif;text-align:center;vertical-align:.3em;border-radius:999px;text-shadow:none;box-shadow:0 1px #fff}pre[data-line] .line-highlight[data-end]:after{content:attr(data-end);top:auto;bottom:.4em}html body{font-family:'Helvetica Neue',Helvetica,'Segoe UI',Arial,freesans,sans-serif;font-size:16px;line-height:1.6;color:#333;background-color:#fff;overflow:initial;box-sizing:border-box;word-wrap:break-word}html body>:first-child{margin-top:0}html body h1,html body h2,html body h3,html body h4,html body h5,html body h6{line-height:1.2;margin-top:1em;margin-bottom:16px;color:#000}html body h1{font-size:2.25em;font-weight:300;padding-bottom:.3em}html body h2{font-size:1.75em;font-weight:400;padding-bottom:.3em}html body h3{font-size:1.5em;font-weight:500}html body h4{font-size:1.25em;font-weight:600}html body h5{font-size:1.1em;font-weight:600}html body h6{font-size:1em;font-weight:600}html body h1,html body h2,html body h3,html body h4,html body h5{font-weight:600}html body h5{font-size:1em}html body h6{color:#5c5c5c}html body strong{color:#000}html body del{color:#5c5c5c}html body a:not([href]){color:inherit;text-decoration:none}html body a{color:#08c;text-decoration:none}html body a:hover{color:#00a3f5;text-decoration:none}html body img{max-width:100%}html body>p{margin-top:0;margin-bottom:16px;word-wrap:break-word}html body>ol,html body>ul{margin-bottom:16px}html body ol,html body ul{padding-left:2em}html body ol.no-list,html body ul.no-list{padding:0;list-style-type:none}html body ol ol,html body ol ul,html body ul ol,html body ul ul{margin-top:0;margin-bottom:0}html body li{margin-bottom:0}html body li.task-list-item{list-style:none}html body li>p{margin-top:0;margin-bottom:0}html body .task-list-item-checkbox{margin:0 .2em .25em -1.8em;vertical-align:middle}html body .task-list-item-checkbox:hover{cursor:pointer}html body blockquote{margin:16px 0;font-size:inherit;padding:0 15px;color:#5c5c5c;background-color:#f0f0f0;border-left:4px solid #d6d6d6}html body blockquote>:first-child{margin-top:0}html body blockquote>:last-child{margin-bottom:0}html body hr{height:4px;margin:32px 0;background-color:#d6d6d6;border:0 none}html body table{margin:10px 0 15px 0;border-collapse:collapse;border-spacing:0;display:block;width:100%;overflow:auto;word-break:normal;word-break:keep-all}html body table th{font-weight:700;color:#000}html body table td,html body table th{border:1px solid #d6d6d6;padding:6px 13px}html body dl{padding:0}html body dl dt{padding:0;margin-top:16px;font-size:1em;font-style:italic;font-weight:700}html body dl dd{padding:0 16px;margin-bottom:16px}html body code{font-family:Menlo,Monaco,Consolas,'Courier New',monospace;font-size:.85em;color:#000;background-color:#f0f0f0;border-radius:3px;padding:.2em 0}html body code::after,html body code::before{letter-spacing:-.2em;content:'\00a0'}html body pre>code{padding:0;margin:0;word-break:normal;white-space:pre;background:0 0;border:0}html body .highlight{margin-bottom:16px}html body .highlight pre,html body pre{padding:1em;overflow:auto;line-height:1.45;border:#d6d6d6;border-radius:3px}html body .highlight pre{margin-bottom:0;word-break:normal}html body pre code,html body pre tt{display:inline;max-width:initial;padding:0;margin:0;overflow:initial;line-height:inherit;word-wrap:normal;background-color:transparent;border:0}html body pre code:after,html body pre code:before,html body pre tt:after,html body pre tt:before{content:normal}html body blockquote,html body dl,html body ol,html body p,html body pre,html body ul{margin-top:0;margin-bottom:16px}html body kbd{color:#000;border:1px solid #d6d6d6;border-bottom:2px solid #c7c7c7;padding:2px 4px;background-color:#f0f0f0;border-radius:3px}@media print{html body{background-color:#fff}html body h1,html body h2,html body h3,html body h4,html body h5,html body h6{color:#000;page-break-after:avoid}html body blockquote{color:#5c5c5c}html body pre{page-break-inside:avoid}html body table{display:table}html body img{display:block;max-width:100%;max-height:100%}html body code,html body pre{word-wrap:break-word;white-space:pre}}.markdown-preview{width:100%;height:100%;box-sizing:border-box}.markdown-preview ul{list-style:disc}.markdown-preview ul ul{list-style:circle}.markdown-preview ul ul ul{list-style:square}.markdown-preview ol{list-style:decimal}.markdown-preview ol ol,.markdown-preview ul ol{list-style-type:lower-roman}.markdown-preview ol ol ol,.markdown-preview ol ul ol,.markdown-preview ul ol ol,.markdown-preview ul ul ol{list-style-type:lower-alpha}.markdown-preview .newpage,.markdown-preview .pagebreak{page-break-before:always}.markdown-preview pre.line-numbers{position:relative;padding-left:3.8em;counter-reset:linenumber}.markdown-preview pre.line-numbers>code{position:relative}.markdown-preview pre.line-numbers .line-numbers-rows{position:absolute;pointer-events:none;top:1em;font-size:100%;left:0;width:3em;letter-spacing:-1px;border-right:1px solid #999;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.markdown-preview pre.line-numbers .line-numbers-rows>span{pointer-events:none;display:block;counter-increment:linenumber}.markdown-preview pre.line-numbers .line-numbers-rows>span:before{content:counter(linenumber);color:#999;display:block;padding-right:.8em;text-align:right}.markdown-preview .mathjax-exps .MathJax_Display{text-align:center!important}.markdown-preview:not([data-for=preview]) .code-chunk .code-chunk-btn-group{display:none}.markdown-preview:not([data-for=preview]) .code-chunk .status{display:none}.markdown-preview:not([data-for=preview]) .code-chunk .output-div{margin-bottom:16px}.markdown-preview .md-toc{padding:0}.markdown-preview .md-toc .md-toc-link-wrapper .md-toc-link{display:inline;padding:.25rem 0}.markdown-preview .md-toc .md-toc-link-wrapper .md-toc-link div,.markdown-preview .md-toc .md-toc-link-wrapper .md-toc-link p{display:inline}.markdown-preview .md-toc .md-toc-link-wrapper.highlighted .md-toc-link{font-weight:800}.scrollbar-style::-webkit-scrollbar{width:8px}.scrollbar-style::-webkit-scrollbar-track{border-radius:10px;background-color:transparent}.scrollbar-style::-webkit-scrollbar-thumb{border-radius:5px;background-color:rgba(150,150,150,.66);border:4px solid rgba(150,150,150,.66);background-clip:content-box}html body[for=html-export]:not([data-presentation-mode]){position:relative;width:100%;height:100%;top:0;left:0;margin:0;padding:0;overflow:auto}html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{position:relative;top:0;min-height:100vh}@media screen and (min-width:914px){html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{padding:2em calc(50% - 457px + 2em)}}@media screen and (max-width:914px){html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{padding:2em}}@media screen and (max-width:450px){html body[for=html-export]:not([data-presentation-mode]) .markdown-preview{font-size:14px!important;padding:1em}}@media print{html body[for=html-export]:not([data-presentation-mode]) #sidebar-toc-btn{display:none}}html body[for=html-export]:not([data-presentation-mode]) #sidebar-toc-btn{position:fixed;bottom:8px;left:8px;font-size:28px;cursor:pointer;color:inherit;z-index:99;width:32px;text-align:center;opacity:.4}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] #sidebar-toc-btn{opacity:1}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc{position:fixed;top:0;left:0;width:300px;height:100%;padding:32px 0 48px 0;font-size:14px;box-shadow:0 0 4px rgba(150,150,150,.33);box-sizing:border-box;overflow:auto;background-color:inherit}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc::-webkit-scrollbar{width:8px}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc::-webkit-scrollbar-track{border-radius:10px;background-color:transparent}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc::-webkit-scrollbar-thumb{border-radius:5px;background-color:rgba(150,150,150,.66);border:4px solid rgba(150,150,150,.66);background-clip:content-box}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc a{text-decoration:none}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc{padding:0 16px}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper .md-toc-link{display:inline;padding:.25rem 0}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper .md-toc-link div,html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper .md-toc-link p{display:inline}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .md-sidebar-toc .md-toc .md-toc-link-wrapper.highlighted .md-toc-link{font-weight:800}html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .markdown-preview{left:300px;width:calc(100% - 300px);padding:2em calc(50% - 457px - 300px / 2);margin:0;box-sizing:border-box}@media screen and (max-width:1274px){html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .markdown-preview{padding:2em}}@media screen and (max-width:450px){html body[for=html-export]:not([data-presentation-mode])[html-show-sidebar-toc] .markdown-preview{width:100%}}html body[for=html-export]:not([data-presentation-mode]):not([html-show-sidebar-toc]) .markdown-preview{left:50%;transform:translateX(-50%)}html body[for=html-export]:not([data-presentation-mode]):not([html-show-sidebar-toc]) .md-sidebar-toc{display:none}
  8. /* Please visit the URL below for more information: */
  9. /* https://shd101wyy.github.io/markdown-preview-enhanced/#/customize-css */
  10. </style>
  11. <!-- The content below will be included at the end of the <head> element. --><script type="text/javascript">
  12. document.addEventListener("DOMContentLoaded", function () {
  13. // your code here
  14. });
  15. </script></head><body for="html-export">
  16. <div class="crossnote markdown-preview ">
  17. <h1 id="蓝牙ble模块集成说明">蓝牙BLE模块集成说明 </h1>
  18. <h2 id="目录结构">目录结构 </h2>
  19. <pre data-role="codeBlock" data-info="" class="language-text"><code>capability-ble/src/
  20. └── main/
  21. ├── AndroidManifest.xml # Android 清单文件(权限声明)
  22. └── java/
  23. └── com/narutohuo/xindazhou/ble/
  24. ├── api/ # API 接口层
  25. │ └── BLEService.kt # 蓝牙服务接口定义
  26. ├── factory/ # 工厂类层
  27. │ └── BLEServiceFactory.kt # 服务工厂(统一入口)
  28. ├── impl/ # 实现层
  29. │ └── BLEServiceImpl.kt # 蓝牙服务具体实现(单例模式)
  30. ├── callback/ # 回调接口层
  31. │ └── BLECallback.kt # 蓝牙回调接口(包含 ConnectCallback、HandshakeCallback、CommandCallback、DataCallback)
  32. ├── config/ # 配置层
  33. │ └── BLEConstants.kt # 蓝牙常量定义(用户类型、功能码、UUID等)
  34. ├── model/ # 数据模型层
  35. │ ├── BLEDevice.kt # BLE设备模型
  36. │ ├── BLECommand.kt # BLE指令模型(包含 AppControlInstruction、SystemControlInstruction)
  37. │ ├── BLEResponse.kt # BLE响应模型
  38. │ └── Command.kt # 指令协议数据类
  39. └── util/ # 工具类层
  40. ├── BleScanner.kt # BLE扫描工具
  41. ├── BleConnector.kt # BLE连接工具
  42. ├── BLECrypto.kt # 加密解密工具(AES)
  43. ├── BLEPacketBuilder.kt # 协议包构建工具
  44. ├── BLEPacketParser.kt # 协议包解析工具
  45. ├── BLEPacketSplitter.kt # 协议包分包工具
  46. └── BlePacketSender.kt # 协议包发送工具
  47. </code></pre><h2 id="功能说明">功能说明 </h2>
  48. <h3 id="1-api-层---接口定义">1. api/ 层 - 接口定义 </h3>
  49. <ul>
  50. <li><strong>职责</strong>:定义蓝牙服务的公共接口,业务层只依赖接口</li>
  51. <li><strong>内容</strong>:<code>BLEService</code> 接口
  52. <ul>
  53. <li><strong>基础连接功能</strong>:
  54. <ul>
  55. <li><code>startScan()</code> - 开始扫描BLE设备</li>
  56. <li><code>stopScan()</code> - 停止扫描</li>
  57. <li><code>connect()</code> - 连接设备(自动完成握手)</li>
  58. <li><code>disconnect()</code> - 断开连接</li>
  59. <li><code>isConnected()</code> - 检查连接状态</li>
  60. </ul>
  61. </li>
  62. <li><strong>协议封装功能</strong>:
  63. <ul>
  64. <li><code>sendCommand()</code> - 发送自定义指令</li>
  65. <li><code>sendAppControlCommand()</code> - 发送应用控制指令</li>
  66. <li><code>sendSystemControlCommand()</code> - 发送系统控制指令</li>
  67. <li><code>queryVehicleInfo()</code> - 查询车辆信息</li>
  68. </ul>
  69. </li>
  70. <li><strong>便捷方法</strong>:
  71. <ul>
  72. <li><code>setDefense()</code>, <code>powerOn()</code>, <code>powerOff()</code>, <code>findCar()</code></li>
  73. <li><code>unlockSeat()</code>, <code>unlockHandlebar()</code>, <code>lockHandlebar()</code></li>
  74. <li><code>unlockTrunk()</code>, <code>lockTrunk()</code></li>
  75. <li><code>setSensorUnlock()</code>, <code>setAmbientLight()</code>, <code>setAmbientLightColor()</code></li>
  76. <li><code>setAutoPowerOffTime()</code>, <code>setSpeakerVolume()</code></li>
  77. </ul>
  78. </li>
  79. <li><strong>数据接收监听</strong>:
  80. <ul>
  81. <li><code>setDataReceivedListener()</code> - 设置数据接收监听器</li>
  82. </ul>
  83. </li>
  84. </ul>
  85. </li>
  86. </ul>
  87. <h3 id="2-factory-层---工厂类">2. factory/ 层 - 工厂类 </h3>
  88. <ul>
  89. <li><strong>职责</strong>:提供统一的服务实例获取方式</li>
  90. <li><strong>内容</strong>:<code>BLEServiceFactory</code> 对象(单例)</li>
  91. <li><strong>作用</strong>:
  92. <ul>
  93. <li>隐藏实现细节,统一入口</li>
  94. <li>提供 <code>create(context)</code> 方法获取服务实例</li>
  95. </ul>
  96. </li>
  97. </ul>
  98. <h3 id="3-impl-层---具体实现">3. impl/ 层 - 具体实现 </h3>
  99. <ul>
  100. <li><strong>职责</strong>:实现 <code>BLEService</code> 接口的具体逻辑</li>
  101. <li><strong>内容</strong>:<code>BLEServiceImpl</code> 单例实现类</li>
  102. <li><strong>功能</strong>:
  103. <ul>
  104. <li>封装 Android BLE SDK 调用</li>
  105. <li>封装 GATT 连接管理</li>
  106. <li>封装握手认证(连接时自动完成)</li>
  107. <li>封装数据分包和组包</li>
  108. <li>封装 AES 加密/解密</li>
  109. <li>封装协议包构建和解析</li>
  110. </ul>
  111. </li>
  112. </ul>
  113. <h3 id="4-model-层---数据模型">4. model/ 层 - 数据模型 </h3>
  114. <ul>
  115. <li><strong>职责</strong>:定义蓝牙相关的数据模型</li>
  116. <li><strong>内容</strong>:
  117. <ul>
  118. <li><code>BLEDevice</code> - BLE设备模型</li>
  119. <li><code>BLECommand</code> - BLE指令模型</li>
  120. <li><code>BLEResponse</code> - BLE响应模型</li>
  121. <li><code>Command</code> - 指令协议数据类</li>
  122. <li><code>AppControlInstruction</code> - 应用控制指令常量(对象)</li>
  123. <li><code>SystemControlInstruction</code> - 系统控制指令常量(对象)</li>
  124. </ul>
  125. </li>
  126. </ul>
  127. <h3 id="5-callback-层---回调接口">5. callback/ 层 - 回调接口 </h3>
  128. <ul>
  129. <li><strong>职责</strong>:定义蓝牙相关的回调接口</li>
  130. <li><strong>内容</strong>:
  131. <ul>
  132. <li><code>BLECallback.kt</code> 文件中包含:
  133. <ul>
  134. <li><code>ConnectCallback</code> - 连接回调接口</li>
  135. <li><code>HandshakeCallback</code> - 握手指令回调接口</li>
  136. <li><code>CommandCallback</code> - 指令发送回调接口</li>
  137. <li><code>DataCallback</code> - 数据接收回调接口</li>
  138. </ul>
  139. </li>
  140. </ul>
  141. </li>
  142. </ul>
  143. <h3 id="6-config-层---配置">6. config/ 层 - 配置 </h3>
  144. <ul>
  145. <li><strong>职责</strong>:定义蓝牙相关的常量配置</li>
  146. <li><strong>内容</strong>:
  147. <ul>
  148. <li><code>BLEConstants</code> - 蓝牙常量定义(用户类型、功能码、UUID等)</li>
  149. </ul>
  150. </li>
  151. </ul>
  152. <h3 id="7-util-层---工具类">7. util/ 层 - 工具类 </h3>
  153. <ul>
  154. <li><strong>职责</strong>:提供蓝牙相关的工具方法</li>
  155. <li><strong>内容</strong>:
  156. <ul>
  157. <li><code>BleScanner</code> - BLE扫描工具</li>
  158. <li><code>BleConnector</code> - BLE连接工具</li>
  159. <li><code>BLECrypto</code> - 加密解密工具</li>
  160. <li><code>BLEPacketBuilder</code> - 协议包构建工具</li>
  161. <li><code>BLEPacketParser</code> - 协议包解析工具</li>
  162. <li><code>BLEPacketSplitter</code> - 协议包分包工具</li>
  163. <li><code>BlePacketSender</code> - 协议包发送工具</li>
  164. </ul>
  165. </li>
  166. </ul>
  167. <h2 id="架构设计">架构设计 </h2>
  168. <ul>
  169. <li><strong>接口隔离</strong>:业务层只依赖接口,不依赖实现</li>
  170. <li><strong>工厂模式</strong>:统一入口,隐藏创建逻辑</li>
  171. <li><strong>单例模式</strong>:<code>BLEServiceImpl</code> 使用单例模式,全局共享同一个连接状态</li>
  172. <li><strong>完全封装</strong>:所有底层细节(连接、握手、分包、加密等)都在能力层内部完成</li>
  173. </ul>
  174. <h2 id="使用方式">使用方式 </h2>
  175. <h3 id="完整示例单例模式">完整示例(单例模式) </h3>
  176. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token keyword keyword-import">import</span> androidx<span class="token punctuation">.</span>fragment<span class="token punctuation">.</span>app<span class="token punctuation">.</span>Fragment
  177. <span class="token keyword keyword-import">import</span> android<span class="token punctuation">.</span>widget<span class="token punctuation">.</span>Toast
  178. <span class="token keyword keyword-import">import</span> com<span class="token punctuation">.</span>narutohuo<span class="token punctuation">.</span>xindazhou<span class="token punctuation">.</span>ble<span class="token punctuation">.</span>factory<span class="token punctuation">.</span>BLEServiceFactory
  179. <span class="token keyword keyword-import">import</span> com<span class="token punctuation">.</span>narutohuo<span class="token punctuation">.</span>xindazhou<span class="token punctuation">.</span>ble<span class="token punctuation">.</span>api<span class="token punctuation">.</span>BLEService
  180. <span class="token keyword keyword-import">import</span> com<span class="token punctuation">.</span>narutohuo<span class="token punctuation">.</span>xindazhou<span class="token punctuation">.</span>ble<span class="token punctuation">.</span>config<span class="token punctuation">.</span>BLEConstants
  181. <span class="token keyword keyword-class">class</span> VehicleFragment <span class="token operator">:</span> <span class="token function">Fragment</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  182. <span class="token comment">// 1. 获取单例实例(全局只有一个,在任何地方获取的都是同一个)</span>
  183. <span class="token keyword keyword-private">private</span> <span class="token keyword keyword-val">val</span> bleService<span class="token operator">:</span> BLEService <span class="token operator">=</span> BLEServiceFactory<span class="token punctuation">.</span><span class="token function">create</span><span class="token punctuation">(</span><span class="token function">requireContext</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span>
  184. <span class="token keyword keyword-override">override</span> <span class="token keyword keyword-fun">fun</span> <span class="token function">onViewCreated</span><span class="token punctuation">(</span>view<span class="token operator">:</span> View<span class="token punctuation">,</span> savedInstanceState<span class="token operator">:</span> Bundle<span class="token operator">?</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  185. <span class="token keyword keyword-super">super</span><span class="token punctuation">.</span><span class="token function">onViewCreated</span><span class="token punctuation">(</span>view<span class="token punctuation">,</span> savedInstanceState<span class="token punctuation">)</span>
  186. <span class="token comment">// 2. 连接设备(一键完成扫描+连接+握手)</span>
  187. <span class="token function">connectDevice</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  188. <span class="token punctuation">}</span>
  189. <span class="token keyword keyword-private">private</span> <span class="token keyword keyword-fun">fun</span> <span class="token function">connectDevice</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  190. <span class="token comment">// 获取用户信息(实际项目中从用户认证信息中获取)</span>
  191. <span class="token keyword keyword-val">val</span> userId <span class="token operator">=</span> <span class="token function">getUserId</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token comment">// 16字节</span>
  192. <span class="token keyword keyword-val">val</span> userType <span class="token operator">=</span> BLEConstants<span class="token punctuation">.</span>USER_TYPE_OWNER <span class="token comment">// 0x02=车主</span>
  193. bleService<span class="token punctuation">.</span><span class="token function">initializeAndConnect</span><span class="token punctuation">(</span>
  194. userId <span class="token operator">=</span> userId<span class="token punctuation">,</span>
  195. userType <span class="token operator">=</span> userType<span class="token punctuation">,</span>
  196. onConnected <span class="token operator">=</span> <span class="token punctuation">{</span>
  197. activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
  198. <span class="token comment">// 连接成功,可以直接使用指令了</span>
  199. Toast<span class="token punctuation">.</span><span class="token function">makeText</span><span class="token punctuation">(</span>context<span class="token punctuation">,</span> <span class="token string-literal singleline"><span class="token string">"连接成功"</span></span><span class="token punctuation">,</span> Toast<span class="token punctuation">.</span>LENGTH_SHORT<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">show</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  200. <span class="token punctuation">}</span>
  201. <span class="token punctuation">}</span><span class="token punctuation">,</span>
  202. onError <span class="token operator">=</span> <span class="token punctuation">{</span> error <span class="token operator">-&gt;</span>
  203. activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
  204. Toast<span class="token punctuation">.</span><span class="token function">makeText</span><span class="token punctuation">(</span>context<span class="token punctuation">,</span> <span class="token string-literal singleline"><span class="token string">"连接失败: </span><span class="token interpolation"><span class="token interpolation-punctuation punctuation">$</span><span class="token expression">error</span></span><span class="token string">"</span></span><span class="token punctuation">,</span> Toast<span class="token punctuation">.</span>LENGTH_SHORT<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">show</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  205. <span class="token punctuation">}</span>
  206. <span class="token punctuation">}</span>
  207. <span class="token punctuation">)</span>
  208. <span class="token punctuation">}</span>
  209. <span class="token comment">// 3. 发送指令(连接成功后调用)</span>
  210. <span class="token keyword keyword-private">private</span> <span class="token keyword keyword-fun">fun</span> <span class="token function">lockCar</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  211. <span class="token comment">// 设防</span>
  212. bleService<span class="token punctuation">.</span><span class="token function">setDefense</span><span class="token punctuation">(</span>enabled <span class="token operator">=</span> <span class="token boolean">true</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> response <span class="token operator">-&gt;</span>
  213. activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
  214. <span class="token keyword keyword-if">if</span> <span class="token punctuation">(</span>response<span class="token punctuation">.</span>success<span class="token punctuation">)</span> <span class="token punctuation">{</span>
  215. Toast<span class="token punctuation">.</span><span class="token function">makeText</span><span class="token punctuation">(</span>context<span class="token punctuation">,</span> <span class="token string-literal singleline"><span class="token string">"设防成功"</span></span><span class="token punctuation">,</span> Toast<span class="token punctuation">.</span>LENGTH_SHORT<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">show</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  216. <span class="token punctuation">}</span> <span class="token keyword keyword-else">else</span> <span class="token punctuation">{</span>
  217. Toast<span class="token punctuation">.</span><span class="token function">makeText</span><span class="token punctuation">(</span>context<span class="token punctuation">,</span> <span class="token string-literal singleline"><span class="token string">"设防失败: </span><span class="token interpolation"><span class="token interpolation-punctuation punctuation">${</span><span class="token expression">response<span class="token punctuation">.</span>errorMessage</span><span class="token interpolation-punctuation punctuation">}</span></span><span class="token string">"</span></span><span class="token punctuation">,</span> Toast<span class="token punctuation">.</span>LENGTH_SHORT<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">show</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  218. <span class="token punctuation">}</span>
  219. <span class="token punctuation">}</span>
  220. <span class="token punctuation">}</span>
  221. <span class="token punctuation">}</span>
  222. <span class="token keyword keyword-private">private</span> <span class="token keyword keyword-fun">fun</span> <span class="token function">powerOn</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  223. <span class="token comment">// 上电</span>
  224. bleService<span class="token punctuation">.</span><span class="token function">powerOn</span> <span class="token punctuation">{</span> response <span class="token operator">-&gt;</span>
  225. activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
  226. <span class="token keyword keyword-if">if</span> <span class="token punctuation">(</span>response<span class="token punctuation">.</span>success<span class="token punctuation">)</span> <span class="token punctuation">{</span>
  227. Toast<span class="token punctuation">.</span><span class="token function">makeText</span><span class="token punctuation">(</span>context<span class="token punctuation">,</span> <span class="token string-literal singleline"><span class="token string">"上电成功"</span></span><span class="token punctuation">,</span> Toast<span class="token punctuation">.</span>LENGTH_SHORT<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">show</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  228. <span class="token punctuation">}</span>
  229. <span class="token punctuation">}</span>
  230. <span class="token punctuation">}</span>
  231. <span class="token punctuation">}</span>
  232. <span class="token keyword keyword-private">private</span> <span class="token keyword keyword-fun">fun</span> <span class="token function">findCar</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  233. <span class="token comment">// 寻车</span>
  234. bleService<span class="token punctuation">.</span><span class="token function">findCar</span> <span class="token punctuation">{</span> response <span class="token operator">-&gt;</span>
  235. activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
  236. <span class="token keyword keyword-if">if</span> <span class="token punctuation">(</span>response<span class="token punctuation">.</span>success<span class="token punctuation">)</span> <span class="token punctuation">{</span>
  237. Toast<span class="token punctuation">.</span><span class="token function">makeText</span><span class="token punctuation">(</span>context<span class="token punctuation">,</span> <span class="token string-literal singleline"><span class="token string">"寻车成功"</span></span><span class="token punctuation">,</span> Toast<span class="token punctuation">.</span>LENGTH_SHORT<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">show</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  238. <span class="token punctuation">}</span>
  239. <span class="token punctuation">}</span>
  240. <span class="token punctuation">}</span>
  241. <span class="token punctuation">}</span>
  242. <span class="token keyword keyword-private">private</span> <span class="token keyword keyword-fun">fun</span> <span class="token function">getUserId</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token operator">:</span> ByteArray <span class="token punctuation">{</span>
  243. <span class="token comment">// 实际项目中从用户认证信息中获取</span>
  244. <span class="token keyword keyword-return">return</span> <span class="token function">ByteArray</span><span class="token punctuation">(</span><span class="token number">16</span><span class="token punctuation">)</span> <span class="token punctuation">{</span> it<span class="token punctuation">.</span><span class="token function">toByte</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">}</span> <span class="token comment">// 示例数据</span>
  245. <span class="token punctuation">}</span>
  246. <span class="token keyword keyword-override">override</span> <span class="token keyword keyword-fun">fun</span> <span class="token function">onDestroyView</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  247. <span class="token keyword keyword-super">super</span><span class="token punctuation">.</span><span class="token function">onDestroyView</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  248. <span class="token comment">// 4. 断开连接(可选,单例会保持连接状态)</span>
  249. <span class="token keyword keyword-if">if</span> <span class="token punctuation">(</span>bleService<span class="token punctuation">.</span><span class="token function">isConnected</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
  250. bleService<span class="token punctuation">.</span><span class="token function">disconnect</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  251. <span class="token punctuation">}</span>
  252. <span class="token punctuation">}</span>
  253. <span class="token punctuation">}</span>
  254. </code></pre><h3 id="步骤说明">步骤说明 </h3>
  255. <ol>
  256. <li><strong>获取单例实例</strong>:使用 <code>BLEServiceFactory.create(context)</code> 获取服务实例(全局只有一个)</li>
  257. <li><strong>连接设备</strong>:调用 <code>initializeAndConnect()</code> 连接设备(自动完成扫描+连接+握手)</li>
  258. <li><strong>发送指令</strong>:连接成功后,调用各种指令方法(如 <code>setDefense()</code>、<code>powerOn()</code>、<code>findCar()</code> 等)</li>
  259. <li><strong>断开连接</strong>:在 <code>onDestroyView()</code> 中断开连接(可选)</li>
  260. </ol>
  261. <h3 id="重要说明">重要说明 </h3>
  262. <ol>
  263. <li><strong>单例模式</strong>:<code>BLEService</code> 是单例,全局只有一个实例,在任何地方获取的都是同一个</li>
  264. <li><strong>连接状态共享</strong>:所有使用该服务的地方共享同一个连接状态</li>
  265. <li><strong>第一次需要 Context</strong>:第一次获取单例时需要传入 Context,后续获取可以不带(但建议统一使用工厂类)</li>
  266. <li><strong>连接是必须的</strong>:蓝牙不是自动连接的,需要手动调用 <code>initializeAndConnect()</code> 连接设备</li>
  267. </ol>
  268. <h3 id="两种连接方式的区别">两种连接方式的区别 </h3>
  269. <h4 id="方式一一键连接推荐最简单">方式一:一键连接(推荐,最简单) </h4>
  270. <p>使用 <code>initializeAndConnect()</code> - <strong>自动完成所有步骤</strong>:</p>
  271. <ul>
  272. <li>✅ 自动扫描设备</li>
  273. <li>✅ 自动找到设备后连接</li>
  274. <li>✅ 自动完成握手和配对</li>
  275. <li>✅ 一个方法搞定,无需手动控制</li>
  276. </ul>
  277. <p><strong>适用场景</strong>:大多数情况,直接连接即可</p>
  278. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token comment">// 一键完成:扫描 + 连接 + 握手</span>
  279. bleService<span class="token punctuation">.</span><span class="token function">initializeAndConnect</span><span class="token punctuation">(</span>
  280. userId <span class="token operator">=</span> userId<span class="token punctuation">,</span>
  281. userType <span class="token operator">=</span> userType<span class="token punctuation">,</span>
  282. onConnected <span class="token operator">=</span> <span class="token punctuation">{</span>
  283. <span class="token comment">// 连接成功,可以直接使用指令了</span>
  284. <span class="token punctuation">}</span><span class="token punctuation">,</span>
  285. onError <span class="token operator">=</span> <span class="token punctuation">{</span> error <span class="token operator">-&gt;</span>
  286. <span class="token comment">// 连接失败</span>
  287. <span class="token punctuation">}</span>
  288. <span class="token punctuation">)</span>
  289. </code></pre><h4 id="方式二手动控制需要选择设备时使用">方式二:手动控制(需要选择设备时使用) </h4>
  290. <p>使用 <code>startScan()</code> + <code>connect()</code> - <strong>手动控制每个步骤</strong>:</p>
  291. <ul>
  292. <li>1️⃣ 手动开始扫描</li>
  293. <li>2️⃣ 手动选择要连接的设备(可以显示设备列表让用户选择)</li>
  294. <li>3️⃣ 手动停止扫描</li>
  295. <li>4️⃣ 手动连接选中的设备</li>
  296. </ul>
  297. <p><strong>适用场景</strong>:需要显示设备列表让用户选择,或者需要控制扫描过程</p>
  298. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token comment">// 1. 开始扫描</span>
  299. bleService<span class="token punctuation">.</span><span class="token function">startScan</span> <span class="token punctuation">{</span> device<span class="token operator">:</span> BLEDevice <span class="token operator">-&gt;</span>
  300. <span class="token comment">// 找到设备,可以显示在列表中让用户选择</span>
  301. <span class="token function">println</span><span class="token punctuation">(</span><span class="token string-literal singleline"><span class="token string">"找到设备: </span><span class="token interpolation"><span class="token interpolation-punctuation punctuation">${</span><span class="token expression">device<span class="token punctuation">.</span>name</span><span class="token interpolation-punctuation punctuation">}</span></span><span class="token string">, 地址: </span><span class="token interpolation"><span class="token interpolation-punctuation punctuation">${</span><span class="token expression">device<span class="token punctuation">.</span>address</span><span class="token interpolation-punctuation punctuation">}</span></span><span class="token string">"</span></span><span class="token punctuation">)</span>
  302. <span class="token comment">// 可以添加到设备列表,让用户选择</span>
  303. <span class="token punctuation">}</span>
  304. <span class="token comment">// 2. 停止扫描(用户选择设备后)</span>
  305. bleService<span class="token punctuation">.</span><span class="token function">stopScan</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
  306. <span class="token comment">// 3. 连接用户选择的设备</span>
  307. <span class="token keyword keyword-val">val</span> userId <span class="token operator">=</span> <span class="token function">getUserId</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token comment">// 16字节</span>
  308. <span class="token keyword keyword-val">val</span> userType <span class="token operator">=</span> BLEConstants<span class="token punctuation">.</span>USER_TYPE_OWNER
  309. <span class="token keyword keyword-val">val</span> selectedDevice <span class="token operator">=</span> <span class="token function">BLEDevice</span><span class="token punctuation">(</span>name <span class="token operator">=</span> <span class="token string-literal singleline"><span class="token string">"用户选择的设备"</span></span><span class="token punctuation">,</span> address <span class="token operator">=</span> <span class="token string-literal singleline"><span class="token string">"MAC地址"</span></span><span class="token punctuation">,</span> rssi <span class="token operator">=</span> <span class="token number">0</span><span class="token punctuation">)</span>
  310. bleService<span class="token punctuation">.</span><span class="token function">connect</span><span class="token punctuation">(</span>selectedDevice<span class="token punctuation">,</span> userId<span class="token punctuation">,</span> userType<span class="token punctuation">)</span> <span class="token punctuation">{</span> response <span class="token operator">-&gt;</span>
  311. <span class="token keyword keyword-if">if</span> <span class="token punctuation">(</span>response<span class="token punctuation">.</span>success<span class="token punctuation">)</span> <span class="token punctuation">{</span>
  312. <span class="token comment">// 连接成功,已自动完成握手</span>
  313. <span class="token punctuation">}</span> <span class="token keyword keyword-else">else</span> <span class="token punctuation">{</span>
  314. <span class="token comment">// 连接失败</span>
  315. <span class="token punctuation">}</span>
  316. <span class="token punctuation">}</span>
  317. </code></pre><p><strong>总结</strong>:</p>
  318. <ul>
  319. <li><strong>一键连接</strong>:最简单,自动完成所有步骤,推荐使用</li>
  320. <li><strong>手动控制</strong>:需要用户选择设备时使用,可以显示设备列表</li>
  321. </ul>
  322. <h2 id="api-参考">API 参考 </h2>
  323. <h3 id="基础连接功能">基础连接功能 </h3>
  324. <h4 id="startscancallback-bledevice---unit"><code>startScan(callback: (BLEDevice) -&gt; Unit)</code> </h4>
  325. <p>开始扫描 BLE 设备</p>
  326. <h4 id="stopscan"><code>stopScan()</code> </h4>
  327. <p>停止扫描设备</p>
  328. <h4 id="connectdevice-bledevice-userid-bytearray-usertype-byte-callback-bleresponse---unit"><code>connect(device: BLEDevice, userId: ByteArray, userType: Byte, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  329. <p>连接设备(自动完成握手和配对)</p>
  330. <h4 id="disconnect"><code>disconnect()</code> </h4>
  331. <p>断开连接</p>
  332. <h4 id="isconnected-boolean"><code>isConnected(): Boolean</code> </h4>
  333. <p>检查是否已连接</p>
  334. <h3 id="高级连接功能推荐使用">高级连接功能(推荐使用) </h3>
  335. <h4 id="initializeandconnect"><code>initializeAndConnect()</code> </h4>
  336. <p>一键完成扫描+连接+握手(最简单的方式)</p>
  337. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token keyword keyword-fun">fun</span> <span class="token function">initializeAndConnect</span><span class="token punctuation">(</span>
  338. userId<span class="token operator">:</span> ByteArray<span class="token punctuation">,</span>
  339. userType<span class="token operator">:</span> Byte<span class="token punctuation">,</span>
  340. onConnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">,</span>
  341. onDisconnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">)</span><span class="token operator">?</span> <span class="token operator">=</span> <span class="token keyword keyword-null">null</span><span class="token punctuation">,</span>
  342. onDataReceived<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">(</span>ByteArray<span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">)</span><span class="token operator">?</span> <span class="token operator">=</span> <span class="token keyword keyword-null">null</span><span class="token punctuation">,</span>
  343. onError<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">(</span>String<span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">)</span><span class="token operator">?</span> <span class="token operator">=</span> <span class="token keyword keyword-null">null</span><span class="token punctuation">,</span>
  344. scanTimeoutMs<span class="token operator">:</span> Long <span class="token operator">=</span> <span class="token number">10000</span>
  345. <span class="token punctuation">)</span>
  346. </code></pre><p><strong>参数说明</strong>:</p>
  347. <ul>
  348. <li><code>userId</code>: 用户ID(16字节),用于握手认证</li>
  349. <li><code>userType</code>: 用户类型(0x01=最高权限, 0x02=车主, 0x03=分享用户)</li>
  350. <li><code>onConnected</code>: 连接成功回调(已自动完成握手和配对)</li>
  351. <li><code>onDisconnected</code>: 断开连接回调(可选)</li>
  352. <li><code>onDataReceived</code>: 数据接收回调(可选)</li>
  353. <li><code>onError</code>: 错误回调(可选)</li>
  354. <li><code>scanTimeoutMs</code>: 扫描超时时间(毫秒),默认10秒</li>
  355. </ul>
  356. <h4 id="scanandconnect"><code>scanAndConnect()</code> </h4>
  357. <p>扫描并连接设备(一键完成扫描+连接)</p>
  358. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token keyword keyword-fun">fun</span> <span class="token function">scanAndConnect</span><span class="token punctuation">(</span>
  359. userId<span class="token operator">:</span> ByteArray<span class="token punctuation">,</span>
  360. userType<span class="token operator">:</span> Byte<span class="token punctuation">,</span>
  361. scanTimeoutMs<span class="token operator">:</span> Long <span class="token operator">=</span> <span class="token number">10000</span><span class="token punctuation">,</span>
  362. onScanning<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">)</span><span class="token operator">?</span> <span class="token operator">=</span> <span class="token keyword keyword-null">null</span><span class="token punctuation">,</span>
  363. onDeviceFound<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">(</span>BLEDevice<span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">)</span><span class="token operator">?</span> <span class="token operator">=</span> <span class="token keyword keyword-null">null</span><span class="token punctuation">,</span>
  364. onConnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">,</span>
  365. onError<span class="token operator">:</span> <span class="token punctuation">(</span>String<span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit
  366. <span class="token punctuation">)</span>
  367. </code></pre><h4 id="setconnectionlistener"><code>setConnectionListener()</code> </h4>
  368. <p>设置连接状态监听器(连接/断开时自动回调)</p>
  369. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token keyword keyword-fun">fun</span> <span class="token function">setConnectionListener</span><span class="token punctuation">(</span>
  370. onConnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">,</span>
  371. onDisconnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit<span class="token punctuation">,</span>
  372. onError<span class="token operator">:</span> <span class="token punctuation">(</span>String<span class="token punctuation">)</span> <span class="token operator">-&gt;</span> Unit
  373. <span class="token punctuation">)</span>
  374. </code></pre><h3 id="便捷方法">便捷方法 </h3>
  375. <h4 id="setdefenseenabled-boolean-callback-bleresponse---unit"><code>setDefense(enabled: Boolean, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  376. <p>设防/撤防</p>
  377. <h4 id="poweroncallback-bleresponse---unit"><code>powerOn(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  378. <p>上电</p>
  379. <h4 id="poweroffcallback-bleresponse---unit"><code>powerOff(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  380. <p>下电</p>
  381. <h4 id="findcarcallback-bleresponse---unit"><code>findCar(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  382. <p>寻车</p>
  383. <h4 id="unlockseatcallback-bleresponse---unit"><code>unlockSeat(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  384. <p>打开座桶锁</p>
  385. <h4 id="unlockhandlebarcallback-bleresponse---unit"><code>unlockHandlebar(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  386. <p>打开龙头锁</p>
  387. <h4 id="lockhandlebarcallback-bleresponse---unit"><code>lockHandlebar(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  388. <p>关闭龙头锁</p>
  389. <h4 id="unlocktrunkcallback-bleresponse---unit"><code>unlockTrunk(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  390. <p>打开尾箱锁</p>
  391. <h4 id="locktrunkcallback-bleresponse---unit"><code>lockTrunk(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  392. <p>关闭尾箱锁</p>
  393. <h4 id="setsensorunlockenabled-boolean-callback-bleresponse---unit"><code>setSensorUnlock(enabled: Boolean, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  394. <p>设置感应解锁开关</p>
  395. <h4 id="setambientlightenabled-boolean-callback-bleresponse---unit"><code>setAmbientLight(enabled: Boolean, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  396. <p>设置氛围灯开关</p>
  397. <h4 id="setambientlightcolorr-int-g-int-b-int-callback-bleresponse---unit"><code>setAmbientLightColor(r: Int, g: Int, b: Int, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  398. <p>设置氛围灯颜色</p>
  399. <h4 id="setautopowerofftimetimeseconds-int-callback-bleresponse---unit"><code>setAutoPowerOffTime(timeSeconds: Int, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  400. <p>设置自动下电时间</p>
  401. <h4 id="setspeakervolumevolume-int-callback-bleresponse---unit"><code>setSpeakerVolume(volume: Int, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  402. <p>设置蓝牙音箱音量</p>
  403. <h3 id="通用方法">通用方法 </h3>
  404. <h4 id="sendappcontrolcommandinstructiontype-byte-data-bytearray-callback-bleresponse---unit"><code>sendAppControlCommand(instructionType: Byte, data: ByteArray, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  405. <p>发送应用控制指令</p>
  406. <p><strong>指令类型</strong>(<code>AppControlInstruction</code> 对象):</p>
  407. <ul>
  408. <li><code>AppControlInstruction.SET_DEFENSE</code> - 设撤防</li>
  409. <li><code>AppControlInstruction.POWER_ON_OFF</code> - 上下电</li>
  410. <li><code>AppControlInstruction.FIND_CAR</code> - 寻车</li>
  411. <li><code>AppControlInstruction.SEAT_LOCK</code> - 座桶锁</li>
  412. <li><code>AppControlInstruction.HANDLEBAR_LOCK</code> - 龙头锁</li>
  413. <li><code>AppControlInstruction.TRUNK_LOCK</code> - 尾箱锁</li>
  414. </ul>
  415. <h4 id="sendsystemcontrolcommandinstructiontype-byte-data-bytearray-callback-bleresponse---unit"><code>sendSystemControlCommand(instructionType: Byte, data: ByteArray, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  416. <p>发送系统控制指令</p>
  417. <p><strong>指令类型</strong>(<code>SystemControlInstruction</code> 对象):</p>
  418. <ul>
  419. <li><code>SystemControlInstruction.SENSOR_UNLOCK</code> - 感应解锁开关</li>
  420. <li><code>SystemControlInstruction.AMBIENT_LIGHT</code> - 氛围灯开关</li>
  421. <li><code>SystemControlInstruction.AMBIENT_LIGHT_COLOR</code> - 氛围灯颜色</li>
  422. <li><code>SystemControlInstruction.AUTO_POWER_OFF_TIME</code> - 自动下电时间</li>
  423. <li><code>SystemControlInstruction.SPEAKER_VOLUME</code> - 蓝牙音箱音量</li>
  424. <li>更多指令类型请查看 <code>SystemControlInstruction</code> 对象</li>
  425. </ul>
  426. <h4 id="queryvehicleinfocallback-bleresponse---unit"><code>queryVehicleInfo(callback: (BLEResponse) -&gt; Unit)</code> </h4>
  427. <p>查询车辆信息</p>
  428. <h4 id="sendcommandcommand-blecommand-callback-bleresponse---unit"><code>sendCommand(command: BLECommand, callback: (BLEResponse) -&gt; Unit)</code> </h4>
  429. <p>发送自定义指令</p>
  430. <h4 id="setdatareceivedlistenerlistener-datacallback"><code>setDataReceivedListener(listener: DataCallback)</code> </h4>
  431. <p>设置数据接收监听器</p>
  432. <h2 id="常量定义">常量定义 </h2>
  433. <h3 id="用户类型">用户类型 </h3>
  434. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code>BLEConstants<span class="token punctuation">.</span>USER_TYPE_HIGHEST <span class="token comment">// 0x01 - 最高权限</span>
  435. BLEConstants<span class="token punctuation">.</span>USER_TYPE_OWNER <span class="token comment">// 0x02 - 车主</span>
  436. BLEConstants<span class="token punctuation">.</span>USER_TYPE_SHARED <span class="token comment">// 0x03 - 分享用户</span>
  437. </code></pre><h3 id="功能码">功能码 </h3>
  438. <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code>BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_HANDSHAKE <span class="token comment">// 0x01 - 握手</span>
  439. BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_APP_CONTROL <span class="token comment">// 0x02 - 应用控制</span>
  440. BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_SYSTEM_CONTROL <span class="token comment">// 0x03 - 系统控制</span>
  441. BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_SYSTEM_QUERY <span class="token comment">// 0x04 - 系统查询</span>
  442. BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_SYSTEM_SETTING <span class="token comment">// 0x05 - 系统设置</span>
  443. </code></pre><h2 id="注意事项">注意事项 </h2>
  444. <ol>
  445. <li>
  446. <p><strong>单例模式</strong>:<code>BLEServiceImpl</code> 使用单例模式,全局共享同一个连接状态</p>
  447. </li>
  448. <li>
  449. <p><strong>自动握手</strong>:<code>connect()</code> 方法内部自动完成握手,无需单独调用握手方法</p>
  450. </li>
  451. <li>
  452. <p><strong>线程安全</strong>:BLE 回调可能在非主线程执行,需要在回调中使用 <code>activity?.runOnUiThread {}</code> 切换到主线程更新 UI</p>
  453. </li>
  454. <li>
  455. <p><strong>错误处理</strong>:所有 API 都通过 <code>BLEResponse</code> 返回结果,需要检查 <code>response.success</code> 判断是否成功</p>
  456. </li>
  457. <li>
  458. <p><strong>生命周期管理</strong>:建议在 <code>onDestroyView()</code> (Fragment) 或 <code>onDestroy()</code> (Activity) 中调用 <code>disconnect()</code> 断开连接</p>
  459. </li>
  460. <li>
  461. <p><strong>获取服务实例</strong>:推荐使用 <code>BLEServiceFactory.create(context)</code> 获取服务实例(工厂方法内部使用单例模式)</p>
  462. </li>
  463. <li>
  464. <p><strong>Fragment 中使用</strong>:在 Fragment 中使用时,使用 <code>requireContext()</code> 获取 Context,回调中使用 <code>activity?.runOnUiThread {}</code> 更新 UI</p>
  465. </li>
  466. </ol>
  467. </div>
  468. </body></html>