| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501 |
- <!DOCTYPE html><html><head>
- <title>集成说明</title>
- <meta charset="utf-8">
- <meta name="viewport" content="width=device-width, initial-scale=1.0">
-
- <link rel="stylesheet" href="file:////Users/Zhuanz/.cursor/extensions/shd101wyy.markdown-preview-enhanced-0.8.20-universal/crossnote/dependencies/katex/katex.min.css">
-
-
-
-
-
- <style>
- 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}
- /* Please visit the URL below for more information: */
- /* https://shd101wyy.github.io/markdown-preview-enhanced/#/customize-css */
- </style>
- <!-- The content below will be included at the end of the <head> element. --><script type="text/javascript">
- document.addEventListener("DOMContentLoaded", function () {
- // your code here
- });
- </script></head><body for="html-export">
-
-
- <div class="crossnote markdown-preview ">
-
- <h1 id="蓝牙ble模块集成说明">蓝牙BLE模块集成说明 </h1>
- <h2 id="目录结构">目录结构 </h2>
- <pre data-role="codeBlock" data-info="" class="language-text"><code>capability-ble/src/
- └── main/
- ├── AndroidManifest.xml # Android 清单文件(权限声明)
- └── java/
- └── com/narutohuo/xindazhou/ble/
- ├── api/ # API 接口层
- │ └── BLEService.kt # 蓝牙服务接口定义
- ├── factory/ # 工厂类层
- │ └── BLEServiceFactory.kt # 服务工厂(统一入口)
- ├── impl/ # 实现层
- │ └── BLEServiceImpl.kt # 蓝牙服务具体实现(单例模式)
- ├── callback/ # 回调接口层
- │ └── BLECallback.kt # 蓝牙回调接口(包含 ConnectCallback、HandshakeCallback、CommandCallback、DataCallback)
- ├── config/ # 配置层
- │ └── BLEConstants.kt # 蓝牙常量定义(用户类型、功能码、UUID等)
- ├── model/ # 数据模型层
- │ ├── BLEDevice.kt # BLE设备模型
- │ ├── BLECommand.kt # BLE指令模型(包含 AppControlInstruction、SystemControlInstruction)
- │ ├── BLEResponse.kt # BLE响应模型
- │ └── Command.kt # 指令协议数据类
- └── util/ # 工具类层
- ├── BleScanner.kt # BLE扫描工具
- ├── BleConnector.kt # BLE连接工具
- ├── BLECrypto.kt # 加密解密工具(AES)
- ├── BLEPacketBuilder.kt # 协议包构建工具
- ├── BLEPacketParser.kt # 协议包解析工具
- ├── BLEPacketSplitter.kt # 协议包分包工具
- └── BlePacketSender.kt # 协议包发送工具
- </code></pre><h2 id="功能说明">功能说明 </h2>
- <h3 id="1-api-层---接口定义">1. api/ 层 - 接口定义 </h3>
- <ul>
- <li><strong>职责</strong>:定义蓝牙服务的公共接口,业务层只依赖接口</li>
- <li><strong>内容</strong>:<code>BLEService</code> 接口
- <ul>
- <li><strong>基础连接功能</strong>:
- <ul>
- <li><code>startScan()</code> - 开始扫描BLE设备</li>
- <li><code>stopScan()</code> - 停止扫描</li>
- <li><code>connect()</code> - 连接设备(自动完成握手)</li>
- <li><code>disconnect()</code> - 断开连接</li>
- <li><code>isConnected()</code> - 检查连接状态</li>
- </ul>
- </li>
- <li><strong>协议封装功能</strong>:
- <ul>
- <li><code>sendCommand()</code> - 发送自定义指令</li>
- <li><code>sendAppControlCommand()</code> - 发送应用控制指令</li>
- <li><code>sendSystemControlCommand()</code> - 发送系统控制指令</li>
- <li><code>queryVehicleInfo()</code> - 查询车辆信息</li>
- </ul>
- </li>
- <li><strong>便捷方法</strong>:
- <ul>
- <li><code>setDefense()</code>, <code>powerOn()</code>, <code>powerOff()</code>, <code>findCar()</code></li>
- <li><code>unlockSeat()</code>, <code>unlockHandlebar()</code>, <code>lockHandlebar()</code></li>
- <li><code>unlockTrunk()</code>, <code>lockTrunk()</code></li>
- <li><code>setSensorUnlock()</code>, <code>setAmbientLight()</code>, <code>setAmbientLightColor()</code></li>
- <li><code>setAutoPowerOffTime()</code>, <code>setSpeakerVolume()</code></li>
- </ul>
- </li>
- <li><strong>数据接收监听</strong>:
- <ul>
- <li><code>setDataReceivedListener()</code> - 设置数据接收监听器</li>
- </ul>
- </li>
- </ul>
- </li>
- </ul>
- <h3 id="2-factory-层---工厂类">2. factory/ 层 - 工厂类 </h3>
- <ul>
- <li><strong>职责</strong>:提供统一的服务实例获取方式</li>
- <li><strong>内容</strong>:<code>BLEServiceFactory</code> 对象(单例)</li>
- <li><strong>作用</strong>:
- <ul>
- <li>隐藏实现细节,统一入口</li>
- <li>提供 <code>create(context)</code> 方法获取服务实例</li>
- </ul>
- </li>
- </ul>
- <h3 id="3-impl-层---具体实现">3. impl/ 层 - 具体实现 </h3>
- <ul>
- <li><strong>职责</strong>:实现 <code>BLEService</code> 接口的具体逻辑</li>
- <li><strong>内容</strong>:<code>BLEServiceImpl</code> 单例实现类</li>
- <li><strong>功能</strong>:
- <ul>
- <li>封装 Android BLE SDK 调用</li>
- <li>封装 GATT 连接管理</li>
- <li>封装握手认证(连接时自动完成)</li>
- <li>封装数据分包和组包</li>
- <li>封装 AES 加密/解密</li>
- <li>封装协议包构建和解析</li>
- </ul>
- </li>
- </ul>
- <h3 id="4-model-层---数据模型">4. model/ 层 - 数据模型 </h3>
- <ul>
- <li><strong>职责</strong>:定义蓝牙相关的数据模型</li>
- <li><strong>内容</strong>:
- <ul>
- <li><code>BLEDevice</code> - BLE设备模型</li>
- <li><code>BLECommand</code> - BLE指令模型</li>
- <li><code>BLEResponse</code> - BLE响应模型</li>
- <li><code>Command</code> - 指令协议数据类</li>
- <li><code>AppControlInstruction</code> - 应用控制指令常量(对象)</li>
- <li><code>SystemControlInstruction</code> - 系统控制指令常量(对象)</li>
- </ul>
- </li>
- </ul>
- <h3 id="5-callback-层---回调接口">5. callback/ 层 - 回调接口 </h3>
- <ul>
- <li><strong>职责</strong>:定义蓝牙相关的回调接口</li>
- <li><strong>内容</strong>:
- <ul>
- <li><code>BLECallback.kt</code> 文件中包含:
- <ul>
- <li><code>ConnectCallback</code> - 连接回调接口</li>
- <li><code>HandshakeCallback</code> - 握手指令回调接口</li>
- <li><code>CommandCallback</code> - 指令发送回调接口</li>
- <li><code>DataCallback</code> - 数据接收回调接口</li>
- </ul>
- </li>
- </ul>
- </li>
- </ul>
- <h3 id="6-config-层---配置">6. config/ 层 - 配置 </h3>
- <ul>
- <li><strong>职责</strong>:定义蓝牙相关的常量配置</li>
- <li><strong>内容</strong>:
- <ul>
- <li><code>BLEConstants</code> - 蓝牙常量定义(用户类型、功能码、UUID等)</li>
- </ul>
- </li>
- </ul>
- <h3 id="7-util-层---工具类">7. util/ 层 - 工具类 </h3>
- <ul>
- <li><strong>职责</strong>:提供蓝牙相关的工具方法</li>
- <li><strong>内容</strong>:
- <ul>
- <li><code>BleScanner</code> - BLE扫描工具</li>
- <li><code>BleConnector</code> - BLE连接工具</li>
- <li><code>BLECrypto</code> - 加密解密工具</li>
- <li><code>BLEPacketBuilder</code> - 协议包构建工具</li>
- <li><code>BLEPacketParser</code> - 协议包解析工具</li>
- <li><code>BLEPacketSplitter</code> - 协议包分包工具</li>
- <li><code>BlePacketSender</code> - 协议包发送工具</li>
- </ul>
- </li>
- </ul>
- <h2 id="架构设计">架构设计 </h2>
- <ul>
- <li><strong>接口隔离</strong>:业务层只依赖接口,不依赖实现</li>
- <li><strong>工厂模式</strong>:统一入口,隐藏创建逻辑</li>
- <li><strong>单例模式</strong>:<code>BLEServiceImpl</code> 使用单例模式,全局共享同一个连接状态</li>
- <li><strong>完全封装</strong>:所有底层细节(连接、握手、分包、加密等)都在能力层内部完成</li>
- </ul>
- <h2 id="使用方式">使用方式 </h2>
- <h3 id="完整示例单例模式">完整示例(单例模式) </h3>
- <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
- <span class="token keyword keyword-import">import</span> android<span class="token punctuation">.</span>widget<span class="token punctuation">.</span>Toast
- <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
- <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
- <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
- <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>
-
- <span class="token comment">// 1. 获取单例实例(全局只有一个,在任何地方获取的都是同一个)</span>
- <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>
-
- <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>
- <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>
-
- <span class="token comment">// 2. 连接设备(一键完成扫描+连接+握手)</span>
- <span class="token function">connectDevice</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
- <span class="token punctuation">}</span>
-
- <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>
- <span class="token comment">// 获取用户信息(实际项目中从用户认证信息中获取)</span>
- <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>
- <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>
-
- bleService<span class="token punctuation">.</span><span class="token function">initializeAndConnect</span><span class="token punctuation">(</span>
- userId <span class="token operator">=</span> userId<span class="token punctuation">,</span>
- userType <span class="token operator">=</span> userType<span class="token punctuation">,</span>
- onConnected <span class="token operator">=</span> <span class="token punctuation">{</span>
- activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
- <span class="token comment">// 连接成功,可以直接使用指令了</span>
- 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>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span><span class="token punctuation">,</span>
- onError <span class="token operator">=</span> <span class="token punctuation">{</span> error <span class="token operator">-></span>
- activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
- 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>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">)</span>
- <span class="token punctuation">}</span>
-
- <span class="token comment">// 3. 发送指令(连接成功后调用)</span>
- <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>
- <span class="token comment">// 设防</span>
- 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">-></span>
- activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
- <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>
- 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>
- <span class="token punctuation">}</span> <span class="token keyword keyword-else">else</span> <span class="token punctuation">{</span>
- 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>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
-
- <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>
- <span class="token comment">// 上电</span>
- bleService<span class="token punctuation">.</span><span class="token function">powerOn</span> <span class="token punctuation">{</span> response <span class="token operator">-></span>
- activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
- <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>
- 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>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
-
- <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>
- <span class="token comment">// 寻车</span>
- bleService<span class="token punctuation">.</span><span class="token function">findCar</span> <span class="token punctuation">{</span> response <span class="token operator">-></span>
- activity<span class="token operator">?</span><span class="token punctuation">.</span><span class="token function">runOnUiThread</span> <span class="token punctuation">{</span>
- <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>
- 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>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
-
- <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>
- <span class="token comment">// 实际项目中从用户认证信息中获取</span>
- <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>
- <span class="token punctuation">}</span>
-
- <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>
- <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>
- <span class="token comment">// 4. 断开连接(可选,单例会保持连接状态)</span>
- <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>
- bleService<span class="token punctuation">.</span><span class="token function">disconnect</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- </code></pre><h3 id="步骤说明">步骤说明 </h3>
- <ol>
- <li><strong>获取单例实例</strong>:使用 <code>BLEServiceFactory.create(context)</code> 获取服务实例(全局只有一个)</li>
- <li><strong>连接设备</strong>:调用 <code>initializeAndConnect()</code> 连接设备(自动完成扫描+连接+握手)</li>
- <li><strong>发送指令</strong>:连接成功后,调用各种指令方法(如 <code>setDefense()</code>、<code>powerOn()</code>、<code>findCar()</code> 等)</li>
- <li><strong>断开连接</strong>:在 <code>onDestroyView()</code> 中断开连接(可选)</li>
- </ol>
- <h3 id="重要说明">重要说明 </h3>
- <ol>
- <li><strong>单例模式</strong>:<code>BLEService</code> 是单例,全局只有一个实例,在任何地方获取的都是同一个</li>
- <li><strong>连接状态共享</strong>:所有使用该服务的地方共享同一个连接状态</li>
- <li><strong>第一次需要 Context</strong>:第一次获取单例时需要传入 Context,后续获取可以不带(但建议统一使用工厂类)</li>
- <li><strong>连接是必须的</strong>:蓝牙不是自动连接的,需要手动调用 <code>initializeAndConnect()</code> 连接设备</li>
- </ol>
- <h3 id="两种连接方式的区别">两种连接方式的区别 </h3>
- <h4 id="方式一一键连接推荐最简单">方式一:一键连接(推荐,最简单) </h4>
- <p>使用 <code>initializeAndConnect()</code> - <strong>自动完成所有步骤</strong>:</p>
- <ul>
- <li>✅ 自动扫描设备</li>
- <li>✅ 自动找到设备后连接</li>
- <li>✅ 自动完成握手和配对</li>
- <li>✅ 一个方法搞定,无需手动控制</li>
- </ul>
- <p><strong>适用场景</strong>:大多数情况,直接连接即可</p>
- <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token comment">// 一键完成:扫描 + 连接 + 握手</span>
- bleService<span class="token punctuation">.</span><span class="token function">initializeAndConnect</span><span class="token punctuation">(</span>
- userId <span class="token operator">=</span> userId<span class="token punctuation">,</span>
- userType <span class="token operator">=</span> userType<span class="token punctuation">,</span>
- onConnected <span class="token operator">=</span> <span class="token punctuation">{</span>
- <span class="token comment">// 连接成功,可以直接使用指令了</span>
- <span class="token punctuation">}</span><span class="token punctuation">,</span>
- onError <span class="token operator">=</span> <span class="token punctuation">{</span> error <span class="token operator">-></span>
- <span class="token comment">// 连接失败</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">)</span>
- </code></pre><h4 id="方式二手动控制需要选择设备时使用">方式二:手动控制(需要选择设备时使用) </h4>
- <p>使用 <code>startScan()</code> + <code>connect()</code> - <strong>手动控制每个步骤</strong>:</p>
- <ul>
- <li>1️⃣ 手动开始扫描</li>
- <li>2️⃣ 手动选择要连接的设备(可以显示设备列表让用户选择)</li>
- <li>3️⃣ 手动停止扫描</li>
- <li>4️⃣ 手动连接选中的设备</li>
- </ul>
- <p><strong>适用场景</strong>:需要显示设备列表让用户选择,或者需要控制扫描过程</p>
- <pre data-role="codeBlock" data-info="kotlin" class="language-kotlin kotlin"><code><span class="token comment">// 1. 开始扫描</span>
- 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">-></span>
- <span class="token comment">// 找到设备,可以显示在列表中让用户选择</span>
- <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>
- <span class="token comment">// 可以添加到设备列表,让用户选择</span>
- <span class="token punctuation">}</span>
- <span class="token comment">// 2. 停止扫描(用户选择设备后)</span>
- bleService<span class="token punctuation">.</span><span class="token function">stopScan</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
- <span class="token comment">// 3. 连接用户选择的设备</span>
- <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>
- <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 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>
- 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">-></span>
- <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>
- <span class="token comment">// 连接成功,已自动完成握手</span>
- <span class="token punctuation">}</span> <span class="token keyword keyword-else">else</span> <span class="token punctuation">{</span>
- <span class="token comment">// 连接失败</span>
- <span class="token punctuation">}</span>
- <span class="token punctuation">}</span>
- </code></pre><p><strong>总结</strong>:</p>
- <ul>
- <li><strong>一键连接</strong>:最简单,自动完成所有步骤,推荐使用</li>
- <li><strong>手动控制</strong>:需要用户选择设备时使用,可以显示设备列表</li>
- </ul>
- <h2 id="api-参考">API 参考 </h2>
- <h3 id="基础连接功能">基础连接功能 </h3>
- <h4 id="startscancallback-bledevice---unit"><code>startScan(callback: (BLEDevice) -> Unit)</code> </h4>
- <p>开始扫描 BLE 设备</p>
- <h4 id="stopscan"><code>stopScan()</code> </h4>
- <p>停止扫描设备</p>
- <h4 id="connectdevice-bledevice-userid-bytearray-usertype-byte-callback-bleresponse---unit"><code>connect(device: BLEDevice, userId: ByteArray, userType: Byte, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>连接设备(自动完成握手和配对)</p>
- <h4 id="disconnect"><code>disconnect()</code> </h4>
- <p>断开连接</p>
- <h4 id="isconnected-boolean"><code>isConnected(): Boolean</code> </h4>
- <p>检查是否已连接</p>
- <h3 id="高级连接功能推荐使用">高级连接功能(推荐使用) </h3>
- <h4 id="initializeandconnect"><code>initializeAndConnect()</code> </h4>
- <p>一键完成扫描+连接+握手(最简单的方式)</p>
- <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>
- userId<span class="token operator">:</span> ByteArray<span class="token punctuation">,</span>
- userType<span class="token operator">:</span> Byte<span class="token punctuation">,</span>
- onConnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-></span> Unit<span class="token punctuation">,</span>
- 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">-></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>
- 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">-></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>
- 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">-></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>
- scanTimeoutMs<span class="token operator">:</span> Long <span class="token operator">=</span> <span class="token number">10000</span>
- <span class="token punctuation">)</span>
- </code></pre><p><strong>参数说明</strong>:</p>
- <ul>
- <li><code>userId</code>: 用户ID(16字节),用于握手认证</li>
- <li><code>userType</code>: 用户类型(0x01=最高权限, 0x02=车主, 0x03=分享用户)</li>
- <li><code>onConnected</code>: 连接成功回调(已自动完成握手和配对)</li>
- <li><code>onDisconnected</code>: 断开连接回调(可选)</li>
- <li><code>onDataReceived</code>: 数据接收回调(可选)</li>
- <li><code>onError</code>: 错误回调(可选)</li>
- <li><code>scanTimeoutMs</code>: 扫描超时时间(毫秒),默认10秒</li>
- </ul>
- <h4 id="scanandconnect"><code>scanAndConnect()</code> </h4>
- <p>扫描并连接设备(一键完成扫描+连接)</p>
- <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>
- userId<span class="token operator">:</span> ByteArray<span class="token punctuation">,</span>
- userType<span class="token operator">:</span> Byte<span class="token punctuation">,</span>
- scanTimeoutMs<span class="token operator">:</span> Long <span class="token operator">=</span> <span class="token number">10000</span><span class="token punctuation">,</span>
- 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">-></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>
- 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">-></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>
- onConnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-></span> Unit<span class="token punctuation">,</span>
- onError<span class="token operator">:</span> <span class="token punctuation">(</span>String<span class="token punctuation">)</span> <span class="token operator">-></span> Unit
- <span class="token punctuation">)</span>
- </code></pre><h4 id="setconnectionlistener"><code>setConnectionListener()</code> </h4>
- <p>设置连接状态监听器(连接/断开时自动回调)</p>
- <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>
- onConnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-></span> Unit<span class="token punctuation">,</span>
- onDisconnected<span class="token operator">:</span> <span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-></span> Unit<span class="token punctuation">,</span>
- onError<span class="token operator">:</span> <span class="token punctuation">(</span>String<span class="token punctuation">)</span> <span class="token operator">-></span> Unit
- <span class="token punctuation">)</span>
- </code></pre><h3 id="便捷方法">便捷方法 </h3>
- <h4 id="setdefenseenabled-boolean-callback-bleresponse---unit"><code>setDefense(enabled: Boolean, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>设防/撤防</p>
- <h4 id="poweroncallback-bleresponse---unit"><code>powerOn(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>上电</p>
- <h4 id="poweroffcallback-bleresponse---unit"><code>powerOff(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>下电</p>
- <h4 id="findcarcallback-bleresponse---unit"><code>findCar(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>寻车</p>
- <h4 id="unlockseatcallback-bleresponse---unit"><code>unlockSeat(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>打开座桶锁</p>
- <h4 id="unlockhandlebarcallback-bleresponse---unit"><code>unlockHandlebar(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>打开龙头锁</p>
- <h4 id="lockhandlebarcallback-bleresponse---unit"><code>lockHandlebar(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>关闭龙头锁</p>
- <h4 id="unlocktrunkcallback-bleresponse---unit"><code>unlockTrunk(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>打开尾箱锁</p>
- <h4 id="locktrunkcallback-bleresponse---unit"><code>lockTrunk(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>关闭尾箱锁</p>
- <h4 id="setsensorunlockenabled-boolean-callback-bleresponse---unit"><code>setSensorUnlock(enabled: Boolean, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>设置感应解锁开关</p>
- <h4 id="setambientlightenabled-boolean-callback-bleresponse---unit"><code>setAmbientLight(enabled: Boolean, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>设置氛围灯开关</p>
- <h4 id="setambientlightcolorr-int-g-int-b-int-callback-bleresponse---unit"><code>setAmbientLightColor(r: Int, g: Int, b: Int, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>设置氛围灯颜色</p>
- <h4 id="setautopowerofftimetimeseconds-int-callback-bleresponse---unit"><code>setAutoPowerOffTime(timeSeconds: Int, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>设置自动下电时间</p>
- <h4 id="setspeakervolumevolume-int-callback-bleresponse---unit"><code>setSpeakerVolume(volume: Int, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>设置蓝牙音箱音量</p>
- <h3 id="通用方法">通用方法 </h3>
- <h4 id="sendappcontrolcommandinstructiontype-byte-data-bytearray-callback-bleresponse---unit"><code>sendAppControlCommand(instructionType: Byte, data: ByteArray, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>发送应用控制指令</p>
- <p><strong>指令类型</strong>(<code>AppControlInstruction</code> 对象):</p>
- <ul>
- <li><code>AppControlInstruction.SET_DEFENSE</code> - 设撤防</li>
- <li><code>AppControlInstruction.POWER_ON_OFF</code> - 上下电</li>
- <li><code>AppControlInstruction.FIND_CAR</code> - 寻车</li>
- <li><code>AppControlInstruction.SEAT_LOCK</code> - 座桶锁</li>
- <li><code>AppControlInstruction.HANDLEBAR_LOCK</code> - 龙头锁</li>
- <li><code>AppControlInstruction.TRUNK_LOCK</code> - 尾箱锁</li>
- </ul>
- <h4 id="sendsystemcontrolcommandinstructiontype-byte-data-bytearray-callback-bleresponse---unit"><code>sendSystemControlCommand(instructionType: Byte, data: ByteArray, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>发送系统控制指令</p>
- <p><strong>指令类型</strong>(<code>SystemControlInstruction</code> 对象):</p>
- <ul>
- <li><code>SystemControlInstruction.SENSOR_UNLOCK</code> - 感应解锁开关</li>
- <li><code>SystemControlInstruction.AMBIENT_LIGHT</code> - 氛围灯开关</li>
- <li><code>SystemControlInstruction.AMBIENT_LIGHT_COLOR</code> - 氛围灯颜色</li>
- <li><code>SystemControlInstruction.AUTO_POWER_OFF_TIME</code> - 自动下电时间</li>
- <li><code>SystemControlInstruction.SPEAKER_VOLUME</code> - 蓝牙音箱音量</li>
- <li>更多指令类型请查看 <code>SystemControlInstruction</code> 对象</li>
- </ul>
- <h4 id="queryvehicleinfocallback-bleresponse---unit"><code>queryVehicleInfo(callback: (BLEResponse) -> Unit)</code> </h4>
- <p>查询车辆信息</p>
- <h4 id="sendcommandcommand-blecommand-callback-bleresponse---unit"><code>sendCommand(command: BLECommand, callback: (BLEResponse) -> Unit)</code> </h4>
- <p>发送自定义指令</p>
- <h4 id="setdatareceivedlistenerlistener-datacallback"><code>setDataReceivedListener(listener: DataCallback)</code> </h4>
- <p>设置数据接收监听器</p>
- <h2 id="常量定义">常量定义 </h2>
- <h3 id="用户类型">用户类型 </h3>
- <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>
- BLEConstants<span class="token punctuation">.</span>USER_TYPE_OWNER <span class="token comment">// 0x02 - 车主</span>
- BLEConstants<span class="token punctuation">.</span>USER_TYPE_SHARED <span class="token comment">// 0x03 - 分享用户</span>
- </code></pre><h3 id="功能码">功能码 </h3>
- <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>
- BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_APP_CONTROL <span class="token comment">// 0x02 - 应用控制</span>
- BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_SYSTEM_CONTROL <span class="token comment">// 0x03 - 系统控制</span>
- BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_SYSTEM_QUERY <span class="token comment">// 0x04 - 系统查询</span>
- BLEConstants<span class="token punctuation">.</span>FUNCTION_CODE_SYSTEM_SETTING <span class="token comment">// 0x05 - 系统设置</span>
- </code></pre><h2 id="注意事项">注意事项 </h2>
- <ol>
- <li>
- <p><strong>单例模式</strong>:<code>BLEServiceImpl</code> 使用单例模式,全局共享同一个连接状态</p>
- </li>
- <li>
- <p><strong>自动握手</strong>:<code>connect()</code> 方法内部自动完成握手,无需单独调用握手方法</p>
- </li>
- <li>
- <p><strong>线程安全</strong>:BLE 回调可能在非主线程执行,需要在回调中使用 <code>activity?.runOnUiThread {}</code> 切换到主线程更新 UI</p>
- </li>
- <li>
- <p><strong>错误处理</strong>:所有 API 都通过 <code>BLEResponse</code> 返回结果,需要检查 <code>response.success</code> 判断是否成功</p>
- </li>
- <li>
- <p><strong>生命周期管理</strong>:建议在 <code>onDestroyView()</code> (Fragment) 或 <code>onDestroy()</code> (Activity) 中调用 <code>disconnect()</code> 断开连接</p>
- </li>
- <li>
- <p><strong>获取服务实例</strong>:推荐使用 <code>BLEServiceFactory.create(context)</code> 获取服务实例(工厂方法内部使用单例模式)</p>
- </li>
- <li>
- <p><strong>Fragment 中使用</strong>:在 Fragment 中使用时,使用 <code>requireContext()</code> 获取 Context,回调中使用 <code>activity?.runOnUiThread {}</code> 更新 UI</p>
- </li>
- </ol>
- </div>
-
-
-
-
-
-
-
-
-
- </body></html>
|