diff --git a/404.html b/404.html index c013c16..c8b9ffd 100644 --- a/404.html +++ b/404.html @@ -5,13 +5,13 @@ 找不到页面 | Valo Reader - +
跳到主要内容

找不到页面

我们找不到您要找的页面。

请联系原始链接来源网站的所有者,并告知他们链接已损坏。

- + \ No newline at end of file diff --git a/assets/js/23fe4fcc.5a18a3a9.js b/assets/js/23fe4fcc.5a18a3a9.js deleted file mode 100644 index 12b99e3..0000000 --- a/assets/js/23fe4fcc.5a18a3a9.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdf_doc_source=self.webpackChunkdf_doc_source||[]).push([[728],{2183:n=>{n.exports=JSON.parse('{"blogPosts":[{"id":"tech_detail_macos","metadata":{"permalink":"/valo-reader-doc/blog/tech_detail_macos","editUrl":"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/tech_detail_macos.md","source":"@site/blog/tech_detail_macos.md","title":"Valo Reader for macOS \u6280\u672f\u6982\u8981","description":"\u6458\u8981","date":"2023-10-19T05:59:23.000Z","formattedDate":"2023\u5e7410\u670819\u65e5","tags":[{"label":"technical","permalink":"/valo-reader-doc/blog/tags/technical"}],"readingTime":23.595,"hasTruncateMarker":false,"authors":[{"name":"Ce Wang","title":"Solo dev","url":"https://github.com/haloWang","imageURL":"https://github.com/haloWang.png","key":"halowang"}],"frontMatter":{"slug":"tech_detail_macos","title":"Valo Reader for macOS \u6280\u672f\u6982\u8981","authors":["halowang"],"tags":["technical"]},"nextItem":{"title":"\u5f00\u53d1\u76ee\u7684","permalink":"/valo-reader-doc/blog/motivation"}},"content":"## \u6458\u8981\\n\\n\u672c\u6587\u4e3b\u8981\u4ecb\u7ecd\u4e86 Valo Reader for macOS\uff08\u4e0b\u6587\u7b80\u79f0\u4e3a\u201c\u672c\u7a0b\u5e8f\u201d\uff09\u7684[\u9879\u76ee\u5e03\u5c40](#\u9879\u76ee\u5e03\u5c40)\uff0c[\u5de5\u7a0b\u67b6\u6784](#\u5de5\u7a0b\u67b6\u6784)\uff0c[\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757](#\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757)\u4e0e[\u6280\u672f\u7ec6\u8282](#\u6280\u672f\u7ec6\u8282)\u3002\u4f60\u8fd8\u53ef\u4ee5\u5728[\u672c\u6587\u6863](/docs/intro)\u7684\u5176\u4ed6\u9875\u9762\u4e86\u89e3\u672c\u7a0b\u5e8f\u7684[\u5f00\u53d1\u52a8\u673a](./motivation.md)\uff0c[\u4f7f\u7528\u65b9\u5f0f](/docs/usage)\uff0c[\u9690\u79c1\u4e0e\u5b89\u5168](/docs/privacy_security)\u7b49\u5185\u5bb9\\n\\n## \u4ec0\u4e48\u662f Valo Reader?\\n\\nValo Reader \u8fd9\u4e2a\u9879\u76ee\u662f\u6211\u4e2a\u4eba\u5728\u65e5\u5e38\u751f\u6d3b\u4e2d\u9010\u6e10\u840c\u751f\uff0c\u6f14\u5316\u548c\u5b9e\u65bd\u7684\u60f3\u6cd5\u3002\u662f\u6211\u4e3a\u4e86\u5728\u81ea\u5df1\u7535\u5b50\u8bbe\u5907\u8bbe\u5907\u4e0a\u80fd\u8f7b\u677e\u9605\u8bfb\u82f1\u6587\u5185\u5bb9\uff0c\u5e76\u6e10\u8fd1\u5f0f\u5730\u63d0\u9ad8\u81ea\u5df1\u7684\u82f1\u6587\u6c34\u5e73\uff0c\u800c\u81ea\u884c\u8bbe\u8ba1\u548c\u7814\u53d1\u7684\u4e00\u6b3e\u4ea7\u54c1\uff0c\u5176\u6838\u5fc3\u529f\u80fd\u5c55\u793a\u5982\u4e0b\uff1a\\n![intro](../static/img/intro.gif)\\n\\n\u5f53\u4f60\u5728\u4f7f\u7528\u672c\u7a0b\u5e8f\u65f6\uff0c\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u4f60\u611f\u5230\u751f\u758f\u7684\u82f1\u8bed\u5355\u8bcd\u4e0a\u3002\u6309\u4e0b\u6307\u5b9a\u7684\u6309\u952e\uff08\u9ed8\u8ba4\u4e3a Fn\uff09\uff0c\u672c\u7a0b\u5e8f\u5373\u4f1a\u5728\u54cd\u5e94\u7684\u4f4d\u7f6e\u5c55\u793a\u5bf9\u5e94\u7684\u91ca\u4e49\uff0c\u540c\u65f6\u4f7f\u7528\u8bed\u97f3\u5408\u6210\u6765\u53d1\u97f3\u3002\\n\\n\u76ee\u524d\uff0c\u4f60\u53ef\u4ee5\u901a\u8fc7 [Mac App Store](https://apps.apple.com/cn/app/valo-reader/id6448040931) \u6216[\u5176\u4ed6\u65b9\u5f0f](/docs/installation)\u6765\u83b7\u53d6\u672c\u7a0b\u5e8f\\n\\n## \u9879\u76ee\u5e03\u5c40\\n\\n\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u9879\u76ee\u7684\u4e3b\u4f53\u7ed3\u6784\uff1a\\n![project_layout](./../static/img/project_layout.png)\\n\\n\u672c\u6587\u7684\u4f4d\u7f6e\u5219\u5904\u4e8e\u4e0a\u56fe\u4e2d\u7684\u6d45\u9ec4\u8272\u90e8\u5206\uff0c\u7531 [docusaurus](https://docusaurus.io/) \u751f\u6210\\n\\n\u672c\u9879\u76ee\u7684\u60f3\u6cd5\u6700\u65e9\u5b9e\u73b0\u4e8e\u5728\u6d4f\u89c8\u5668\u4e2d\u8fd0\u884c\u7684 js \u811a\u672c\uff0c\u4f60\u53ef\u4ee5\u5728 github \u4e0a\u770b\u5230\u672c\u9879\u76ee\u5728\u6d4f\u89c8\u5668\u4e0a\u7684\u65e9\u671f\u5b9e\u73b0\u53ca\u5bf9\u5e94\u7684[\u529f\u80fd\u5c55\u793a](https://github.com/HaloWang/english_flow#%E5%8A%9F%E8%83%BD%E5%B1%95%E7%A4%BA)\uff0c\u5373\u56fe\u4e2d\u6d45\u7eff\u8272\u90e8\u5206\\n\\n\u800c\u672c\u6587\u4e3b\u8981\u8bf4\u660e\u4e86\u4e0a\u56fe\u4e2d macOS\uff08\u6d45\u84dd\u8272\u90e8\u5206\uff09\u7684\u8fd0\u884c\u65b9\u5f0f\u4e0e\u6280\u672f\u7ec6\u8282\\n\\n## \u5de5\u7a0b\u67b6\u6784\\n\\n\u672c\u7a0b\u5e8f\u7531 `flutter create XXX --platform macos` \u547d\u4ee4\u521b\u5efa\uff0c\u5176\u6e90\u4ee3\u7801\u4e3b\u8981\u5206\u4e3a\u4e24\u90e8\u5206\\n\\n### flutter \u4fa7\\n\\n\u8fd9\u90e8\u5206\u7531 dart \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206\\n\\n#### \u4e3b\u7a0b\u5e8f\\n\\n\u8be5\u90e8\u5206\u7684\u4e3b\u8981\u529f\u80fd\u6709\uff1a\\n\\n1. \u901a\u8fc7 method channel \u5b9e\u73b0 flutter \u4e0e native \u7684\u4ea4\u4e92\\n2. \u6e32\u67d3\u7528\u6237\u8bbe\u7f6e\u9762\u677f\uff08Dashboard\uff09\\n3. \u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\u4e0e\u8bca\u65ad\u68c0\u67e5\u4fe1\u606f\\n4. \u4f7f\u7528\u7b2c\u4e09\u65b9\u4f9d\u8d56\u63d0\u4f9b\u7684\u529f\u80fd\\n\\n#### flutter package: \u6587\u672c\u5757\u8868\u5f81\\n\\n\u8fd9\u90e8\u5206\u662f\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u4ee3\u7801\uff0c\u5728 macOS / iOS / Android \u4e09\u7aef\u5171\u4eab\u90e8\u5206\u903b\u8f91\u4e0e UI\uff0c\u5176\u4e3b\u8981\u529f\u80fd\u7531\u6709\uff1a\\n\\n1. \u89e3\u6790\u7528\u6237\u8bbe\u5907\u5c4f\u5e55\u4e0a\u8bc6\u522b\u5230\u7684\u6587\u672c\u5757\uff0c\u7ed3\u6784\u5316\u6587\u672c\u5757\uff0c\u5e76\u5c06\u5176\u7ef4\u62a4\u5230\u672c\u7a0b\u5e8f\u7684\u5185\u5b58\u4e2d\\n2. \u58f0\u660e\u5e76\u5b9e\u73b0\u7528\u6237\u89c6\u89c9\u7126\u70b9\u4e0e\u5df2\u77e5\u7684\u6587\u672c\u4fe1\u606f\u7684\u4ea4\u4e92\\n3. \u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\\n\\n#### flutter package: \u57fa\u7840\u5e93\\n\\n\u5728\u6211\u7684\u6240\u6709 flutter \u9879\u76ee\u4e2d\u5171\u4eab\u7684\u57fa\u7840\u4f9d\u8d56\\n\\n### native \u4fa7\\n\\n\u8fd9\u90e8\u5206\u7531 swift \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206\\n\\n#### \u4e3b\u7a0b\u5e8f\u90e8\u5206\\n\\n\u6301\u6709 flutter engine\uff0c\u901a\u8fc7 method channel \u6253\u901a flutter \u4e0e `Base`\u3001`OCR` \u4ee5\u53ca\u5e94\u7528\u7a0b\u5e8f\u672c\u8eab\u7684\u53cc\u5411\u8c03\u7528\\n\\n\u521b\u5efa\u9879\u76ee\u65f6\u9ed8\u8ba4\u7684 `FlutterViewController` \u5df2\u88ab\u5220\u9664\\n\\n#### CocoaPods dependency: Cocoa \u529f\u80fd\u5e93\\n\\n\u4e0e Cocoa framework \u4ea4\u4e92\uff0c\u5982\uff1a\u6743\u9650\u7533\u8bf7\uff0c\u952e\u76d8\u76d1\u542c\uff0c\u9f20\u6807\u76d1\u542c\uff0c\u622a\u53d6\u5c4f\u5e55\uff0c\u79fb\u52a8\u627f\u8f7d flutter \u7684 `NSWindow`\u4ee5\u53ca\u63d0\u4f9b\u57fa\u672c\u82e5\u5e72\u539f\u751f\u80fd\u529b\\n\\n\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u6211\u4e5f\u4f1a\u5728\u8be5\u4f9d\u8d56\u4e2d\u5b66\u4e60\uff0c\u5c1d\u8bd5\u548c\u5b9e\u73b0 Cocoa/AppKit \u72ec\u6709\u7684 API \u4e0e\u529f\u80fd\\n\\n#### CocoaPods dependency: OCR \u529f\u80fd\u5e93\\n\\n\u4e0e [Apple Vision framework - Text Recognizing](https://developer.apple.com/documentation/vision/recognizing_text_in_images) \u4ea4\u4e92\uff0c\u5e76\u5b9e\u73b0\u90e8\u5206\u903b\u8f91\uff0c\u5982\uff1a\u53d1\u8d77\u6587\u672c\u8bc6\u522b\u8bf7\u6c42\uff0c\u7ef4\u62a4\u6587\u672c\u8bc6\u522b\u54cd\u5e94\u7f13\u5b58\\n\\n\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u5728 macOS \u548c iOS \u9879\u76ee\u540c\u65f6\u4f9d\u8d56\u5e76\u5171\u4eab\u4ee3\u7801\\n\\n## \u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757\\n\\n\u5728\u8fd0\u884c\u65f6\uff0c\u672c\u7a0b\u5e8f\u4e3b\u8981\u53ef\u5206\u4e3a\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\u3001\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD) \u548c\u539f\u751f\u4fa7 (native) \u4e09\u4e2a\u6a21\u5757\\n\\n\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u7684\u4e3b\u8981\u6a21\u5757\u53ca\u5176\u901a\u8baf\uff1a\\n![modules_light](../static/img/modules_light.png#gh-light-mode-only)\\n![modules_dark](../static/img/modules_dark.png#gh-dark-mode-only)\\n\\n### \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\\n\\nHUD \u662f\u4e00\u4e2a\u7684 `NSWindow` \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a `FlutterViewController`\uff0c\u5e76\u7ed1\u5b9a\u4e86\u81ea\u5df1\u7684 `FlutterEngine` \u548c `FlutterMethodChannel`\\n\\nHUD \u7684\u4e3b\u8981\u4efb\u52a1\u5305\u542b\uff1a\\n\\n- \u901a\u8fc7 `FlutterMethodChannel` \u5411 native \u6d3e\u53d1 ocr \u8bf7\u6c42\\n- \u89e3\u6790 native \u5bf9\u5c4f\u5e55\u6307\u5b9a\u533a\u57df\u7684 ocr \u7ed3\u679c\uff0c\u5e76\u5728\u5185\u5b58\u4e2d\u7ef4\u62a4\u8be5\u7ed3\u679c\u4f9b\u540e\u7ee7\u7a0b\u5e8f\u903b\u8f91\u4f7f\u7528\\n- \u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u4ee5\u5728\u9700\u8981\u65f6\u8ba1\u7b97\u91ca\u4e49\u5c55\u793a\u7684\u4f4d\u7f6e\\n- \u76d1\u542c\u952e\u76d8\u6309\u952e\u72b6\u6001\uff0c\u5728\u53d8\u5316\u65f6\u6d3e\u53d1 ocr \u8bf7\u6c42\u5e76\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\\n- \u5b9e\u65f6\u540c\u6b65 Dashboard engine \u4f20\u9012\u8fc7\u6765\u7684\u7528\u6237\u8bbe\u7f6e\uff0c\u5e76\u4fee\u6539\u5c55\u793a\u903b\u8f91\\n\\n\u4e0b\u56fe\u5c55\u793a\u4e86\u8be5\u6a21\u5757\u6240\u7ef4\u62a4\u7684\u6570\u636e\u6620\u5c04\u5230\u5c4f\u5e55\u4e0a\u65f6\u7684\u53ef\u89c6\u5316\u6548\u679c\uff1a\\n![text_block_representation](../static/img/text_block_representation.png)\\n\\n#### \u5728 HUD \u4e2d\u7684\u72b6\u6001\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 [riverpod](https://riverpod.dev/) \u7ba1\u7406\u7edd\u5927\u90e8\u5206\u7684\u72b6\u6001\u3002\u5728\u6784\u7b51\u672c\u7a0b\u5e8f\u65f6\uff0c\u6211\u7684\u601d\u8003\u8fc7\u7a0b\u4e3b\u8981\u57fa\u4e8e\u72b6\u6001\u4e0e\u72b6\u6001\u53d8\u5316\uff0c\u4ece\u5c55\u793a\u91ca\u4e49\u952e\u70b9\u51fb\u5230\u6e32\u67d3\u91ca\u4e49\u9762\u677f\u52a8\u753b\\n\\n\u5728\u5927\u91cf\u4f7f\u7528 riverpod \u4e2d\u7684 `Provider` / `StateProvider` / `ProviderContainer.listen` \u540e\uff0c\u6211\u53ef\u4ee5\u6784\u5efa\u4e00\u4e2a\u9ad8\u5ea6\u5f02\u6b65\uff0c\u8c03\u7528\u987a\u5e8f\u4e0d\u654f\u611f\uff0c\u72b6\u6001\u53d8\u5316\u9a71\u52a8\u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u5728\u7f16\u7801\u8fc7\u7a0b\u4e2d\uff0c\u8fd9\u79cd\u65b9\u5f0f\u8ba9\u7a0b\u5e8f\u5458\u53ef\u4ee5\u4ece\u4e00\u5927\u4e32\u7684\u547d\u4ee4\u5f0f(Imperative)\u65b9\u6cd5\u8c03\u7528\u4e2d\u89e3\u653e\u51fa\u6765\uff0c\u4ec5\u4ec5\u9700\u8981\u5173\u6ce8\u548c\u786e\u4fdd\u6bcf\u4e2a\u6700\u5c0f\u903b\u8f91\u5355\u5143\u2014\u2014`Provider`\uff0c\u786e\u4fdd\u5176\u5b9e\u73b0\u662f\u6b63\u786e\u7684\u5373\u53ef\uff0c\u8fd9\u51cf\u5c11\u4e86\u6784\u5efa\u590d\u6742\u548c\u957f\u4e32\u903b\u8f91\u51fa\u9519\u7684\u53ef\u80fd\\n\\n\u4e0b\u9762\u7684\u4e24\u5f20\u56fe\u8868\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u7684\u4e3b\u8981\u529f\u80fd\u6d41\u7a0b\\n\\n\u5207\u6362\u5c55\u793a\u91ca\u4e49\u6d41\u7a0b\uff1a\\n\\n```mermaid\\ngraph LR\\n A1[\u76d1\u542c\u952e\u76d8\u72b6\u6001];\\n A[\u952e\u76d8\u72b6\u6001\u53d1\u751f\u6539\u53d8];\\n query_button_down{\u5c55\u793a\u91ca\u4e49\u952e\u6309\u4e0b?};\\n C[\u505c\u6b62\u5b9a\u65f6\u5668];\\n D[\u542f\u52a8\u5b9a\u65f6\u5668\u5e76\u91cd\u7f6e\u6587\u672c\u5757\u72b6\u6001];\\n mouse_state[\u9f20\u6807\u72b6\u6001];\\n screen_state[\u5c4f\u5e55\u72b6\u6001];\\n send_request_to_native[\\"\u5411 native \u53d1\u9001\u622a\u56fe\u4e0e OCR \u8bf7\u6c42\\"];\\n F[native OCR \u7ed3\u679c\u5df2\u83b7\u5f97];\\n G[\u89e3\u6790 OCR \u7ed3\u679c];\\n H1[\u66f4\u65b0\u6587\u672c\u5757\u72b6\u6001];\\n\\n mouse_state--\x3esend_request_to_native\\n screen_state--\x3esend_request_to_native\\n send_request_to_native-.->F;\\n F--\x3eG;\\n G--\x3eH1;\\n A1-.->A;\\n A--\x3equery_button_down;\\n query_button_down--\x3e|false|C;\\n query_button_down--\x3e|true|D;\\n D-.->send_request_to_native;\\n```\\n\\n\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u53d8\u66f4\u6d41\u7a0b\uff1a\\n\\n```mermaid\\ngraph LR;\\n H2[\u6587\u672c\u5757\u72b6\u6001\u53d8\u66f4];\\n most_wanted_state_logic[\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u76d1\u542c\u903b\u8f91];\\n most_wanted_state_changed[\u72b6\u6001\u6539\u53d8];\\n mouse_change0[\u76d1\u542c\u9f20\u6807\u72b6\u6001];\\n mouse_change[\u9f20\u6807\u72b6\u6001\u6539\u53d8];\\n sync_mouse_to_flutter[\u540c\u6b65\u9f20\u6807\u72b6\u6001\u81f3 flutter];\\n sync_current_screen[\u540c\u6b65\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u6570\u636e];\\n sync_current_screen_to_flutter[\u540c\u6b65\u5c4f\u5e55\u6570\u636e\u81f3 flutter];\\n hasTextBlock{\u72b6\u6001\u7684\u503c\u662f\u5426\u4e3a\u7a7a\\\\n\u5373\u9f20\u6807\u5f53\u524d\u4f4d\u7f6e\u6ca1\u6709\u8bc6\u522b\u5230\u6587\u672c};\\n queryDB[\u67e5\u8be2\u6570\u636e\u5e93];\\n dbHasData{\u6570\u636e\u5e93\u5305\u542b\u91ca\u4e49};\\n hideAllFlow[\u9690\u85cf\u6240\u6709\u91ca\u4e49\u9762\u677f];\\n asParam[\u4f5c\u4e3aOCR\u8bf7\u6c42\u7684\u53c2\u6570];\\n screen_changed{\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u53d1\u751f\u53d8\u52a8?}\\n\\n \u5207\u6362\u5c55\u793a\u91ca\u4e49-.->H2\\n mouse_change0-.->mouse_change--\x3esync_mouse_to_flutter;\\n mouse_change --\x3e sync_current_screen --\x3e sync_current_screen_to_flutter;\\n sync_current_screen_to_flutter --\x3e \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001\\n sync_current_screen --\x3e screen_changed --\x3e |true| \u6539\u53d8\u627f\u8f7dHUD\u7684NSPanel\u4f4d\u7f6e;\\n screen_changed --\x3e |false| return_0;\\n H2 --\x3e most_wanted_state_logic;\\n sync_mouse_to_flutter --\x3e most_wanted_state_logic;\\n sync_mouse_to_flutter -.-> asParam;\\n \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001 -.-> asParam;\\n most_wanted_state_logic --\x3e most_wanted_state_changed --\x3e hasTextBlock;\\n \u5c55\u793a\u91ca\u4e49\u952e\u72b6\u6001 --\x3e most_wanted_state_logic;\\n most_wanted_state_changed --\x3e hideAllFlow;\\n hasTextBlock --\x3e |false| return_1;\\n hasTextBlock --\x3e |true| queryDB;\\n queryDB --\x3e dbHasData;\\n dbHasData --\x3e |true| \u6e32\u67d3\u5c55\u793a\u91ca\u4e49\u9762\u677f\u52a8\u753b\\n dbHasData --\x3e |false| return_2;\\n```\\n\\n### \u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\\n\\nDashboard \u662f\u4e00\u4e2a `NSPanel` \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a `FlutterViewController`\uff0c\u5e76\u7ed1\u5b9a\u4e86\u4e0d\u540c\u4e8e HUD \u7684\u72ec\u7acb engine \u4e0e method channel\\n\\nDashboard \u7684\u4e3b\u8981\u4efb\u52a1\u662f\u4e3a\u7528\u6237\u63d0\u4f9b\u63a7\u5236\u672c\u7a0b\u5e8f\u7684 UI\uff0c\u5305\u62ec\u5feb\u6377\u952e\u8bbe\u7f6e\uff0c\u5f00\u673a\u81ea\u542f\uff0c\u5173\u4e8e\u672c\u7a0b\u5e8f\u7b49\u5e38\u89c1\u7528\u6237\u4ea4\u4e92\\n\\n\u540c\u65f6\uff0cDashboard \u4f1a\u7ef4\u62a4\u672c\u7a0b\u5e8f\u5728\u5185\u5b58\u548c\u786c\u76d8\u4e2d\u7684\u72b6\u6001\uff0c\u5e76\u5c06\u8fd9\u4e2a\u72b6\u6001\u901a\u8fc7 method channel \u540c\u6b65\u81f3\u4e0b\u9762\u5373\u5c06\u8981\u8bb2\u5230\u7684\u91ca\u4e49\u5c55\u793a\u9762\u677f\\n\\n### \u539f\u751f\u4fa7 (native)\\n\\n\u539f\u751f\u4fa7\u662f\u7531 `flutter create` \u547d\u4ee4\u521b\u5efa\u51fa\u6765\u7684\u4f20\u7edf Xcode \u5de5\u7a0b\\n\\n\u5728\u539f\u751f\u4fa7\uff0c\u6211\u4e3b\u8981\u5173\u6ce8\u7684\u95ee\u9898\u5982\u4e0b\uff1a\\n\\n- \u5904\u7406\u6765\u81ea `FlutterMethodChannel` \u7684\u8c03\u7528\uff0c\u5e76\u5728\u9700\u8981\u65f6\u901a\u8fc7 `FlutterResult` \u54cd\u5e94\u5bf9\u5e94\u7684 flutter engine\\n- \u7ef4\u62a4\u627f\u8f7d HUD \u548c Dashboard \u7684\u4e24\u4e2a `NSWindow`\uff0c\u5c24\u5176\u662f\u5f53\u7528\u6237\u7684\u8bbe\u5907\u540c\u65f6\u94fe\u63a5\u591a\u4e2a\u5c4f\u5e55\u65f6\uff0c\u6b63\u786e\u5730\u8bbe\u7f6e HUD \u7684\u4f4d\u7f6e\\n- \u5c01\u88c5\u5e76\u5411 flutter \u63d0\u4f9b\u80fd\u529b\uff0c\u5982\uff1a\\n - \u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u53d8\u5316\u548c\u952e\u76d8\u72b6\u6001\u53d8\u5316\uff0c\u5e76\u5c06\u72b6\u6001\u540c\u6b65\u81f3 flutter\\n - \u4e3a flutter \u63d0\u4f9b OCR \u548c\u8bed\u97f3\u5408\u6210\u80fd\u529b\\n - \u5728\u6267\u884c OCR \u65f6\u4f7f\u7528 swift \u5b9e\u73b0\u5fc5\u8981\u7684\u6027\u80fd\u4f18\u5316\\n\\n## \u6280\u672f\u7ec6\u8282\\n\\n\u5f97\u76ca\u4e8e Apple \u4e00\u8109\u76f8\u627f\u7684 API \u8bbe\u8ba1\u4ee5\u53ca\u540c\u6837\u7684\u7f16\u7a0b\u8bed\u8a00\uff0c\u719f\u7ec3\u4e8e iOS \u5e94\u7528\u5f00\u53d1(Cocoa Touch & UIKit)\u7684\u7a0b\u5e8f\u5458\u5728\u9762\u5bf9\u4e0e macOS \u4ea4\u4e92\u7684 Cocoa \u548c AppKit \u6846\u67b6\u65f6\uff0c\u53ef\u4ee5\u590d\u7528\u5f88\u591a\u601d\u60f3\u4e0e\u903b\u8f91\\n\\n\u4f46\u76f8\u6bd4\u4e8e iOS\uff0cmacOS \u7ed9\u4e86\u7528\u6237\u66f4\u5927\u7684\u821e\u53f0\uff0c\u4e5f\u8ba9\u5f00\u53d1\u8005\u9762\u5bf9\u4e86\u66f4\u591a\u7684\u6311\u6218: \u9f20\u6807\uff0c\u952e\u76d8\uff0c\u7a97\u53e3\u548c\u5c4f\u5e55\\n\\n\u5728\u5f00\u53d1\u672c\u7a0b\u5e8f\u7684\u8fc7\u7a0b\u4e2d\uff0c\u6211\u9047\u5230\u4e86\u8bf8\u591a\u7684\u56f0\u96be\uff0c\u4e5f\u5728\u8305\u585e\u987f\u5f00\u65f6\u6536\u83b7\u4e86\u5f88\u591a\u5feb\u4e50\uff0c\u6211\u5728\u8fd9\u4e00\u7ae0\u8282\u4f1a\u8bb0\u5f55\u4e00\u4e0b\\n\\n### \u591a\u5c4f\u5e55\\n\\nmacOS \u53ca\u5176\u8fd0\u884c\u7684\u786c\u4ef6\u8bbe\u5907\u5e38\u5e38\u8fde\u63a5\u7740\u591a\u5757\u5c4f\u5e55\uff0c\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u529f\u80fd\u5c31\u662f\u5728\u4efb\u610f\u7684\u5c4f\u5e55\u4f4d\u7f6e\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\uff0c\u8fd9\u5c31\u8981\u786e\u4fdd\uff1a\\n\\n1. \u5c06\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\u653e\u7f6e\u4e8e\u6b63\u786e\u7684\u4f4d\u7f6e\u4e0a\\n2. flutter \u5728\u8bf7\u6c42 native \u6355\u6349\u5c4f\u5e55\u65f6\uff0c\u6355\u6349\u7684\u77e9\u5f62\u6846\u7684\u4f4d\u7f6e\u6b63\u786e\\n3. flutter \u5728\u89e3\u6790 ocr response \u540e\uff0c\u7ed3\u679c\u53ef\u4ee5\u6b63\u786e\u5730\u548c\u5c4f\u5e55\u4e0a\u771f\u6b63\u7684\u5185\u5bb9\u5efa\u7acb\u6620\u5c04\\n\\n\u548c UIKit \u4e2d\u5e38\u7528\u7684 CGRect \u4e0d\u540c\uff0cAppKit \u5bf9\u8bbe\u5907\u5c4f\u5e55\u7684\u62bd\u8c61 `NSScreen`\uff0c\u5176\u5750\u6807\u4ee5 NSRect \u8ba1\u7b97\uff0c\u5750\u6807\u7cfb\u539f\u70b9\u4e3a macOS \u539f\u59cb\u5c4f\u5e55\u7684\u5de6\u4e0b\u89d2\u3002\u800c\u6b64\u65f6\uff0c\u5982\u679c\u4f60\u7ed9\u4f60\u7684\u8bbe\u5907\u8fde\u63a5\u4e0a\u4e86\u5176\u4ed6\u7684\u5c4f\u5e55\uff0c`NSScreen.screens` \u6240\u5448\u73b0\u5c4f\u5e55\u5e03\u5c40\uff0c\u53ef\u80fd\u5c31\u4f1a\u53d8\u6210\u4e0b\u56fe\u6240\u793a\u7684\u6837\u5b50\uff1a\\n![appkit_screens_coordinate](../static/img/appkit_screens_coordinate.png)\\n\\n`screen 1` \u7684\u5de6\u4e0b\u89d2\u5750\u6807\u503c\u5e76\u975e\u662f (0, 0)\uff0c\u800c\u662f `screen 1` \u548c `screen 0` \u7684 (0, 0) \u70b9\u7684\u76f8\u5bf9\u4f4d\u7f6e (w1, d2)\u3002\u901a\u8fc7\u76d1\u542c\u9f20\u6807\u79fb\u52a8\u4e8b\u4ef6\u83b7\u53d6\u7684\u9f20\u6807\u4f4d\u7f6e `event.locationInWindow`\uff0c\u4e5f\u662f\u76f8\u5bf9\u4e8e\u539f\u70b9\u7684 `NSPoint`\\n\\n\u5728\u5f00\u53d1\u65f6\uff0c\u4ec5\u4ec5\u5c06\u81ea\u5df1\u7684\u601d\u7ef4\u4ece iOS \u7684 CGRect \u5750\u6807\u7cfb\u8f6c\u5316\u81f3 NSRect \u5750\u6807\u7cfb\u8fd8\u7b97\u7b80\u5355\u3002\u4f46\u5728\u540e\u7ee7\u7684\u903b\u8f91\u4e2d\uff0c\u56e0\u4e3a OCR \u7ed3\u679c\u7684 `VNRectangleObservation.boundingBox` \u53c8\u4f1a\u56de\u5230 CGRect \u5750\u6807\u7cfb\uff0c\u5750\u6807\u7cfb\u7684\u9891\u7e41\u8f6c\u5316\u786e\u5b9e\u4f1a\u7ed9\u4eba\u5e26\u6765\u4e00\u5b9a\u7684\u56f0\u6270\\n\\n### \u5c4f\u5e55\u6355\u6349\\n\\n\u60f3\u6355\u6349 macOS \u7684\u5c4f\u5e55\uff0c\u4f60\u53ef\u4ee5\u8c03\u7528 `CGDisplayCreateImage` \u51fd\u6570\\n\\n\u4f46\u503c\u5f97\u6ce8\u610f\u7684\u662f\uff0c\u60f3\u8981\u622a\u53d6\u5c4f\u5e55\u4e0a\u5176\u4ed6\u8fdb\u7a0b\u7684\u5185\u5bb9(\u6bd4\u5982 IDE \u6216\u8005\u6d4f\u89c8\u5668)\uff0c\u9700\u8981\u9884\u5148\u901a\u8fc7 `CGRequestScreenCaptureAccess`\uff0c\u7533\u8bf7\u5230\u622a\u53d6\u5176\u4ed6\u8fdb\u7a0b UI \u7684\u6743\u9650\uff0c\u5426\u5219 `CGDisplayCreateImage` \u53ea\u80fd\u62ff\u5230 macOS \u7684\u684c\u9762\uff08\u4ee5\u53ca\u7a0b\u5e8f\u672c\u8eab\uff09\\n\\n### OCR\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 Apple \u4e3a\u5f00\u53d1\u8005\u63d0\u4f9b\u7684 [Vision - Recognizing Text](https://developer.apple.com/documentation/vision/recognizing_text_in_images) \u8fdb\u884c\u5c4f\u5e55\u6587\u672c\u7684\u63d0\u53d6\\n\\nOCR \u8bf7\u6c42\u7684\u8c03\u7528\u7531 flutter side \u5524\u8d77\uff1a\\n\\n```dart\\nvoid dispatchOCRRequestToNative() async {\\n final x = 100;\\n final y = 100;\\n final width = 300;\\n final height = 80;\\n final dimensions = [x, y, width, height];\\n final result = await methodChannel.invokeMethod(\\"captureAndOCR\\", dimensions);\\n\\n // parse the result from native\\n // ...\\n}\\n```\\n\\nNative side \u5728\u63a5\u6536\u5230\u8bf7\u6c42\u540e\uff0c\u4f1a\u6267\u884c\u622a\u5c4f\u548c\u8bc6\u522b\u64cd\u4f5c\uff1a\\n\\n```swift\\nimport Vision\\n// ...\\nlet result : FlutterResult = ...\\nlet image = captureScreen()\\nlet handler = VNImageRequestHandler(cgImage: image)\\nlet request = VNRecognizeTextRequest { request, error in\\n // ocr finished\\n let parsedResult = parse(request.results)\\n result(parsedResult)\\n}\\nhandler.perform([request])\\n```\\n\\n#### \u6027\u80fd\u8868\u73b0\\n\\n\u5728 macbook 2021 \u7684 m1 pro \u4e0a\uff0c\u4ee5 320\u271575 \u7684\u8bbe\u8ba1\u5206\u8fa8\u7387\u622a\u56fe(\u5b9e\u9645\u5206\u8fa8\u7387\u4e3a 640\u2715150)\uff0c\u6bcf\u79d2 11 \u5e27\u7684\u60c5\u51b5\u4e0b\u8fdb\u884c\u957f\u65f6\u95f4\u7684\u622a\u56fe\u548c OCR \u64cd\u4f5c\uff0c\u6574\u4e2a\u6d41\u7a0b\u7684\u5ef6\u8fdf\u5e73\u5747\u7ea6\u4e3a 65ms\uff0c\u6211\u8ba4\u4e3a\u8fd8\u7b97\u662f\u4e00\u4e2a\u53ef\u63a5\u53d7\u7684\u72b6\u6001\\n\\n#### \u8282\u80fd\u4f18\u5316\\n\\n\u5373\u4fbf\u662f\u6027\u80fd\u5141\u8bb8\uff0c\u4f18\u5316\u4e5f\u662f\u5e94\u8be5\u505a\u7684\uff0c\u7528\u6237\u4e00\u65e6\u4f7f\u7528\u4f60\u7684 App\uff0c\u5c31\u611f\u89c9 macbook \u7684 C \u9762\u53d1\u70ed\uff0c\u8fd9\u662f\u65e0\u6cd5\u5bb9\u5fcd\u7684\\n\\n\u5f53\u524d\u5728 OCR \u6d41\u7a0b\u4e2d\u4e3b\u8981\u7684\u6027\u80fd\u4f18\u5316\u6b65\u9aa4\u662f\u5728 native \u7ef4\u62a4\u4e00\u4e2a\u5148\u8fdb\u5148\u51fa\uff0c\u6700\u5927\u5bb9\u91cf\u4e3a 40 \u5e27\u7684\u5b57\u5178\uff0c\u4ee5\u56fe\u7247\u6570\u636e\u4e3a key \u7f13\u5b58 OCR \u7684\u7ed3\u679c\uff0c\u4e0b\u9762\u662f\u7b80\u5316\u540e\u7684\u4ee3\u7801\uff1a\\n\\n```swift\\nlet cacheManager = CacheManager()\\n// ...\\nlet image = captureScreen()\\nlet key = image.dataProvider.data\\nif let cachedResult = cacheManager[key] {\\n delegate.onOCRResult(cachedResult)\\n return\\n}\\n// ...\\ndispatchOCRRequestToDeviceGPU(image) { result\\n cacheManager[key] = result\\n // ... other logic\\n}\\n// ...\\n```\\n\\n\u5f53\u9f20\u6807\u6307\u9488\u4f4d\u7f6e\u4e0d\u53d8\uff0c\u622a\u56fe\u83b7\u53d6\u7684\u56fe\u7247\u4e0d\u53d8\u65f6\uff0cnative \u5728\u5904\u7406 OCR \u8bf7\u6c42\u65f6\u4f1a\u5148\u547d\u4e2d\u7f13\u5b58\uff0c\u5e76\u76f4\u63a5\u8fd4\u56de\u7ed3\u679c\uff0c\u4ee5\u51cf\u5c11\u975e\u5fc5\u9700\u7684 GPU \u8c03\u7528\\n\\n\u5728\u7cfb\u7edf\u81ea\u5e26\u7684\u6d3b\u52a8\u76d1\u89c6\u5668\u4e2d\u67e5\u770b\u8fdb\u7a0b\u3002\u53d1\u73b0\uff0c\u5728\u5e94\u7528\u7684\u7f13\u5b58\u7b56\u7565\u540e\uff0c\u5728\u672c\u7a0b\u5e8f\u6d3b\u52a8\u65f6\uff0c\u5176 CPU/GPU \u5360\u7528\u7387\u786e\u5b9e\u964d\u4f4e\u4e86\u5f88\u591a\uff0c\u540c\u65f6\uff0cOCR \u7684\u7f13\u5b58\u7ed3\u679c\u88ab\u964d\u4e3a\u4e86 5ms\\n\\n:::note\\n\u6211\u611f\u89c9\uff0c\u4ee5 `CFData` \u4f5c\u4e3a key \u67e5\u8be2\u5b57\u5178\u8fd8\u4e0d\u662f\u6548\u7387\u6700\u9ad8\u7684\u7b97\u6cd5\uff0c\u5e94\u8be5\u53ef\u4ee5\u7ee7\u7eed\u63a2\u7d22\u4e00\u4e0b\\n:::\\n\\n#### \u4e0d\u622a\u53d6\u81ea\u5df1\\n\\n\u5728\u6211\u8fdb\u884c\u5f00\u53d1\u65f6\uff0c\u53d1\u73b0\u5f53\u672c\u7a0b\u5e8f\u5728\u5c55\u793a\u5355\u8bcd\u91ca\u4e49 UI (HUD)\u65f6\uff0c\u56e0\u4e3a HUD \u672c\u8eab\u4e5f\u4f1a\u5728\u4e00\u5b9a\u8303\u56f4\u5185\u88ab\u622a\u5c4f\u51fd\u6570\u6355\u83b7\uff0c\u5bfc\u81f4 HUD \u4f1a\u5f71\u54cd OCR \u7684\u7ed3\u679c\uff0c\u800c\u5728 OCR \u7ed3\u679c\u53d8\u52a8\u540e\uff0cHUD \u53c8\u4f1a\u968f\u4e4b\u53d1\u751f\u53d8\u52a8\uff0c\u518d\u6b21\u5f71\u54cd OCR \u7ed3\u679c\uff0c\u5c31\u8fd9\u6837\u5faa\u73af\u5f80\u590d\uff0c\u8fde\u7f13\u5b58\u4e5f\u90fd\u5931\u6548\u4e86\u3002\u5728\u67e5\u9605\u548c\u5c1d\u8bd5\u5927\u91cf\u7684 API \u540e\uff0c\u6211\u7ec8\u4e8e\u5728 `NSWindow` \u4e2d\u627e\u5230\u4e86 `sharingType` \u8fd9\u4e2a\u5c5e\u6027\uff0c\u5c4f\u5e55\u6355\u6349\u65b9\u6cd5\u6355\u83b7 HUD \u81ea\u8eab\uff0c\u603b\u7b97\u662f\u6253\u7834\u4e86\u8fd9\u4e2a\u94fe\u6761\\n\\n### \u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d\\n\\n\u5bf9 Vision - VNRecognizeTextRequest \u7684\u7ed3\u679c\u89e3\u6790\u662f\u4e2a\u590d\u6742\u7684\u4efb\u52a1\\n\\n\u6267\u884c\u8fd9\u4e2a\u590d\u6742\u4efb\u52a1\u7684\u539f\u56e0\u6709\u4e09\uff1a\\n\\n1. \u672c\u7a0b\u5e8f\u8bbe\u8ba1\u7684\u4ea4\u4e92\u662f\u201c\u7528\u6237\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u611f\u5174\u8da3\u7684\u5355\u8bcd\u4e0a\u65f6\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u201d\uff0c\u8fd9\u5c31\u8981\u6c42\u6211\u4eec\u77e5\u9053\u5c4f\u5e55\u4e0a\u6bcf\u4e2a\u5355\u8bcd\u7684\u5177\u4f53\u4f4d\u7f6e\u548c\u5185\u5bb9\\n2. \u672c\u7a0b\u5e8f\u9488\u5bf9\u6e90\u4ee3\u7801\u7279\u5316\uff0c\u5728\u9762\u5bf9[\u9a7c\u5cf0\u547d\u540d\u6cd5](https://zh.wikipedia.org/zh-cn/%E9%A9%BC%E5%B3%B0%E5%BC%8F%E5%A4%A7%E5%B0%8F%E5%86%99)\u65f6\uff0c\u9700\u8981\u77e5\u9053\u6784\u6210\u4e00\u4e2a symbol \u7684\u6bcf\u4e2a\u5355\u8bcd\u7684\u610f\u601d\uff1a`ThisIsAVeryVeryLongClassName` -> `This`,~~`Is`~~,~~`A`~~,`Very`,`Very`,`Long`,`Class`,`Name` (\u56e0\u8fc7\u4e8e\u7b80\u5355\uff0c\u4e22\u5f03\u957f\u5ea6\u5c0f\u4e8e 3 \u7684\u82f1\u8bed\u5355\u8bcd)\\n3. \u6211\u4eec\u8981\u4ee5\u5355\u8bcd\u7684\u6587\u672c\u5185\u5bb9\u4e3a key\uff0c\u67e5\u8be2\u6570\u636e\u5e93\u3002\u5728\u6587\u672c\u5e8f\u5217\u5305\u542b\u7279\u6b8a\u5b57\u7b26\u65f6\uff0c\u6211\u4eec\u662f\u65e0\u6cd5\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u5230\u5355\u8bcd\u91ca\u4e49\u7684\u3002\u6240\u4ee5\u8981\u79fb\u9664\u975e\u5b57\u6bcd\u5b57\u7b26\u3002\u540c\u65f6\uff0c\u8fd9\u4e00\u64cd\u4f5c\u4e5f\u53ef\u4ee5\u81ea\u7136\u800c\u7136\u5730\u9002\u914d\u7f16\u7a0b\u8bed\u8a00\u4e2d\u7684\u5176\u4ed6\u547d\u540d\u65b9\u5f0f\uff0c\u6bd4\u5982\u79fb\u9664\u4e0b\u5212\u7ebf\u5c31\u53ef\u4ee5\u9002\u5e94[\u86c7\u5f62\u547d\u540d\u6cd5](https://zh.wikipedia.org/zh-cn/%E8%9B%87%E5%BD%A2%E5%91%BD%E5%90%8D%E6%B3%95)\\n\\n\u5047\u8bbe\u6211\u4eec\u5728\u5bf9\u4e0b\u9762\u7684\u56fe\u7247\u6267\u884c OCR \u8bf7\u6c42\\n\\n![ocr_text_source](ocr_text_source.png)\\n\\n\u5728 Native \u7aef\u53d6 `VNRecognizeTextRequest.results.topCandidates(1)` \u540e\uff0c\u7ed3\u679c\u5982\u4e0b\\n\\n```swift\\n[\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Vision provides its text-recognition capabilities through VNRecognizeText\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Request, an image-based request type that finds and extracts text in images. The\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"following example shows how to use VNImageRequestHandler to perform a\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"VNRecognizeTextRequest for recognizing text in the specified CGImage.\\",\\n]\\n```\\n\\n\u5728 Native \u7aef\u4ee5**\u7a7a\u683c**\u5b57\u7b26\u548c `VNRecognizedText.boundingBox(for:)` \u5bf9 OCR \u7ed3\u679c\u8fdb\u884c\u7b2c\u4e00\u6b21\u6574\u7406\uff0c\u4e0a\u8ff0\u7684\u7ed3\u679c\u4f1a\u53d8\u4e3a\u8fd9\u6837\uff1a\\n\\n```swift\\n[\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Vision\\", \\"provides\\", \\"its\\", \\"text-recognition\\", \\"capabilities\\", \\"through\\", \\"VNRecognizeText\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Request,\\", \\"an\\", \\"image-based\\", \\"request\\", \\"type\\", \\"that\\", \\"finds\\", \\"and\\", \\"extracts\\", \\"text\\", \\"in\\", \\"images.\\", \\"The\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"following\\", \\"example\\", \\"shows\\", \\"how\\", \\"to\\", \\"use\\", \\"VNImageRequestHandler\\", \\"to\\", \\"perform\\", \\"a\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"VNRecognizeTextRequest\\", \\"for\\", \\"recognizing\\", \\"text\\", \\"in\\", \\"the\\", \\"specified\\", \\"CGImage.\\",\\n]\\n```\\n\\n\u6700\u540e\u5c06 boundingBox \u7684\u8fd4\u56de\u503c\u548c\u622a\u5c4f\u5c3a\u5bf8\u8fdb\u884c\u4e58\u7b97\uff0c\u8fd4\u56de\u7ed9 flutter \u7aef\u3002Flutter \u7aef\u5728\u62ff\u5230\u7ed3\u679c\u540e\u4e3b\u8981\u8fdb\u884c\u4e24\u4e2a\u6b65\u9aa4\uff1a\\n\\n1. \u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u79fb\u9664\u975e\u82f1\u6587\u5b57\u7b26\uff0c\u8ba1\u7b97\u65b0\u7684 rect\\n2. \u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u5bf9\u5305\u542b\u5927\u5199\u5b57\u6bcd\u7684\u5b57\u7b26\u4e32\u8fdb\u884c\u5206\u5272\uff0c\u8ba1\u7b97\u65b0\u7684 rect\\n\\n\u8fd9\u6837\uff0c\u6587\u672c\u5757\u7684\u5185\u5bb9\u5728 flutter \u7aef\u5c31\u4f1a\u53d8\u6210\u8fd9\u6837\uff1a\\n\\n```dart\\n[\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Vision\\", \\"provides\\", \\"its\\", \\"text\\", \\"recognition\\", \\"capabilities\\", \\"through\\", \\"Recognize\\", \\"Text\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Request\\",\\"image\\", \\"based\\", \\"request\\", \\"type\\", \\"that\\", \\"finds\\", \\"and\\", \\"extracts\\", \\"text\\", \\"images\\", \\"The\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"following\\", \\"example\\", \\"shows\\", \\"how\\", \\"use\\", \\"Image\\", \\"Request\\", \\"Handler\\", \\"perform\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Recognize\\", \\"Text\\", \\"Request\\", \\"for\\", \\"recognizing\\", \\"text\\", \\"the\\", \\"specified\\", \\"Image\\",\\n]\\n```\\n\\n\u6700\u540e\uff0cflutter \u4fa7\u4f1a\u4ee5\u4e0b\u9762\u7684\u4ee3\u7801\u5bf9\u8fd9\u4e9b OCR \u4fe1\u606f\u8fdb\u884c\u62bd\u8c61\uff1a\\n\\n```dart\\n/// The representation of block\\n///\\n/// \u6587\u672c\u5757\u8868\u5f81\\nclass Block {\\n String text;\\n double x;\\n double y;\\n double w;\\n double h;\\n}\\n\\n// ...\\n\\n/// All text representations in memory\\n///\\n/// \u6240\u6709\u6587\u672c\u5757\u8868\u5f81\\nfinal blocks = StateProvider>((_) => []);\\n```\\n\\n\u4e0a\u9762\u7684\u6570\u636e\u5728\u5c4f\u5e55\u4e0a\u7684\u53ef\u89c6\u5316\u6548\u679c\u5982\u4e0b\uff1a\\n![text_block_representation_2](../static/img/text_block_representation_1.png)\\n\\n\u5176\u4e2d\u5916\u56f4\u7eff\u6846\u4ee3\u8868\u622a\u5c4f\u8303\u56f4\uff0c\u5185\u90e8\u7684\u5c0f\u7eff\u6846\u4ee3\u8868 native \u7aef\u8fd4\u56de\u7684\u7ed3\u679c\uff0c\u5c0f\u7eff\u6846\u5185\u90e8\u7684\u84dd\u8272\u6846\u4ee3\u8868 flutter side \u7b2c\u4e8c\u6b21\u89e3\u6790\u7ed3\u679c\u3002\u53ef\u4ee5\u770b\u5230\uff0c`\\"VNImageRequestHandler\\"` \u8fd9\u4e2a\u5b57\u7b26\u4e32\u88ab\u5206\u5272\u4e3a\u4e86 `\\"Image\\"`, `\\"Request\\"`, `\\"Handler\\"` \u8fd9\u4e09\u4e2a\u6587\u672c\u5757\u3002\u8fd9\u6837\uff0c\u6211\u5c31\u53ef\u4ee5\u77e5\u9053\u7528\u6237\u5230\u5e95\u5bf9\u54ea\u4e00\u6bb5\u5b57\u6bcd\u5e8f\u5217\uff08\u5355\u8bcd\uff09\u611f\u5174\u8da3\u4e86\uff08\u5047\u8bbe\u6e90\u4ee3\u7801\u7b26\u53f7\u7684\u547d\u540d\u662f\u89c4\u8303\u7684 \ud83d\ude07\uff09\u3002\\n\\n:::note\\n\u4f60\u4e5f\u53ef\u4ee5\u6253\u5f00\u672c\u7a0b\u5e8f\u7684\u7528\u6237\u8bbe\u7f6e\u9762\u677f\u6765\u76f4\u63a5\u89c2\u5bdf\u5176\u8fd0\u884c\u65f6\u8868\u73b0\uff1a\\n![dashboard_inspect](./../static/img/dashboard_inspect.png)\\n:::\\n\\n### \u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65\\n\\n\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u4f1a\u521b\u5efa\u4e24\u4e2a flutter engine\uff0c\u5206\u522b\u7528\u4e8e\u6e32\u67d3\\"\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\\"\u548c\\"\u6b22\u8fce\u4e0e\u8bbe\u7f6e\u9875\u9762(Dashboard)\\"\\n\\n\u8fd9\u4e24\u4e2a\u5f15\u64ce\u5747\u901a\u8fc7\u57fa\u672c\u65b9\u5f0f\u521b\u5efa\uff0c\u5747\u91c7\u7528 `FlutterMethodChannel` \u4e0e native \u8fdb\u884c\u901a\u8baf\\n\\n\u53cc\u5f15\u64ce\u5171\u4eab\u540c\u4e00\u4efd\u4ee3\u7801\uff0c\u7ecf\u7531\u4e0d\u540c\u7684 [dart \u5165\u53e3\u51fd\u6570](https://github.com/dart-lang/sdk/blob/master/runtime/docs/compiler/aot/entry_point_pragma.md)\u542f\u52a8\uff1a\\n\\n```swift\\n// \u8bbe\u7f6e\u9875\u9762 (Dashboard)\\ndashboardEngine = FlutterEngine(name: \\"dashboard\\", project: nil)\\ndashboardChannel = FlutterMethodChannel(name: \\"dashboard\\", binaryMessenger: dashboardEngine.binaryMessenger)\\ndashboardEngine.run(withEntrypoint: \\"_dashboard\\")\\n//...\\n\\n// \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\\nhudEngine = FlutterEngine(name: \\"hud\\", project: nil)\\nhudChannel = FlutterMethodChannel(name: \\"hud\\", binaryMessenger: hudEngine.binaryMessenger)\\nhudChannel.run(withEntrypoint: \\"_hud\\")\\n```\\n\\n:::note\\n\u6211\u4e2a\u4eba\u6ca1\u6709\u4f7f\u7528\u5b98\u65b9\u63d0\u4f9b\u7684 [FlutterEngineGroup \u65b9\u6848](https://docs.flutter.dev/add-to-app/multiple-flutters)\uff0c\u56e0\u4e3a\u5728\u5f53\u521d\u8fdb\u884c\u5c1d\u8bd5\u65f6\u53d1\u73b0\u4e86\u5f15\u64ce\u65e0\u54cd\u5e94\u7684\u95ee\u9898\uff0c\u81f3\u4eca\u4ecd\u7136\u662f P2 \u7ea7\u522b\u7684 open [issue](https://github.com/flutter/flutter/issues/119403)\\n:::\\n\\n#### \u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 [riverpod](https://riverpod.dev/) \u6765\u7ba1\u7406\u7edd\u5927\u90e8\u5206\u72b6\u6001\uff0c\u6240\u4ee5\u5728\u540c\u6b65\u72b6\u6001\u65f6\uff0c\u6211\u4e5f\u671f\u671b\u81ea\u5df1\u7684\u81ea\u5df1\u7684\u5fc3\u667a\u6a21\u578b\u53ef\u4ee5\u66f4\u8d34\u8fd1 riverpod\\n\\n\u5728\u8fd0\u884c\u65f6\uff0cDashboard engine \u9700\u8981\u5c06\u7528\u6237\u8bbe\u7f6e\u540c\u6b65\u81f3 HUD engine (\u6bd4\u5982\u66f4\u6539\u67e5\u8bcd\u5feb\u6377\u952e)\uff0c\u6211\u5728\u8fd9\u91cc\u76d1\u542c\u4e86 Dashboard engine \u5bf9\u5e94\u7684 `StateProvider` \u7684\u53d8\u66f4\uff0c\u5e76\u901a\u8fc7 method channel\uff0c\u7ecf\u7531 native \u8f6c\u53d1\u81f3 HUD engine\uff0cHUD engine \u5728\u6536\u5230\u4e86\u901a\u77e5\u540e\uff0c\u518d\u66f4\u65b0\u81ea\u5df1\u7ef4\u62a4\u7684 `StateProvider`\u3002\u4ece\u800c\u5b9e\u73b0\u4e86\u8de8\u5f15\u64ce\u7684\u72b6\u6001\u540c\u6b65\\n\\n### \u91ca\u4e49\u67e5\u8be2\\n\\n\u672c\u7a0b\u5e8f\u5728\u67e5\u8be2\u5355\u8bcd\u91ca\u4e49\u65f6\u4f7f\u7528\u672c\u5730 sqlite \u6570\u636e\u5e93\uff0c\u6570\u636e\u5e93\u6e90\u4e8e [ecdict-ultimate](https://github.com/skywind3000/ECDICT-ultimate)\\n\\n\u9274\u4e8e\u5f00\u6e90\u6570\u636e\u5e93\u8fc7\u4e8e\u5de8\u5927\uff0c\u4e14\u5b58\u5728\u4e00\u4e9b\u672c\u7a0b\u5e8f\u65e0\u9700\u4f7f\u7528\u7684 columns\uff0c\u6211\u53c8\u4f7f\u7528 dart \u548c [drift](https://pub.dev/packages/drift) \u5bf9\u5176\u8fdb\u884c\u4e86\u6e05\u6d17\u548c\u526a\u88c1\\n\\n\u5728\u67e5\u8be2\u65f6\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5c06\u67e5\u8be2\u7684\u672c\u6587\u4e32\u201c\u6253\u6563\u201d\uff0c\u5e76\u4e00\u9f50\u8bf7\u6c42\u6570\u636e\u5e93\uff0c\u4ee5\u671f\u80fd\u547d\u4e2d\u4e0d\u65ad\u5343\u53d8\u4e07\u5316\u7684\u82f1\u8bed\u5355\u8bcd\uff1a\\n\\n```dart\\nfinal sequence = \\"qwertyuiopasdfghjkl\\";\\nfinal sequences = [\\"qwe\\",\\"wer\\",\\"ert\\",...,\\"qwer\\",\\"wert\\",...,\\"qwert\\",...,\\"qwertyuiopasdfghjkl\\"];\\nfinal List results = await queryDB(keys:sequences); // average latency: ~6ms\\n```\\n\\n\u5728\u8bf7\u6c42\u5b8c\u6210\u540e\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5bf9\u8bf7\u6c42\u7ed3\u679c\u8fdb\u884c\u53bb\u91cd\uff0c\u903b\u8f91\u5982\u4e0b\uff1a\\n\\n- \u7ed3\u679c\u4e2d\u5305\u542b owl (\u732b\u5934\u9e70)\\n- \u7ed3\u679c\u4e2d\u5305\u542b knowledge(\u77e5\u8bc6)\\n- \u5e8f\u5217 \\"knowledge\\" \u5305\u542b \\"owl\\"\uff0c\u79fb\u9664\u67e5\u8be2\u7ed3\u679c\u4e2d\u7684 owl\\n\\n:::note\\n\u76ee\u524d\uff0c\u8be5\u903b\u8f91\u8fd8\u6ca1\u6709\u5b9e\u73b0\u5bf9\u51b2\u7a81\u7684\u5224\u5b9a\uff1a\u5982\u5411\u6570\u636e\u5e93\u67e5\u8be2\u7684 key \u4e3a `\\"gitignore\\"`\uff0c\u67e5\u8be2\u7ed3\u679c\u4e3a `[\\"git\\", \\"tig\\", \\"ignore\\"]`\uff0c\u76ee\u524d\u8fd9\u4e09\u4e2a\u7ed3\u679c\u90fd\u4f1a\u88ab\u6e32\u67d3\u5230 HUD \u4e0a\u3002\u4f46\u663e\u7136\uff0c\u9009\u53d6 git \u548c ignore \u8fd9\u4e24\u4e2a\u67e5\u8be2\u7ed3\u679c\u9020\u6210\u7684 \u201c\u51b2\u7a81\u201d \u662f\u6700\u5c0f\u7684\\n:::\\n\\n### \u53d1\u97f3\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 [AVFoundation - Speech synthesis](https://developer.apple.com/documentation/avfoundation/speech_synthesis) API \u6765\u5b9e\u65f6\u5408\u6210\u8bed\u97f3\\n\\n\u8be5 API \u53ef\u4ee5\u540c\u65f6\u5728 iOS/macOS \u4e0a\u4f7f\u7528\uff0c\u4e14\u53d1\u97f3\u8f83\u4e3a\u51c6\u786e\uff0c\u4ee5\u6211\u81ea\u8eab\u7684\u80fd\u529b(\u5168\u56fd\u5927\u5b66\u82f1\u8bed\u56db\u7ea7)\u6765\u770b\uff0c\u6548\u679c\u8fd8\u7b97\u6ee1\u610f\uff0c\u786e\u5b9e\u6bd4\u6211\u7684\u53d1\u97f3\u51c6 \ud83e\udd23\\n\\n\u5f53\u7136\uff0c\u540e\u7ee7\u5982\u679c\u6709\u66f4\u9ad8\u7684\u9700\u6c42\uff0c\u4e5f\u6709\u5f88\u591a\u5176\u4ed6\u53ef\u9009\u65b9\u6848"},{"id":"motivation","metadata":{"permalink":"/valo-reader-doc/blog/motivation","editUrl":"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/motivation.md","source":"@site/blog/motivation.md","title":"\u5f00\u53d1\u76ee\u7684","description":"\u6211\u7684\u9700\u6c42","date":"2023-03-15T19:25:06.000Z","formattedDate":"2023\u5e743\u670815\u65e5","tags":[{"label":"motivation","permalink":"/valo-reader-doc/blog/tags/motivation"}],"readingTime":5.835,"hasTruncateMarker":false,"authors":[{"name":"Ce Wang","title":"Solo dev","url":"https://github.com/haloWang","imageURL":"https://github.com/haloWang.png","key":"halowang"}],"frontMatter":{"slug":"motivation","title":"\u5f00\u53d1\u76ee\u7684","authors":["halowang"],"tags":["motivation"]},"prevItem":{"title":"Valo Reader for macOS \u6280\u672f\u6982\u8981","permalink":"/valo-reader-doc/blog/tech_detail_macos"}},"content":"## \u6211\u7684\u9700\u6c42\\n\\n\u4ee5\u4e0b\u662f\u6211\u7684\u4e24\u4e2a\u5e38\u7528\u9700\u6c42\uff1a\\n\\n- \u9605\u8bfb\u548c\u7406\u89e3\u6d4f\u89c8\u5668\u4e2d\u7684\u82f1\u6587\u5185\u5bb9\uff0c\u5982 github\uff0cstackoverflow\uff0creddit\\n- \u5728\u684c\u9762\u5e73\u53f0\u4e0a\u9605\u8bfb\u548c\u7406\u89e3\u4e00\u4e9b\u4ece\u672a\u88ab\u7ffb\u8bd1\u6210\u6c49\u8bed\u7684\u82f1\u6587\u4e66\u7c4d\uff0c\u8fd9\u4e9b\u4e66\u7c4d\u7684\u5927\u90e8\u5206\u4e3a PDF \u6587\u4ef6\\n\\n## \u6211\u9047\u5230\u7684\u56f0\u96be\\n\\n\u4f46\u5728\u8fdb\u884c\u4e0a\u8ff0\u7684\u9605\u8bfb\u7406\u89e3\u6d3b\u52a8\u65f6\uff0c\u6211\u9047\u5230\u4e86\u4e00\u4e9b\u56f0\u96be\uff1a\\n\\n- \u6211\u7ecf\u5e38\u5fd8\u8bb0\u82f1\u8bed\u5355\u8bcd\u7684\u610f\u601d\uff0c\u6216\u4e0d\u6562\u786e\u8ba4\u4e00\u4e2a\u5355\u8bcd\u7684\u610f\u601d\u3002\u751a\u81f3\uff0c\u6211\u7ecf\u5e38\u4f1a\u5fd8\u8bb0 30 \u79d2\u4e4b\u524d\u521a\u521a\u67e5\u8fc7\u7684\u5355\u8bcd\\n- \u5f53\u524d\u5404\u79cd\u67e5\u5355\u8bcd\u7684\u5e94\u7528\u7a0b\u5e8f\u548c\u7f51\u9875\u5bf9\u4e8e\u6211\u6765\u6765\u8bf4\u90fd\u6709\u4e9b\u7e41\u7410\uff0c\u6211\u9700\u8981\u7b49\u5f85\u8bcd\u5178\u8f6f\u4ef6\u7684\u7f51\u7edc\u8bf7\u6c42\uff0c\u7528\u4eba\u8111\u5ffd\u7565\u8bcd\u5178\u8f6f\u4ef6\u4e2d\u7684\u5e7f\u544a\u3002\u6211\u8fd8\u8981\u4fdd\u8bc1\u81ea\u5df1\u5728\u7e41\u7410\u7684\u64cd\u4f5c\u6d41\u7a0b\u4e2d\u4e0d\u4f1a\u5206\u5fc3\\n- \u4f7f\u7528\u5168\u6587\u7ffb\u8bd1\u8f6f\u4ef6\uff0c\u4f1a\u5bfc\u81f4\u76f8\u5f53\u4e00\u90e8\u5206\u7684\u4fe1\u606f\u4e22\u5931\u6216\u504f\u79bb\u539f\u672c\u542b\u4e49\uff0c\u6211\u9700\u8981\u5bf9\u5168\u6587\u7ffb\u8bd1\u8f6f\u4ef6\u7ea0\u9519\uff0c\u6216\u8005\u76f4\u63a5\u67e5\u770b\u539f\u6587\u624d\u80fd\u4e86\u89e3\u5230\u771f\u6b63\u7684\u91ca\u4e49\\n\\n\u4e3e\u4e2a\u4f8b\u5b50\uff0c\u6211\u4e0d\u77e5\u9053\u201cbabel\u201d\u8fd9\u4e2a\u5355\u8bcd\u7684\u6c49\u8bed\u610f\u601d\u3002\u4f46\u662f\uff0c\u5f53\u8fd9\u4e2a\u8bcd\u5728\u82f1\u8bed\u8bed\u6599\u4e2d\u51fa\u73b0\u65f6\uff0c\u5b83\u5c06\u4f60\u7684\u601d\u7eea\u4ece\u201c\u601d\u8003\u4f5c\u8005\u8868\u8fbe\u7684\u542b\u4e49\u201d\uff0c\u8f6c\u53d8\u4e3a\u201c\u601d\u8003 babel \u662f\u4ec0\u4e48\u610f\u601d\u201d\u3002\u5f53\u6211\uff0c\u67e5\u8be2\u5230\u4e86\u201cbabel\u201d\u7684\u610f\u601d\u540e\uff0c\u5728\u8fd4\u56de\u6765\u5c06\u81ea\u5df1\u7684\u6ce8\u610f\u529b\u6295\u5165\u4e4b\u524d\u7684\u4e0a\u4e0b\u6587\u3002\u8fd9\u4e2a\u53d1\u751f\u5728\u4eba\u8111\u4e2d\u7684\u4e0a\u4e0b\u6587\u5207\u6362\uff0c\u8fc7\u7a0b\u8017\u65f6\uff0c\u5bb9\u6613\u51fa\u9519\uff0c\u6b63\u53cd\u9988\u51e0\u4e4e\u6ca1\u6709\u3002\\n\\n\u6211\u8ba4\u4e3a\uff0c\u8fd9\u4e9b\u56f0\u96be\u589e\u52a0\u4e86\u6211\u4ece\u4e8b\u8111\u529b\u6d3b\u52a8\u65f6\u7684\u5fc3\u667a\u8d1f\u62c5\uff0c\u964d\u4f4e\u4e86\u6211\u7684\u9605\u8bfb\u6548\u7387\u3002\u751a\u81f3\uff0c\u76f4\u63a5\u5bfc\u81f4\u6211\u5206\u5fc3\uff0c\u7cbe\u795e\u4e0d\u96c6\u4e2d\uff0c\u6700\u7ec8\u653e\u5f03\u9605\u8bfb\uff0c\u653e\u5f03\u83b7\u53d6\u77e5\u8bc6\u3002\\n\\n\u9762\u5bf9\u4e0a\u8ff0\u7684\u56f0\u96be\uff0c\u6211\u4eec\u901a\u5e38\u7684\u89e3\u51b3\u529e\u6cd5\u90fd\u662f\u201c\u80cc\u5355\u8bcd\u201d\u3002\u53ef\u662f\uff0c\u80cc\u5355\u8bcd\u5bf9\u4e8e\u6211\u6765\u8bf4\u662f\u4e2a\u75db\u82e6\u7684\u8fc7\u7a0b\uff1a\\n\\n- \u6211\u9700\u8981\u80cc\u8bf5\u5927\u91cf\u7684\uff0c\u4f7f\u7528\u9891\u7387\u6781\u4f4e\uff0c\u5e76\u4e14\u56e0\u4f7f\u7528\u9891\u7387\u4f4e\u800c\u975e\u5e38\u5bb9\u6613\u53d8\u5f97\u751f\u758f\u7684\u5355\u8bcd\\n- \u8003\u7eb2\u5185\u7684\u5355\u8bcd\u4ece\u6765\u90fd\u662f\u4e3a\u5927\u5bb6\u51c6\u5907\u7684\uff0c\u80cc\u8bf5\u8003\u7eb2\u5185\u7684\u5355\u8bcd\u4e0d\u80fd\u8986\u76d6\u5230\u4e13\u4e1a\u9886\u57df\u7684\u8bcd\u6c47\\n- \u5728\u6c49\u8bed\u73af\u5883\u4e2d\uff0c\u4fdd\u8bc1\u5927\u8111\u5bf9\u5927\u91cf\u82f1\u8bed\u5355\u8bcd\u7684\u719f\u6089\u548c\u5feb\u901f\u54cd\u5e94\uff0c\u9700\u8981\u82b1\u8d39\u5f88\u591a\u989d\u5916\u7684\u65f6\u95f4\u4e0e\u7cbe\u529b\\n\\n\u5bf9\u6211\u6765\u8bf4\uff0c\u80cc\u5355\u8bcd\u7684\u6548\u7387\u592a\u4f4e\u4e86\uff0c\u5956\u52b1\u6765\u7684\u592a\u665a\uff0c\u592a\u4e0d\u786e\u5b9a\uff0c\u9700\u8981\u5f3a\u5927\u7684\u610f\u5fd7\u529b\u624d\u80fd\u6301\u7eed\u4e0b\u53bb\u3002\u6295\u5165\u4ea7\u51fa\u6bd4\u592a\u9ad8\u3002\\n\\n### \u77db\u76fe\\n\\n\u4e0a\u8ff0\u8fd9\u4e9b\u52a8\u4f5c\u90fd\u4f1a\u5bf9\u4e8e\u6211\u6765\u8bf4\u662f\u8f83\u5927\u7684\u5fc3\u667a\u8d1f\u62c5\u3002\u4e0d\u65ad\u5730\u5e72\u6270\u7740\u6211\u7684**_\u9605\u8bfb - \u7406\u89e3_**\u8fc7\u7a0b\u3002\u963b\u788d\u6211\u901a\u8fc7\u82f1\u8bed\u83b7\u53d6\u4fe1\u606f\u4e0e\u77e5\u8bc6\uff0c\u800c\u6211\u5f53\u524d\u6240\u671f\u671b\u83b7\u53d6\u7684\u4fe1\u606f\u4e0e\u77e5\u8bc6\uff0c\u5927\u90fd\u7531\u82f1\u8bed\u4e66\u5199\u800c\u6210\u3002\\n\\n## \u89e3\u51b3\u529e\u6cd5\\n\\n\u9762\u5bf9\u8fd9\u4e9b\u56f0\u96be\u3002\u6211\u7ecf\u8fc7\u79cd\u79cd\u601d\u8003\uff0c\u63a2\u7d22\uff0c\u5b9e\u8df5\u3002\u5f00\u53d1\u4e86\u8fd9\u6b3e\u540d\u4e3a [Valo Reader](http://localhost:3000/valo-reader-doc/docs/intro) \u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u8fd9\u6b3e\u5e94\u7528\u7a0b\u5e8f\u901a\u8fc7 [OCR](https://zh.wikipedia.org/wiki/%E5%85%89%E5%AD%A6%E5%AD%97%E7%AC%A6%E8%AF%86%E5%88%AB) \u6280\u672f\uff0c\u5b9e\u65f6\u5730\u8bc6\u522b\u7528\u6237\u5f53\u524d\u6b63\u5728\u9605\u8bfb\u7684\u82f1\u6587\u6587\u672c\uff0c\u7ecf\u7531\u7a0b\u5e8f\u5185\u7f6e\u8bcd\u5178\u548c\u7528\u6237\u8bbe\u7f6e\uff0c\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u3002\u5728\u8fd9\u6b3e\u5e94\u7528\u7a0b\u4e2d\uff0c\u6211\u671f\u671b\u63a5\u8fd1\u53ef\u80fd\u5730\u964d\u4f4e\u7528\u6237\u5728\u9605\u8bfb\u82f1\u6587\u8bed\u6599\u65f6\uff0c\u56e0\u4e3a\u638c\u63e1\u7684\u82f1\u8bed\u8bcd\u6c47\u91cf\u8f83\u5c11\uff0c\u800c\u4ea7\u751f\u7684\u989d\u5916\u5fc3\u667a\u8d1f\u62c5\u3002\\n\\n\ud83c\udf89 \u6211\u5e0c\u671b\uff0c\u6211\u4eec\u5728\u4f7f\u7528\u8fd9\u6b3e\u8f6f\u4ef6\u9605\u8bfb\u82f1\u6587\uff0c\u4e0d\u9700\u8981\u6709\u5355\u8bcd\u57fa\u7840\u4e86\uff0c\u8fd9\u53ef\u4ee5**\u5927\u5e45\u51cf\u5c11**\u751f\u758f\u5355\u8bcd\u5e26\u6765\u7684\u5fc3\u667a\u8d1f\u62c5\u3002\u52a9\u529b\u6211\u4eec\u7684**_\u9605\u8bfb - \u7406\u89e3_**\u8fc7\u7a0b\u3002\\n\\n![AppIcon](/img/logo.png)\\n\\n:::note\\n\u60f3\u8981\u5728\u4e0d\u540c\u5e73\u53f0\u4e0a\u83b7\u53d6 Valo Reader\uff0c\u8bf7\u53c2\u9605[\u5b89\u88c5](/docs/installation)\\n:::\\n\\n## \u4e3a\u4ec0\u4e48\u4e0d\u662f macOS \u81ea\u5e26\u7684\u4e09\u6307\u8f7b\u70b9\\n\\n\u5f53\u7528\u6237\u5728\u4f7f\u7528 macOS \u65f6\u53ef\u4ee5\u901a\u8fc7\u89e6\u6478\u677f\u5feb\u901f\u5524\u8d77\u7cfb\u7edf\u81ea\u5e26\u7684\u5b57\u5178\u67e5\u8be2\u5355\u8bcd, \u8fd9\u4e2a\u529f\u80fd\u88ab macOS \u79f0\u4e3a\u67e5\u8be2\u4e0e\u6570\u636e\u68c0\u6d4b\u5668(Look up & data detectors)\\n\\n\u76f8\u6bd4\u4e8e\u7cfb\u7edf\u81ea\u5e26\u529f\u80fd, Valo Reader \u5f53\u524d\u4e3b\u8981\u6709\u4ee5\u4e0b\u4f18\u52bf:\\n\\n- \u6240\u89c1\u5373\u53ef\u67e5: \u53ef\u4ee5\u53ca\u65f6\u67e5\u8be2\u56fe\u7247, \u89c6\u9891\u4e43\u81f3\u7535\u5b50\u6e38\u620f\u4e2d\u7684\u82f1\u8bed\u5355\u8bcd, \u53ea\u8981\u8fd9\u4e2a\u82f1\u8bed\u5355\u8bcd\u663e\u793a\u5728\u4e86\u5c4f\u5e55\u4e0a, \u8fd9\u662f macOS \u7cfb\u7edf\u6240\u505a\u4e0d\u5230\u7684\\n- \u7f16\u7a0b\u53cb\u597d: \u53ef\u4ee5\u67e5\u8be2\u6e90\u4ee3\u7801\u4e2d\u7684\u6bcf\u4e2a\u5355\u8bcd: `PascalCase` \u5c06\u88ab Valo Reader \u8bc6\u522b\u4e3a `\\"pascal\\"` \u548c `\\"case\\"` \u5e76\u88ab\u5206\u522b\u67e5\u8be2\\n\\n## \u4e3a\u4ec0\u4e48\u4e0d\u662f\u5176\u4ed6\u8bcd\u5178\u8f6f\u4ef6\\n\\n\u4e5f\u6709\u5f88\u591a\u8f6f\u4ef6\u5177\u6709\u5b9e\u65f6\u7684\u529f\u80fd, \u4f46\u662f\u636e\u89c2\u5bdf, \u4ed6\u4eec\u7684\u5728\u5c4f\u5e55\u4e0a\u53d6\u8bcd\u7684\u529f\u80fd\u4f3c\u4e4e\u90fd\u6ca1\u6709\u76f4\u63a5\u8c03\u7528 [Vision](https://developer.apple.com/documentation/vision/recognizing_text_in_images) framework. \u5bfc\u81f4\u67e5\u8be2\u7f13\u6162, \u53d6\u8bcd\u4e0d\u7cbe\u786e\u7b49\u95ee\u9898"}]}')}}]); \ No newline at end of file diff --git a/assets/js/23fe4fcc.907478e0.js b/assets/js/23fe4fcc.907478e0.js new file mode 100644 index 0000000..44bcccf --- /dev/null +++ b/assets/js/23fe4fcc.907478e0.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdf_doc_source=self.webpackChunkdf_doc_source||[]).push([[728],{2183:n=>{n.exports=JSON.parse('{"blogPosts":[{"id":"tech_detail_macos","metadata":{"permalink":"/valo-reader-doc/blog/tech_detail_macos","editUrl":"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/tech_detail_macos.md","source":"@site/blog/tech_detail_macos.md","title":"Valo Reader for macOS \u6280\u672f\u6982\u8981","description":"\u6458\u8981","date":"2023-10-19T05:59:23.000Z","formattedDate":"2023\u5e7410\u670819\u65e5","tags":[{"label":"technical","permalink":"/valo-reader-doc/blog/tags/technical"}],"readingTime":23.58,"hasTruncateMarker":false,"authors":[{"name":"Ce Wang","title":"Solo dev","url":"https://github.com/haloWang","imageURL":"https://github.com/haloWang.png","key":"halowang"}],"frontMatter":{"slug":"tech_detail_macos","title":"Valo Reader for macOS \u6280\u672f\u6982\u8981","authors":["halowang"],"tags":["technical"]},"nextItem":{"title":"\u5f00\u53d1\u76ee\u7684","permalink":"/valo-reader-doc/blog/motivation"}},"content":"## \u6458\u8981\\n\\n\u672c\u6587\u4e3b\u8981\u4ecb\u7ecd\u4e86 Valo Reader for macOS\uff08\u4e0b\u6587\u7b80\u79f0\u4e3a\u201c\u672c\u7a0b\u5e8f\u201d\uff09\u7684[\u9879\u76ee\u5e03\u5c40](#\u9879\u76ee\u5e03\u5c40)\uff0c[\u5de5\u7a0b\u67b6\u6784](#\u5de5\u7a0b\u67b6\u6784)\uff0c[\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757](#\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757)\u4e0e[\u6280\u672f\u7ec6\u8282](#\u6280\u672f\u7ec6\u8282)\u3002\u4f60\u8fd8\u53ef\u4ee5\u5728[\u672c\u6587\u6863](/docs/intro)\u7684\u5176\u4ed6\u9875\u9762\u4e86\u89e3\u672c\u7a0b\u5e8f\u7684[\u5f00\u53d1\u52a8\u673a](./motivation.md)\uff0c[\u4f7f\u7528\u65b9\u5f0f](/docs/usage)\uff0c[\u9690\u79c1\u4e0e\u5b89\u5168](/docs/privacy_security)\u7b49\u5185\u5bb9\\n\\n## \u4ec0\u4e48\u662f Valo Reader?\\n\\nValo Reader \u8fd9\u4e2a\u9879\u76ee\u662f\u6211\u4e2a\u4eba\u5728\u65e5\u5e38\u751f\u6d3b\u4e2d\u9010\u6e10\u840c\u751f\uff0c\u6f14\u5316\u548c\u5b9e\u65bd\u7684\u60f3\u6cd5\u3002\u662f\u6211\u4e3a\u4e86\u5728\u81ea\u5df1\u7535\u5b50\u8bbe\u5907\u8bbe\u5907\u4e0a\u80fd\u8f7b\u677e\u9605\u8bfb\u82f1\u6587\u5185\u5bb9\uff0c\u5e76\u6e10\u8fd1\u5f0f\u5730\u63d0\u9ad8\u81ea\u5df1\u7684\u82f1\u6587\u6c34\u5e73\uff0c\u800c\u81ea\u884c\u8bbe\u8ba1\u548c\u7814\u53d1\u7684\u4e00\u6b3e\u4ea7\u54c1\uff0c\u5176\u6838\u5fc3\u529f\u80fd\u5c55\u793a\u5982\u4e0b\uff1a\\n![intro](../static/img/intro.gif)\\n\\n\u5f53\u4f60\u5728\u4f7f\u7528\u672c\u7a0b\u5e8f\u65f6\uff0c\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u4f60\u611f\u5230\u751f\u758f\u7684\u82f1\u8bed\u5355\u8bcd\u4e0a\u3002\u6309\u4e0b\u6307\u5b9a\u7684\u6309\u952e\uff08\u9ed8\u8ba4\u4e3a Fn\uff09\uff0c\u672c\u7a0b\u5e8f\u5373\u4f1a\u5728\u54cd\u5e94\u7684\u4f4d\u7f6e\u5c55\u793a\u5bf9\u5e94\u7684\u91ca\u4e49\uff0c\u540c\u65f6\u4f7f\u7528\u8bed\u97f3\u5408\u6210\u6765\u53d1\u97f3\u3002\\n\\n\u76ee\u524d\uff0c\u4f60\u53ef\u4ee5\u901a\u8fc7 [Mac App Store](https://apps.apple.com/cn/app/valo-reader/id6448040931) \u6216[\u5176\u4ed6\u65b9\u5f0f](/docs/installation)\u6765\u83b7\u53d6\u672c\u7a0b\u5e8f\\n\\n## \u9879\u76ee\u5e03\u5c40\\n\\n\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u9879\u76ee\u7684\u4e3b\u4f53\u7ed3\u6784\uff1a\\n![project_layout](./../static/img/project_layout.png)\\n\\n\u672c\u6587\u7684\u4f4d\u4e8e\u4e0a\u56fe\u4e2d\u7684\u6d45\u9ec4\u8272\u90e8\u5206\uff0c\u7531 [docusaurus](https://docusaurus.io/) \u751f\u6210\\n\\n\u672c\u9879\u76ee\u7684\u60f3\u6cd5\u6700\u65e9\u5b9e\u73b0\u4e8e\u5728\u6d4f\u89c8\u5668\u4e2d\u8fd0\u884c\u7684 js \u811a\u672c\uff0c\u4f60\u53ef\u4ee5\u5728 github \u4e0a\u770b\u5230\u672c\u9879\u76ee\u5728\u6d4f\u89c8\u5668\u4e0a\u7684\u65e9\u671f\u5b9e\u73b0\u53ca\u5bf9\u5e94\u7684[\u529f\u80fd\u5c55\u793a](https://github.com/HaloWang/english_flow#%E5%8A%9F%E8%83%BD%E5%B1%95%E7%A4%BA)\uff0c\u5373\u56fe\u4e2d\u6d45\u7eff\u8272\u90e8\u5206\\n\\n\u800c\u672c\u6587\u4e3b\u8981\u8bf4\u660e\u4e86\u4e0a\u56fe\u4e2d macOS\uff08\u6d45\u84dd\u8272\u90e8\u5206\uff09\u7684\u8fd0\u884c\u65b9\u5f0f\u4e0e\u6280\u672f\u7ec6\u8282\\n\\n## \u5de5\u7a0b\u67b6\u6784\\n\\n\u672c\u7a0b\u5e8f\u7531 `flutter create XXX --platform macos` \u547d\u4ee4\u521b\u5efa\uff0c\u5176\u6e90\u4ee3\u7801\u4e3b\u8981\u5206\u4e3a\u4e24\u90e8\u5206\\n\\n### flutter \u4fa7\\n\\n\u8fd9\u90e8\u5206\u7531 dart \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206\\n\\n#### \u4e3b\u7a0b\u5e8f\\n\\n\u8be5\u90e8\u5206\u7684\u4e3b\u8981\u529f\u80fd\u6709\uff1a\\n\\n1. \u901a\u8fc7 method channel \u5b9e\u73b0 flutter \u4e0e native \u7684\u4ea4\u4e92\\n2. \u6e32\u67d3\u7528\u6237\u8bbe\u7f6e\u9762\u677f\uff08Dashboard\uff09\\n3. \u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\u4e0e\u8bca\u65ad\u68c0\u67e5\u4fe1\u606f\\n4. \u4f7f\u7528\u7b2c\u4e09\u65b9\u4f9d\u8d56\u63d0\u4f9b\u7684\u529f\u80fd\\n\\n#### flutter package: \u6587\u672c\u5757\u8868\u5f81\\n\\n\u8fd9\u90e8\u5206\u662f\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u4ee3\u7801\uff0c\u5728 macOS / iOS / Android \u4e09\u7aef\u5171\u4eab\u90e8\u5206\u903b\u8f91\u4e0e UI\uff0c\u5176\u4e3b\u8981\u529f\u80fd\u7531\u6709\uff1a\\n\\n1. \u89e3\u6790\u7528\u6237\u8bbe\u5907\u5c4f\u5e55\u4e0a\u8bc6\u522b\u5230\u7684\u6587\u672c\u5757\uff0c\u7ed3\u6784\u5316\u6587\u672c\u5757\uff0c\u5e76\u5c06\u5176\u7ef4\u62a4\u5230\u672c\u7a0b\u5e8f\u7684\u5185\u5b58\u4e2d\\n2. \u58f0\u660e\u5e76\u5b9e\u73b0\u7528\u6237\u89c6\u89c9\u7126\u70b9\u4e0e\u5df2\u77e5\u7684\u6587\u672c\u4fe1\u606f\u7684\u4ea4\u4e92\\n3. \u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\\n\\n#### flutter package: \u57fa\u7840\u5e93\\n\\n\u5728\u6211\u7684\u6240\u6709 flutter \u9879\u76ee\u4e2d\u5171\u4eab\u7684\u57fa\u7840\u4f9d\u8d56\\n\\n### native \u4fa7\\n\\n\u8fd9\u90e8\u5206\u7531 swift \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206\\n\\n#### \u4e3b\u7a0b\u5e8f\u90e8\u5206\\n\\n\u6301\u6709 flutter engine\uff0c\u901a\u8fc7 method channel \u6253\u901a flutter \u4e0e `Base`\u3001`OCR` \u4ee5\u53ca\u5e94\u7528\u7a0b\u5e8f\u672c\u8eab\u7684\u53cc\u5411\u8c03\u7528\\n\\n\u521b\u5efa\u9879\u76ee\u65f6\u9ed8\u8ba4\u7684 `FlutterViewController` \u5df2\u88ab\u5220\u9664\\n\\n#### CocoaPods dependency: Cocoa \u529f\u80fd\u5e93\\n\\n\u4e0e Cocoa framework \u4ea4\u4e92\uff0c\u5982\uff1a\u6743\u9650\u7533\u8bf7\uff0c\u952e\u76d8\u76d1\u542c\uff0c\u9f20\u6807\u76d1\u542c\uff0c\u622a\u53d6\u5c4f\u5e55\uff0c\u79fb\u52a8\u627f\u8f7d flutter \u7684 `NSWindow`\u4ee5\u53ca\u63d0\u4f9b\u57fa\u672c\u82e5\u5e72\u539f\u751f\u80fd\u529b\\n\\n\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u6211\u4e5f\u4f1a\u5728\u8be5\u4f9d\u8d56\u4e2d\u5b66\u4e60\uff0c\u5c1d\u8bd5\u548c\u5b9e\u73b0 Cocoa/AppKit \u72ec\u6709\u7684 API \u4e0e\u529f\u80fd\\n\\n#### CocoaPods dependency: OCR \u529f\u80fd\u5e93\\n\\n\u4e0e [Apple Vision framework - Text Recognizing](https://developer.apple.com/documentation/vision/recognizing_text_in_images) \u4ea4\u4e92\uff0c\u5e76\u5b9e\u73b0\u90e8\u5206\u903b\u8f91\uff0c\u5982\uff1a\u53d1\u8d77\u6587\u672c\u8bc6\u522b\u8bf7\u6c42\uff0c\u7ef4\u62a4\u6587\u672c\u8bc6\u522b\u54cd\u5e94\u7f13\u5b58\\n\\n\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u5728 macOS \u548c iOS \u9879\u76ee\u540c\u65f6\u4f9d\u8d56\u5e76\u5171\u4eab\u4ee3\u7801\\n\\n## \u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757\\n\\n\u5728\u8fd0\u884c\u65f6\uff0c\u672c\u7a0b\u5e8f\u4e3b\u8981\u53ef\u5206\u4e3a\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\u3001\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD) \u548c\u539f\u751f\u4fa7 (native) \u4e09\u4e2a\u6a21\u5757\\n\\n\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u7684\u4e3b\u8981\u6a21\u5757\u53ca\u5176\u901a\u8baf\uff1a\\n![modules_light](../static/img/modules_light.png#gh-light-mode-only)\\n![modules_dark](../static/img/modules_dark.png#gh-dark-mode-only)\\n\\n### \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\\n\\nHUD \u662f\u4e00\u4e2a\u7684 `NSWindow` \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a `FlutterViewController`\uff0c\u5e76\u7ed1\u5b9a\u4e86\u81ea\u5df1\u7684 `FlutterEngine` \u548c `FlutterMethodChannel`\\n\\nHUD \u7684\u4e3b\u8981\u4efb\u52a1\u5305\u542b\uff1a\\n\\n- \u901a\u8fc7 `FlutterMethodChannel` \u5411 native \u6d3e\u53d1 ocr \u8bf7\u6c42\\n- \u89e3\u6790 native \u5bf9\u5c4f\u5e55\u6307\u5b9a\u533a\u57df\u7684 ocr \u7ed3\u679c\uff0c\u5e76\u5728\u5185\u5b58\u4e2d\u7ef4\u62a4\u8be5\u7ed3\u679c\u4f9b\u540e\u7ee7\u7a0b\u5e8f\u903b\u8f91\u4f7f\u7528\\n- \u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u4ee5\u5728\u9700\u8981\u65f6\u8ba1\u7b97\u91ca\u4e49\u5c55\u793a\u7684\u4f4d\u7f6e\\n- \u76d1\u542c\u952e\u76d8\u6309\u952e\u72b6\u6001\uff0c\u5728\u53d8\u5316\u65f6\u6d3e\u53d1 ocr \u8bf7\u6c42\u5e76\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\\n- \u5b9e\u65f6\u540c\u6b65 Dashboard engine \u4f20\u9012\u8fc7\u6765\u7684\u7528\u6237\u8bbe\u7f6e\uff0c\u5e76\u4fee\u6539\u5c55\u793a\u903b\u8f91\\n\\n\u4e0b\u56fe\u5c55\u793a\u4e86\u8be5\u6a21\u5757\u6240\u7ef4\u62a4\u7684\u6570\u636e\u6620\u5c04\u5230\u5c4f\u5e55\u4e0a\u65f6\u7684\u53ef\u89c6\u5316\u6548\u679c\uff1a\\n![text_block_representation](../static/img/text_block_representation.png)\\n\\n#### \u5728 HUD \u4e2d\u7684\u72b6\u6001\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 [riverpod](https://riverpod.dev/) \u7ba1\u7406\u7edd\u5927\u90e8\u5206\u7684\u72b6\u6001\u3002\u5728\u6784\u7b51\u672c\u7a0b\u5e8f\u65f6\uff0c\u6211\u7684\u601d\u8003\u8fc7\u7a0b\u4e3b\u8981\u57fa\u4e8e\u72b6\u6001\u4e0e\u72b6\u6001\u53d8\u5316\uff0c\u4ece\u5c55\u793a\u91ca\u4e49\u952e\u70b9\u51fb\u5230\u6e32\u67d3\u91ca\u4e49\u9762\u677f\u52a8\u753b\\n\\n\u5728\u5927\u91cf\u4f7f\u7528 riverpod \u4e2d\u7684 `Provider` / `StateProvider` / `ProviderContainer.listen` \u540e\uff0c\u6211\u53ef\u4ee5\u6784\u5efa\u4e00\u4e2a\u9ad8\u5ea6\u5f02\u6b65\uff0c\u8c03\u7528\u987a\u5e8f\u4e0d\u654f\u611f\uff0c\u72b6\u6001\u53d8\u5316\u9a71\u52a8\u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u5728\u7f16\u7801\u8fc7\u7a0b\u4e2d\uff0c\u8fd9\u79cd\u65b9\u5f0f\u8ba9\u7a0b\u5e8f\u5458\u53ef\u4ee5\u4ece\u4e00\u5927\u4e32\u7684\u547d\u4ee4\u5f0f(Imperative)\u65b9\u6cd5\u8c03\u7528\u4e2d\u89e3\u653e\u51fa\u6765\uff0c\u4ec5\u4ec5\u9700\u8981\u5173\u6ce8\u548c\u786e\u4fdd\u6bcf\u4e2a\u6700\u5c0f\u903b\u8f91\u5355\u5143\u2014\u2014`Provider`\uff0c\u786e\u4fdd\u5176\u5b9e\u73b0\u662f\u6b63\u786e\u7684\u5373\u53ef\uff0c\u8fd9\u51cf\u5c11\u4e86\u6784\u5efa\u590d\u6742\u548c\u957f\u4e32\u903b\u8f91\u51fa\u9519\u7684\u53ef\u80fd\\n\\n\u4e0b\u9762\u7684\u4e24\u5f20\u56fe\u8868\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u7684\u4e3b\u8981\u529f\u80fd\u6d41\u7a0b\\n\\n\u5207\u6362\u5c55\u793a\u91ca\u4e49\u6d41\u7a0b\uff1a\\n\\n```mermaid\\ngraph LR\\n A1[\u76d1\u542c\u952e\u76d8\u72b6\u6001];\\n A[\u952e\u76d8\u72b6\u6001\u53d1\u751f\u6539\u53d8];\\n query_button_down{\u5c55\u793a\u91ca\u4e49\u952e\u6309\u4e0b?};\\n C[\u505c\u6b62\u5b9a\u65f6\u5668];\\n D[\u542f\u52a8\u5b9a\u65f6\u5668\u5e76\u91cd\u7f6e\u6587\u672c\u5757\u72b6\u6001];\\n mouse_state[\u9f20\u6807\u72b6\u6001];\\n screen_state[\u5c4f\u5e55\u72b6\u6001];\\n send_request_to_native[\\"\u5411 native \u53d1\u9001\u622a\u56fe\u4e0e OCR \u8bf7\u6c42\\"];\\n F[native OCR \u7ed3\u679c\u5df2\u83b7\u5f97];\\n G[\u89e3\u6790 OCR \u7ed3\u679c];\\n H1[\u66f4\u65b0\u6587\u672c\u5757\u72b6\u6001];\\n\\n mouse_state--\x3esend_request_to_native\\n screen_state--\x3esend_request_to_native\\n send_request_to_native-.->F;\\n F--\x3eG;\\n G--\x3eH1;\\n A1-.->A;\\n A--\x3equery_button_down;\\n query_button_down--\x3e|false|C;\\n query_button_down--\x3e|true|D;\\n D-.->send_request_to_native;\\n```\\n\\n\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u53d8\u66f4\u6d41\u7a0b\uff1a\\n\\n```mermaid\\ngraph LR;\\n H2[\u6587\u672c\u5757\u72b6\u6001\u53d8\u66f4];\\n most_wanted_state_logic[\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u76d1\u542c\u903b\u8f91];\\n most_wanted_state_changed[\u72b6\u6001\u6539\u53d8];\\n mouse_change0[\u76d1\u542c\u9f20\u6807\u72b6\u6001];\\n mouse_change[\u9f20\u6807\u72b6\u6001\u6539\u53d8];\\n sync_mouse_to_flutter[\u540c\u6b65\u9f20\u6807\u72b6\u6001\u81f3 flutter];\\n sync_current_screen[\u540c\u6b65\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u6570\u636e];\\n sync_current_screen_to_flutter[\u540c\u6b65\u5c4f\u5e55\u6570\u636e\u81f3 flutter];\\n hasTextBlock{\u72b6\u6001\u7684\u503c\u662f\u5426\u4e3a\u7a7a\\\\n\u5373\u9f20\u6807\u5f53\u524d\u4f4d\u7f6e\u6ca1\u6709\u8bc6\u522b\u5230\u6587\u672c};\\n queryDB[\u67e5\u8be2\u6570\u636e\u5e93];\\n dbHasData{\u6570\u636e\u5e93\u5305\u542b\u91ca\u4e49};\\n hideAllFlow[\u9690\u85cf\u6240\u6709\u91ca\u4e49\u9762\u677f];\\n asParam[\u4f5c\u4e3aOCR\u8bf7\u6c42\u7684\u53c2\u6570];\\n screen_changed{\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u53d1\u751f\u53d8\u52a8?}\\n\\n \u5207\u6362\u5c55\u793a\u91ca\u4e49-.->H2\\n mouse_change0-.->mouse_change--\x3esync_mouse_to_flutter;\\n mouse_change --\x3e sync_current_screen --\x3e sync_current_screen_to_flutter;\\n sync_current_screen_to_flutter --\x3e \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001\\n sync_current_screen --\x3e screen_changed --\x3e |true| \u6539\u53d8\u627f\u8f7dHUD\u7684NSPanel\u4f4d\u7f6e;\\n screen_changed --\x3e |false| return_0;\\n H2 --\x3e most_wanted_state_logic;\\n sync_mouse_to_flutter --\x3e most_wanted_state_logic;\\n sync_mouse_to_flutter -.-> asParam;\\n \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001 -.-> asParam;\\n most_wanted_state_logic --\x3e most_wanted_state_changed --\x3e hasTextBlock;\\n \u5c55\u793a\u91ca\u4e49\u952e\u72b6\u6001 --\x3e most_wanted_state_logic;\\n most_wanted_state_changed --\x3e hideAllFlow;\\n hasTextBlock --\x3e |false| return_1;\\n hasTextBlock --\x3e |true| queryDB;\\n queryDB --\x3e dbHasData;\\n dbHasData --\x3e |true| \u6e32\u67d3\u5c55\u793a\u91ca\u4e49\u9762\u677f\u52a8\u753b\\n dbHasData --\x3e |false| return_2;\\n```\\n\\n### \u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\\n\\nDashboard \u662f\u4e00\u4e2a `NSPanel` \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a `FlutterViewController`\uff0c\u5e76\u7ed1\u5b9a\u4e86\u4e0d\u540c\u4e8e HUD \u7684\u72ec\u7acb engine \u4e0e method channel\\n\\nDashboard \u7684\u4e3b\u8981\u4efb\u52a1\u662f\u4e3a\u7528\u6237\u63d0\u4f9b\u63a7\u5236\u672c\u7a0b\u5e8f\u7684 UI\uff0c\u5305\u62ec\u5feb\u6377\u952e\u8bbe\u7f6e\uff0c\u5f00\u673a\u81ea\u542f\uff0c\u5173\u4e8e\u672c\u7a0b\u5e8f\u7b49\u5e38\u89c1\u7528\u6237\u4ea4\u4e92\\n\\n\u540c\u65f6\uff0cDashboard \u4f1a\u7ef4\u62a4\u672c\u7a0b\u5e8f\u5728\u5185\u5b58\u548c\u786c\u76d8\u4e2d\u7684\u72b6\u6001\uff0c\u5e76\u5c06\u8fd9\u4e2a\u72b6\u6001\u901a\u8fc7 method channel \u540c\u6b65\u81f3\u4e0b\u9762\u5373\u5c06\u8981\u8bb2\u5230\u7684\u91ca\u4e49\u5c55\u793a\u9762\u677f\\n\\n### \u539f\u751f\u4fa7 (native)\\n\\n\u539f\u751f\u4fa7\u662f\u7531 `flutter create` \u547d\u4ee4\u521b\u5efa\u51fa\u6765\u7684\u4f20\u7edf Xcode \u5de5\u7a0b\\n\\n\u5728\u539f\u751f\u4fa7\uff0c\u6211\u4e3b\u8981\u5173\u6ce8\u7684\u95ee\u9898\u5982\u4e0b\uff1a\\n\\n- \u5904\u7406\u6765\u81ea `FlutterMethodChannel` \u7684\u8c03\u7528\uff0c\u5e76\u5728\u9700\u8981\u65f6\u901a\u8fc7 `FlutterResult` \u54cd\u5e94\u5bf9\u5e94\u7684 flutter engine\\n- \u7ef4\u62a4\u627f\u8f7d HUD \u548c Dashboard \u7684\u4e24\u4e2a `NSWindow`\uff0c\u5c24\u5176\u662f\u5f53\u7528\u6237\u7684\u8bbe\u5907\u540c\u65f6\u94fe\u63a5\u591a\u4e2a\u5c4f\u5e55\u65f6\uff0c\u6b63\u786e\u5730\u8bbe\u7f6e HUD \u7684\u4f4d\u7f6e\\n- \u5c01\u88c5\u5e76\u5411 flutter \u63d0\u4f9b\u80fd\u529b\uff0c\u5982\uff1a\\n - \u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u53d8\u5316\u548c\u952e\u76d8\u72b6\u6001\u53d8\u5316\uff0c\u5e76\u5c06\u72b6\u6001\u540c\u6b65\u81f3 flutter\\n - \u4e3a flutter \u63d0\u4f9b OCR \u548c\u8bed\u97f3\u5408\u6210\u80fd\u529b\\n - \u5728\u6267\u884c OCR \u65f6\u4f7f\u7528 swift \u5b9e\u73b0\u5fc5\u8981\u7684\u6027\u80fd\u4f18\u5316\\n\\n## \u6280\u672f\u7ec6\u8282\\n\\n\u5f97\u76ca\u4e8e Apple \u4e00\u8109\u76f8\u627f\u7684 API \u8bbe\u8ba1\u4ee5\u53ca\u540c\u6837\u7684\u7f16\u7a0b\u8bed\u8a00\uff0c\u719f\u7ec3\u4e8e iOS \u5e94\u7528\u5f00\u53d1(Cocoa Touch & UIKit)\u7684\u7a0b\u5e8f\u5458\u5728\u9762\u5bf9\u4e0e macOS \u4ea4\u4e92\u7684 Cocoa \u548c AppKit \u6846\u67b6\u65f6\uff0c\u53ef\u4ee5\u590d\u7528\u5f88\u591a\u601d\u60f3\u4e0e\u903b\u8f91\\n\\n\u4f46\u76f8\u6bd4\u4e8e iOS\uff0cmacOS \u7ed9\u4e86\u7528\u6237\u66f4\u5927\u7684\u821e\u53f0\uff0c\u4e5f\u8ba9\u5f00\u53d1\u8005\u9762\u5bf9\u4e86\u66f4\u591a\u7684\u6311\u6218: \u9f20\u6807\uff0c\u952e\u76d8\uff0c\u7a97\u53e3\u548c\u5c4f\u5e55\\n\\n\u5728\u5f00\u53d1\u672c\u7a0b\u5e8f\u7684\u8fc7\u7a0b\u4e2d\uff0c\u6211\u9047\u5230\u4e86\u8bf8\u591a\u7684\u56f0\u96be\uff0c\u4e5f\u5728\u8305\u585e\u987f\u5f00\u65f6\u6536\u83b7\u4e86\u5f88\u591a\u5feb\u4e50\uff0c\u6211\u5728\u8fd9\u4e00\u7ae0\u8282\u4f1a\u8bb0\u5f55\u4e00\u4e0b\\n\\n### \u591a\u5c4f\u5e55\\n\\nmacOS \u53ca\u5176\u8fd0\u884c\u7684\u786c\u4ef6\u8bbe\u5907\u5e38\u5e38\u8fde\u63a5\u7740\u591a\u5757\u5c4f\u5e55\uff0c\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u529f\u80fd\u5c31\u662f\u5728\u4efb\u610f\u7684\u5c4f\u5e55\u4f4d\u7f6e\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\uff0c\u8fd9\u5c31\u8981\u786e\u4fdd\uff1a\\n\\n1. \u5c06\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\u653e\u7f6e\u4e8e\u6b63\u786e\u7684\u4f4d\u7f6e\u4e0a\\n2. flutter \u5728\u8bf7\u6c42 native \u6355\u6349\u5c4f\u5e55\u65f6\uff0c\u6355\u6349\u7684\u77e9\u5f62\u6846\u7684\u4f4d\u7f6e\u6b63\u786e\\n3. flutter \u5728\u89e3\u6790 ocr response \u540e\uff0c\u7ed3\u679c\u53ef\u4ee5\u6b63\u786e\u5730\u548c\u5c4f\u5e55\u4e0a\u771f\u6b63\u7684\u5185\u5bb9\u5efa\u7acb\u6620\u5c04\\n\\n\u548c UIKit \u4e2d\u5e38\u7528\u7684 CGRect \u4e0d\u540c\uff0cAppKit \u5bf9\u8bbe\u5907\u5c4f\u5e55\u7684\u62bd\u8c61 `NSScreen`\uff0c\u5176\u5750\u6807\u4ee5 NSRect \u8ba1\u7b97\uff0c\u5750\u6807\u7cfb\u539f\u70b9\u4e3a macOS \u539f\u59cb\u5c4f\u5e55\u7684\u5de6\u4e0b\u89d2\u3002\u800c\u6b64\u65f6\uff0c\u5982\u679c\u4f60\u7ed9\u4f60\u7684\u8bbe\u5907\u8fde\u63a5\u4e0a\u4e86\u5176\u4ed6\u7684\u5c4f\u5e55\uff0c`NSScreen.screens` \u6240\u5448\u73b0\u5c4f\u5e55\u5e03\u5c40\uff0c\u53ef\u80fd\u5c31\u4f1a\u53d8\u6210\u4e0b\u56fe\u6240\u793a\u7684\u6837\u5b50\uff1a\\n![appkit_screens_coordinate](../static/img/appkit_screens_coordinate.png)\\n\\n`screen 1` \u7684\u5de6\u4e0b\u89d2\u5750\u6807\u503c\u5e76\u975e\u662f (0, 0)\uff0c\u800c\u662f `screen 1` \u548c `screen 0` \u7684 (0, 0) \u70b9\u7684\u76f8\u5bf9\u4f4d\u7f6e (w1, d2)\u3002\u901a\u8fc7\u76d1\u542c\u9f20\u6807\u79fb\u52a8\u4e8b\u4ef6\u83b7\u53d6\u7684\u9f20\u6807\u4f4d\u7f6e `event.locationInWindow`\uff0c\u4e5f\u662f\u76f8\u5bf9\u4e8e\u539f\u70b9\u7684 `NSPoint`\\n\\n\u5728\u5f00\u53d1\u65f6\uff0c\u4ec5\u4ec5\u5c06\u81ea\u5df1\u7684\u601d\u7ef4\u4ece iOS \u7684 CGRect \u5750\u6807\u7cfb\u8f6c\u5316\u81f3 NSRect \u5750\u6807\u7cfb\u8fd8\u7b97\u7b80\u5355\u3002\u4f46\u5728\u540e\u7ee7\u7684\u903b\u8f91\u4e2d\uff0c\u56e0\u4e3a OCR \u7ed3\u679c\u7684 `VNRectangleObservation.boundingBox` \u53c8\u4f1a\u56de\u5230 CGRect \u5750\u6807\u7cfb\uff0c\u5750\u6807\u7cfb\u7684\u9891\u7e41\u8f6c\u5316\u786e\u5b9e\u4f1a\u7ed9\u4eba\u5e26\u6765\u4e00\u5b9a\u7684\u56f0\u6270\\n\\n### \u5c4f\u5e55\u6355\u6349\\n\\n\u60f3\u6355\u6349 macOS \u7684\u5c4f\u5e55\uff0c\u4f60\u53ef\u4ee5\u8c03\u7528 `CGDisplayCreateImage` \u51fd\u6570\\n\\n\u4f46\u503c\u5f97\u6ce8\u610f\u7684\u662f\uff0c\u60f3\u8981\u622a\u53d6\u5c4f\u5e55\u4e0a\u5176\u4ed6\u8fdb\u7a0b\u7684\u5185\u5bb9(\u6bd4\u5982 IDE \u6216\u8005\u6d4f\u89c8\u5668)\uff0c\u9700\u8981\u9884\u5148\u901a\u8fc7 `CGRequestScreenCaptureAccess`\uff0c\u7533\u8bf7\u5230\u622a\u53d6\u5176\u4ed6\u8fdb\u7a0b UI \u7684\u6743\u9650\uff0c\u5426\u5219 `CGDisplayCreateImage` \u53ea\u80fd\u62ff\u5230 macOS \u7684\u684c\u9762\uff08\u4ee5\u53ca\u7a0b\u5e8f\u672c\u8eab\uff09\\n\\n### OCR\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 Apple \u4e3a\u5f00\u53d1\u8005\u63d0\u4f9b\u7684 [Vision - Recognizing Text](https://developer.apple.com/documentation/vision/recognizing_text_in_images) \u8fdb\u884c\u5c4f\u5e55\u6587\u672c\u7684\u63d0\u53d6\\n\\nOCR \u8bf7\u6c42\u7684\u8c03\u7528\u7531 flutter side \u5524\u8d77\uff1a\\n\\n```dart\\nvoid dispatchOCRRequestToNative() async {\\n final x = 100;\\n final y = 100;\\n final width = 300;\\n final height = 80;\\n final dimensions = [x, y, width, height];\\n final result = await methodChannel.invokeMethod(\\"captureAndOCR\\", dimensions);\\n\\n // parse the result from native\\n // ...\\n}\\n```\\n\\nNative side \u5728\u63a5\u6536\u5230\u8bf7\u6c42\u540e\uff0c\u4f1a\u6267\u884c\u622a\u5c4f\u548c\u8bc6\u522b\u64cd\u4f5c\uff1a\\n\\n```swift\\nimport Vision\\n// ...\\nlet result : FlutterResult = ...\\nlet image = captureScreen()\\nlet handler = VNImageRequestHandler(cgImage: image)\\nlet request = VNRecognizeTextRequest { request, error in\\n // ocr finished\\n let parsedResult = parse(request.results)\\n result(parsedResult)\\n}\\nhandler.perform([request])\\n```\\n\\n#### \u6027\u80fd\u8868\u73b0\\n\\n\u5728 macbook 2021 \u7684 m1 pro \u4e0a\uff0c\u4ee5 320\u271575 \u7684\u8bbe\u8ba1\u5206\u8fa8\u7387\u622a\u56fe(\u5b9e\u9645\u5206\u8fa8\u7387\u4e3a 640\u2715150)\uff0c\u6bcf\u79d2 11 \u5e27\u7684\u60c5\u51b5\u4e0b\u8fdb\u884c\u957f\u65f6\u95f4\u7684\u622a\u56fe\u548c OCR \u64cd\u4f5c\uff0c\u6574\u4e2a\u6d41\u7a0b\u7684\u5ef6\u8fdf\u5e73\u5747\u7ea6\u4e3a 65ms\uff0c\u6211\u8ba4\u4e3a\u8fd8\u7b97\u662f\u4e00\u4e2a\u53ef\u63a5\u53d7\u7684\u72b6\u6001\\n\\n#### \u8282\u80fd\u4f18\u5316\\n\\n\u5373\u4fbf\u662f\u6027\u80fd\u5141\u8bb8\uff0c\u4f18\u5316\u4e5f\u662f\u5e94\u8be5\u505a\u7684\uff0c\u7528\u6237\u4e00\u65e6\u4f7f\u7528\u4f60\u7684 App\uff0c\u5c31\u611f\u89c9 macbook \u7684 C \u9762\u53d1\u70ed\uff0c\u8fd9\u662f\u65e0\u6cd5\u5bb9\u5fcd\u7684\\n\\n\u5f53\u524d\u5728 OCR \u6d41\u7a0b\u4e2d\u4e3b\u8981\u7684\u6027\u80fd\u4f18\u5316\u6b65\u9aa4\u662f\u5728 native \u7ef4\u62a4\u4e00\u4e2a\u5148\u8fdb\u5148\u51fa\uff0c\u6700\u5927\u5bb9\u91cf\u4e3a 40 \u5e27\u7684\u5b57\u5178\uff0c\u4ee5\u56fe\u7247\u6570\u636e\u4e3a key \u7f13\u5b58 OCR \u7684\u7ed3\u679c\uff0c\u4e0b\u9762\u662f\u7b80\u5316\u540e\u7684\u4ee3\u7801\uff1a\\n\\n```swift\\nlet cacheManager = CacheManager()\\n// ...\\nlet image = captureScreen()\\nlet key = image.dataProvider.data\\nif let cachedResult = cacheManager[key] {\\n delegate.onOCRResult(cachedResult)\\n return\\n}\\n// ...\\ndispatchOCRRequestToDeviceGPU(image) { result\\n cacheManager[key] = result\\n // ... other logic\\n}\\n// ...\\n```\\n\\n\u5f53\u9f20\u6807\u6307\u9488\u4f4d\u7f6e\u4e0d\u53d8\uff0c\u622a\u56fe\u83b7\u53d6\u7684\u56fe\u7247\u4e0d\u53d8\u65f6\uff0cnative \u5728\u5904\u7406 OCR \u8bf7\u6c42\u65f6\u4f1a\u5148\u547d\u4e2d\u7f13\u5b58\uff0c\u5e76\u76f4\u63a5\u8fd4\u56de\u7ed3\u679c\uff0c\u4ee5\u51cf\u5c11\u975e\u5fc5\u9700\u7684 GPU \u8c03\u7528\\n\\n\u5728\u7cfb\u7edf\u81ea\u5e26\u7684\u6d3b\u52a8\u76d1\u89c6\u5668\u4e2d\u67e5\u770b\u8fdb\u7a0b\u3002\u53d1\u73b0\uff0c\u5728\u5e94\u7528\u7684\u7f13\u5b58\u7b56\u7565\u540e\uff0c\u5728\u672c\u7a0b\u5e8f\u6d3b\u52a8\u65f6\uff0c\u5176 CPU/GPU \u5360\u7528\u7387\u786e\u5b9e\u964d\u4f4e\u4e86\u5f88\u591a\uff0c\u540c\u65f6\uff0cOCR \u7684\u7f13\u5b58\u7ed3\u679c\u88ab\u964d\u4e3a\u4e86 5ms\\n\\n:::note\\n\u6211\u611f\u89c9\uff0c\u4ee5 `CFData` \u4f5c\u4e3a key \u67e5\u8be2\u5b57\u5178\u8fd8\u4e0d\u662f\u6548\u7387\u6700\u9ad8\u7684\u7b97\u6cd5\uff0c\u5e94\u8be5\u53ef\u4ee5\u7ee7\u7eed\u63a2\u7d22\u4e00\u4e0b\\n:::\\n\\n#### \u4e0d\u622a\u53d6\u81ea\u5df1\\n\\n\u5728\u6211\u8fdb\u884c\u5f00\u53d1\u65f6\uff0c\u53d1\u73b0\u5f53\u672c\u7a0b\u5e8f\u5728\u5c55\u793a\u5355\u8bcd\u91ca\u4e49 UI (HUD)\u65f6\uff0c\u56e0\u4e3a HUD \u672c\u8eab\u4e5f\u4f1a\u5728\u4e00\u5b9a\u8303\u56f4\u5185\u88ab\u622a\u5c4f\u51fd\u6570\u6355\u83b7\uff0c\u5bfc\u81f4 HUD \u4f1a\u5f71\u54cd OCR \u7684\u7ed3\u679c\uff0c\u800c\u5728 OCR \u7ed3\u679c\u53d8\u52a8\u540e\uff0cHUD \u53c8\u4f1a\u968f\u4e4b\u53d1\u751f\u53d8\u52a8\uff0c\u518d\u6b21\u5f71\u54cd OCR \u7ed3\u679c\uff0c\u5c31\u8fd9\u6837\u5faa\u73af\u5f80\u590d\uff0c\u8fde\u7f13\u5b58\u4e5f\u90fd\u5931\u6548\u4e86\u3002\u5728\u67e5\u9605\u548c\u5c1d\u8bd5\u5927\u91cf\u7684 API \u540e\uff0c\u6211\u7ec8\u4e8e\u5728 `NSWindow` \u4e2d\u627e\u5230\u4e86 `sharingType` \u8fd9\u4e2a\u5c5e\u6027\uff0c\u5c4f\u5e55\u6355\u6349\u65b9\u6cd5\u6355\u83b7 HUD \u81ea\u8eab\uff0c\u603b\u7b97\u662f\u6253\u7834\u4e86\u8fd9\u4e2a\u94fe\u6761\\n\\n### \u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d\\n\\n\u5bf9 Vision - VNRecognizeTextRequest \u7684\u7ed3\u679c\u89e3\u6790\u662f\u4e2a\u590d\u6742\u7684\u4efb\u52a1\\n\\n\u6267\u884c\u8fd9\u4e2a\u590d\u6742\u4efb\u52a1\u7684\u539f\u56e0\u6709\u4e09\uff1a\\n\\n1. \u672c\u7a0b\u5e8f\u8bbe\u8ba1\u7684\u4ea4\u4e92\u662f\u201c\u7528\u6237\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u611f\u5174\u8da3\u7684\u5355\u8bcd\u4e0a\u65f6\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u201d\uff0c\u8fd9\u5c31\u8981\u6c42\u6211\u4eec\u77e5\u9053\u5c4f\u5e55\u4e0a\u6bcf\u4e2a\u5355\u8bcd\u7684\u5177\u4f53\u4f4d\u7f6e\u548c\u5185\u5bb9\\n2. \u672c\u7a0b\u5e8f\u9488\u5bf9\u6e90\u4ee3\u7801\u7279\u5316\uff0c\u5728\u9762\u5bf9[\u9a7c\u5cf0\u547d\u540d\u6cd5](https://zh.wikipedia.org/zh-cn/%E9%A9%BC%E5%B3%B0%E5%BC%8F%E5%A4%A7%E5%B0%8F%E5%86%99)\u65f6\uff0c\u9700\u8981\u77e5\u9053\u6784\u6210\u4e00\u4e2a symbol \u7684\u6bcf\u4e2a\u5355\u8bcd\u7684\u610f\u601d\uff1a`ThisIsAVeryVeryLongClassName` -> `This`,~~`Is`~~,~~`A`~~,`Very`,`Very`,`Long`,`Class`,`Name` (\u56e0\u8fc7\u4e8e\u7b80\u5355\uff0c\u4e22\u5f03\u957f\u5ea6\u5c0f\u4e8e 3 \u7684\u82f1\u8bed\u5355\u8bcd)\\n3. \u6211\u4eec\u8981\u4ee5\u5355\u8bcd\u7684\u6587\u672c\u5185\u5bb9\u4e3a key\uff0c\u67e5\u8be2\u6570\u636e\u5e93\u3002\u5728\u6587\u672c\u5e8f\u5217\u5305\u542b\u7279\u6b8a\u5b57\u7b26\u65f6\uff0c\u6211\u4eec\u662f\u65e0\u6cd5\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u5230\u5355\u8bcd\u91ca\u4e49\u7684\u3002\u6240\u4ee5\u8981\u79fb\u9664\u975e\u5b57\u6bcd\u5b57\u7b26\u3002\u540c\u65f6\uff0c\u8fd9\u4e00\u64cd\u4f5c\u4e5f\u53ef\u4ee5\u81ea\u7136\u800c\u7136\u5730\u9002\u914d\u7f16\u7a0b\u8bed\u8a00\u4e2d\u7684\u5176\u4ed6\u547d\u540d\u65b9\u5f0f\uff0c\u6bd4\u5982\u79fb\u9664\u4e0b\u5212\u7ebf\u5c31\u53ef\u4ee5\u9002\u5e94[\u86c7\u5f62\u547d\u540d\u6cd5](https://zh.wikipedia.org/zh-cn/%E8%9B%87%E5%BD%A2%E5%91%BD%E5%90%8D%E6%B3%95)\\n\\n\u5047\u8bbe\u6211\u4eec\u5728\u5bf9\u4e0b\u9762\u7684\u56fe\u7247\u6267\u884c OCR \u8bf7\u6c42\\n\\n![ocr_text_source](ocr_text_source.png)\\n\\n\u5728 Native \u7aef\u53d6 `VNRecognizeTextRequest.results.topCandidates(1)` \u540e\uff0c\u7ed3\u679c\u5982\u4e0b\\n\\n```swift\\n[\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Vision provides its text-recognition capabilities through VNRecognizeText\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Request, an image-based request type that finds and extracts text in images. The\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"following example shows how to use VNImageRequestHandler to perform a\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"VNRecognizeTextRequest for recognizing text in the specified CGImage.\\",\\n]\\n```\\n\\n\u5728 Native \u7aef\u4ee5**\u7a7a\u683c**\u5b57\u7b26\u548c `VNRecognizedText.boundingBox(for:)` \u5bf9 OCR \u7ed3\u679c\u8fdb\u884c\u7b2c\u4e00\u6b21\u6574\u7406\uff0c\u4e0a\u8ff0\u7684\u7ed3\u679c\u4f1a\u53d8\u4e3a\u8fd9\u6837\uff1a\\n\\n```swift\\n[\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Vision\\", \\"provides\\", \\"its\\", \\"text-recognition\\", \\"capabilities\\", \\"through\\", \\"VNRecognizeText\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Request,\\", \\"an\\", \\"image-based\\", \\"request\\", \\"type\\", \\"that\\", \\"finds\\", \\"and\\", \\"extracts\\", \\"text\\", \\"in\\", \\"images.\\", \\"The\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"following\\", \\"example\\", \\"shows\\", \\"how\\", \\"to\\", \\"use\\", \\"VNImageRequestHandler\\", \\"to\\", \\"perform\\", \\"a\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"VNRecognizeTextRequest\\", \\"for\\", \\"recognizing\\", \\"text\\", \\"in\\", \\"the\\", \\"specified\\", \\"CGImage.\\",\\n]\\n```\\n\\n\u6700\u540e\u5c06 boundingBox \u7684\u8fd4\u56de\u503c\u548c\u622a\u5c4f\u5c3a\u5bf8\u8fdb\u884c\u4e58\u7b97\uff0c\u8fd4\u56de\u7ed9 flutter \u7aef\u3002Flutter \u7aef\u5728\u62ff\u5230\u7ed3\u679c\u540e\u4e3b\u8981\u8fdb\u884c\u4e24\u4e2a\u6b65\u9aa4\uff1a\\n\\n1. \u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u79fb\u9664\u975e\u82f1\u6587\u5b57\u7b26\uff0c\u8ba1\u7b97\u65b0\u7684 rect\\n2. \u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u5bf9\u5305\u542b\u5927\u5199\u5b57\u6bcd\u7684\u5b57\u7b26\u4e32\u8fdb\u884c\u5206\u5272\uff0c\u8ba1\u7b97\u65b0\u7684 rect\\n\\n\u8fd9\u6837\uff0c\u6587\u672c\u5757\u7684\u5185\u5bb9\u5728 flutter \u7aef\u5c31\u4f1a\u53d8\u6210\u8fd9\u6837\uff1a\\n\\n```dart\\n[\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Vision\\", \\"provides\\", \\"its\\", \\"text\\", \\"recognition\\", \\"capabilities\\", \\"through\\", \\"Recognize\\", \\"Text\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Request\\",\\"image\\", \\"based\\", \\"request\\", \\"type\\", \\"that\\", \\"finds\\", \\"and\\", \\"extracts\\", \\"text\\", \\"images\\", \\"The\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"following\\", \\"example\\", \\"shows\\", \\"how\\", \\"use\\", \\"Image\\", \\"Request\\", \\"Handler\\", \\"perform\\",\\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\\n \\"Recognize\\", \\"Text\\", \\"Request\\", \\"for\\", \\"recognizing\\", \\"text\\", \\"the\\", \\"specified\\", \\"Image\\",\\n]\\n```\\n\\n\u6700\u540e\uff0cflutter \u4fa7\u4f1a\u4ee5\u4e0b\u9762\u7684\u4ee3\u7801\u5bf9\u8fd9\u4e9b OCR \u4fe1\u606f\u8fdb\u884c\u62bd\u8c61\uff1a\\n\\n```dart\\n/// The representation of block\\n///\\n/// \u6587\u672c\u5757\u8868\u5f81\\nclass Block {\\n String text;\\n double x;\\n double y;\\n double w;\\n double h;\\n}\\n\\n// ...\\n\\n/// All text representations in memory\\n///\\n/// \u6240\u6709\u6587\u672c\u5757\u8868\u5f81\\nfinal blocks = StateProvider>((_) => []);\\n```\\n\\n\u4e0a\u9762\u7684\u6570\u636e\u5728\u5c4f\u5e55\u4e0a\u7684\u53ef\u89c6\u5316\u6548\u679c\u5982\u4e0b\uff1a\\n![text_block_representation_2](../static/img/text_block_representation_1.png)\\n\\n\u5176\u4e2d\u5916\u56f4\u7eff\u6846\u4ee3\u8868\u622a\u5c4f\u8303\u56f4\uff0c\u5185\u90e8\u7684\u5c0f\u7eff\u6846\u4ee3\u8868 native \u7aef\u8fd4\u56de\u7684\u7ed3\u679c\uff0c\u5c0f\u7eff\u6846\u5185\u90e8\u7684\u84dd\u8272\u6846\u4ee3\u8868 flutter side \u7b2c\u4e8c\u6b21\u89e3\u6790\u7ed3\u679c\u3002\u53ef\u4ee5\u770b\u5230\uff0c`\\"VNImageRequestHandler\\"` \u8fd9\u4e2a\u5b57\u7b26\u4e32\u88ab\u5206\u5272\u4e3a\u4e86 `\\"Image\\"`, `\\"Request\\"`, `\\"Handler\\"` \u8fd9\u4e09\u4e2a\u6587\u672c\u5757\u3002\u8fd9\u6837\uff0c\u6211\u5c31\u53ef\u4ee5\u77e5\u9053\u7528\u6237\u5230\u5e95\u5bf9\u54ea\u4e00\u6bb5\u5b57\u6bcd\u5e8f\u5217\uff08\u5355\u8bcd\uff09\u611f\u5174\u8da3\u4e86\uff08\u5047\u8bbe\u6e90\u4ee3\u7801\u7b26\u53f7\u7684\u547d\u540d\u662f\u89c4\u8303\u7684 \ud83d\ude07\uff09\u3002\\n\\n:::note\\n\u4f60\u4e5f\u53ef\u4ee5\u6253\u5f00\u672c\u7a0b\u5e8f\u7684\u7528\u6237\u8bbe\u7f6e\u9762\u677f\u6765\u76f4\u63a5\u89c2\u5bdf\u5176\u8fd0\u884c\u65f6\u8868\u73b0\uff1a\\n![dashboard_inspect](./../static/img/dashboard_inspect.png)\\n:::\\n\\n### \u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65\\n\\n\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u4f1a\u521b\u5efa\u4e24\u4e2a flutter engine\uff0c\u5206\u522b\u7528\u4e8e\u6e32\u67d3\\"\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\\"\u548c\\"\u6b22\u8fce\u4e0e\u8bbe\u7f6e\u9875\u9762(Dashboard)\\"\\n\\n\u8fd9\u4e24\u4e2a\u5f15\u64ce\u5747\u901a\u8fc7\u57fa\u672c\u65b9\u5f0f\u521b\u5efa\uff0c\u5747\u91c7\u7528 `FlutterMethodChannel` \u4e0e native \u8fdb\u884c\u901a\u8baf\\n\\n\u53cc\u5f15\u64ce\u5171\u4eab\u540c\u4e00\u4efd\u4ee3\u7801\uff0c\u7ecf\u7531\u4e0d\u540c\u7684 [dart \u5165\u53e3\u51fd\u6570](https://github.com/dart-lang/sdk/blob/master/runtime/docs/compiler/aot/entry_point_pragma.md)\u542f\u52a8\uff1a\\n\\n```swift\\n// \u8bbe\u7f6e\u9875\u9762 (Dashboard)\\ndashboardEngine = FlutterEngine(name: \\"dashboard\\", project: nil)\\ndashboardChannel = FlutterMethodChannel(name: \\"dashboard\\", binaryMessenger: dashboardEngine.binaryMessenger)\\ndashboardEngine.run(withEntrypoint: \\"_dashboard\\")\\n//...\\n\\n// \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\\nhudEngine = FlutterEngine(name: \\"hud\\", project: nil)\\nhudChannel = FlutterMethodChannel(name: \\"hud\\", binaryMessenger: hudEngine.binaryMessenger)\\nhudChannel.run(withEntrypoint: \\"_hud\\")\\n```\\n\\n:::note\\n\u6211\u4e2a\u4eba\u6ca1\u6709\u4f7f\u7528\u5b98\u65b9\u63d0\u4f9b\u7684 [FlutterEngineGroup \u65b9\u6848](https://docs.flutter.dev/add-to-app/multiple-flutters)\uff0c\u56e0\u4e3a\u5728\u5f53\u521d\u8fdb\u884c\u5c1d\u8bd5\u65f6\u53d1\u73b0\u4e86\u5f15\u64ce\u65e0\u54cd\u5e94\u7684\u95ee\u9898\uff0c\u81f3\u4eca\u4ecd\u7136\u662f P2 \u7ea7\u522b\u7684 open [issue](https://github.com/flutter/flutter/issues/119403)\\n:::\\n\\n#### \u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 [riverpod](https://riverpod.dev/) \u6765\u7ba1\u7406\u7edd\u5927\u90e8\u5206\u72b6\u6001\uff0c\u6240\u4ee5\u5728\u540c\u6b65\u72b6\u6001\u65f6\uff0c\u6211\u4e5f\u671f\u671b\u81ea\u5df1\u7684\u81ea\u5df1\u7684\u5fc3\u667a\u6a21\u578b\u53ef\u4ee5\u66f4\u8d34\u8fd1 riverpod\\n\\n\u5728\u8fd0\u884c\u65f6\uff0cDashboard engine \u9700\u8981\u5c06\u7528\u6237\u8bbe\u7f6e\u540c\u6b65\u81f3 HUD engine (\u6bd4\u5982\u66f4\u6539\u67e5\u8bcd\u5feb\u6377\u952e)\uff0c\u6211\u5728\u8fd9\u91cc\u76d1\u542c\u4e86 Dashboard engine \u5bf9\u5e94\u7684 `StateProvider` \u7684\u53d8\u66f4\uff0c\u5e76\u901a\u8fc7 method channel\uff0c\u7ecf\u7531 native \u8f6c\u53d1\u81f3 HUD engine\uff0cHUD engine \u5728\u6536\u5230\u4e86\u901a\u77e5\u540e\uff0c\u518d\u66f4\u65b0\u81ea\u5df1\u7ef4\u62a4\u7684 `StateProvider`\u3002\u4ece\u800c\u5b9e\u73b0\u4e86\u8de8\u5f15\u64ce\u7684\u72b6\u6001\u540c\u6b65\\n\\n### \u91ca\u4e49\u67e5\u8be2\\n\\n\u672c\u7a0b\u5e8f\u5728\u67e5\u8be2\u5355\u8bcd\u91ca\u4e49\u65f6\u4f7f\u7528\u672c\u5730 sqlite \u6570\u636e\u5e93\uff0c\u6570\u636e\u5e93\u6e90\u4e8e [ecdict-ultimate](https://github.com/skywind3000/ECDICT-ultimate)\\n\\n\u9274\u4e8e\u5f00\u6e90\u6570\u636e\u5e93\u8fc7\u4e8e\u5de8\u5927\uff0c\u4e14\u5b58\u5728\u4e00\u4e9b\u672c\u7a0b\u5e8f\u65e0\u9700\u4f7f\u7528\u7684 columns\uff0c\u6211\u53c8\u4f7f\u7528 dart \u548c [drift](https://pub.dev/packages/drift) \u5bf9\u5176\u8fdb\u884c\u4e86\u6e05\u6d17\u548c\u526a\u88c1\\n\\n\u5728\u67e5\u8be2\u65f6\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5c06\u67e5\u8be2\u7684\u672c\u6587\u4e32\u201c\u6253\u6563\u201d\uff0c\u5e76\u4e00\u9f50\u8bf7\u6c42\u6570\u636e\u5e93\uff0c\u4ee5\u671f\u80fd\u547d\u4e2d\u4e0d\u65ad\u5343\u53d8\u4e07\u5316\u7684\u82f1\u8bed\u5355\u8bcd\uff1a\\n\\n```dart\\nfinal sequence = \\"qwertyuiopasdfghjkl\\";\\nfinal sequences = [\\"qwe\\",\\"wer\\",\\"ert\\",...,\\"qwer\\",\\"wert\\",...,\\"qwert\\",...,\\"qwertyuiopasdfghjkl\\"];\\nfinal List results = await queryDB(keys:sequences); // average latency: ~6ms\\n```\\n\\n\u5728\u8bf7\u6c42\u5b8c\u6210\u540e\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5bf9\u8bf7\u6c42\u7ed3\u679c\u8fdb\u884c\u53bb\u91cd\uff0c\u903b\u8f91\u5982\u4e0b\uff1a\\n\\n- \u7ed3\u679c\u4e2d\u5305\u542b owl (\u732b\u5934\u9e70)\\n- \u7ed3\u679c\u4e2d\u5305\u542b knowledge(\u77e5\u8bc6)\\n- \u5e8f\u5217 \\"knowledge\\" \u5305\u542b \\"owl\\"\uff0c\u79fb\u9664\u67e5\u8be2\u7ed3\u679c\u4e2d\u7684 owl\\n\\n:::note\\n\u76ee\u524d\uff0c\u8be5\u903b\u8f91\u8fd8\u6ca1\u6709\u5b9e\u73b0\u5bf9\u51b2\u7a81\u7684\u5224\u5b9a\uff1a\u5982\u5411\u6570\u636e\u5e93\u67e5\u8be2\u7684 key \u4e3a `\\"gitignore\\"`\uff0c\u67e5\u8be2\u7ed3\u679c\u4e3a `[\\"git\\", \\"tig\\", \\"ignore\\"]`\uff0c\u76ee\u524d\u8fd9\u4e09\u4e2a\u7ed3\u679c\u90fd\u4f1a\u88ab\u6e32\u67d3\u5230 HUD \u4e0a\u3002\u4f46\u663e\u7136\uff0c\u9009\u53d6 git \u548c ignore \u8fd9\u4e24\u4e2a\u67e5\u8be2\u7ed3\u679c\u9020\u6210\u7684 \u201c\u51b2\u7a81\u201d \u662f\u6700\u5c0f\u7684\\n:::\\n\\n### \u53d1\u97f3\\n\\n\u672c\u7a0b\u5e8f\u4f7f\u7528 [AVFoundation - Speech synthesis](https://developer.apple.com/documentation/avfoundation/speech_synthesis) API \u6765\u5b9e\u65f6\u5408\u6210\u8bed\u97f3\\n\\n\u8be5 API \u53ef\u4ee5\u540c\u65f6\u5728 iOS/macOS \u4e0a\u4f7f\u7528\uff0c\u4e14\u53d1\u97f3\u8f83\u4e3a\u51c6\u786e\uff0c\u4ee5\u6211\u81ea\u8eab\u7684\u80fd\u529b(\u5168\u56fd\u5927\u5b66\u82f1\u8bed\u56db\u7ea7)\u6765\u770b\uff0c\u6548\u679c\u8fd8\u7b97\u6ee1\u610f\uff0c\u786e\u5b9e\u6bd4\u6211\u7684\u53d1\u97f3\u51c6 \ud83e\udd23\\n\\n\u5f53\u7136\uff0c\u540e\u7ee7\u5982\u679c\u6709\u66f4\u9ad8\u7684\u9700\u6c42\uff0c\u4e5f\u6709\u5f88\u591a\u5176\u4ed6\u53ef\u9009\u65b9\u6848"},{"id":"motivation","metadata":{"permalink":"/valo-reader-doc/blog/motivation","editUrl":"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/motivation.md","source":"@site/blog/motivation.md","title":"\u5f00\u53d1\u76ee\u7684","description":"\u6211\u7684\u9700\u6c42","date":"2023-03-15T19:25:06.000Z","formattedDate":"2023\u5e743\u670815\u65e5","tags":[{"label":"motivation","permalink":"/valo-reader-doc/blog/tags/motivation"}],"readingTime":5.835,"hasTruncateMarker":false,"authors":[{"name":"Ce Wang","title":"Solo dev","url":"https://github.com/haloWang","imageURL":"https://github.com/haloWang.png","key":"halowang"}],"frontMatter":{"slug":"motivation","title":"\u5f00\u53d1\u76ee\u7684","authors":["halowang"],"tags":["motivation"]},"prevItem":{"title":"Valo Reader for macOS \u6280\u672f\u6982\u8981","permalink":"/valo-reader-doc/blog/tech_detail_macos"}},"content":"## \u6211\u7684\u9700\u6c42\\n\\n\u4ee5\u4e0b\u662f\u6211\u7684\u4e24\u4e2a\u5e38\u7528\u9700\u6c42\uff1a\\n\\n- \u9605\u8bfb\u548c\u7406\u89e3\u6d4f\u89c8\u5668\u4e2d\u7684\u82f1\u6587\u5185\u5bb9\uff0c\u5982 github\uff0cstackoverflow\uff0creddit\\n- \u5728\u684c\u9762\u5e73\u53f0\u4e0a\u9605\u8bfb\u548c\u7406\u89e3\u4e00\u4e9b\u4ece\u672a\u88ab\u7ffb\u8bd1\u6210\u6c49\u8bed\u7684\u82f1\u6587\u4e66\u7c4d\uff0c\u8fd9\u4e9b\u4e66\u7c4d\u7684\u5927\u90e8\u5206\u4e3a PDF \u6587\u4ef6\\n\\n## \u6211\u9047\u5230\u7684\u56f0\u96be\\n\\n\u4f46\u5728\u8fdb\u884c\u4e0a\u8ff0\u7684\u9605\u8bfb\u7406\u89e3\u6d3b\u52a8\u65f6\uff0c\u6211\u9047\u5230\u4e86\u4e00\u4e9b\u56f0\u96be\uff1a\\n\\n- \u6211\u7ecf\u5e38\u5fd8\u8bb0\u82f1\u8bed\u5355\u8bcd\u7684\u610f\u601d\uff0c\u6216\u4e0d\u6562\u786e\u8ba4\u4e00\u4e2a\u5355\u8bcd\u7684\u610f\u601d\u3002\u751a\u81f3\uff0c\u6211\u7ecf\u5e38\u4f1a\u5fd8\u8bb0 30 \u79d2\u4e4b\u524d\u521a\u521a\u67e5\u8fc7\u7684\u5355\u8bcd\\n- \u5f53\u524d\u5404\u79cd\u67e5\u5355\u8bcd\u7684\u5e94\u7528\u7a0b\u5e8f\u548c\u7f51\u9875\u5bf9\u4e8e\u6211\u6765\u6765\u8bf4\u90fd\u6709\u4e9b\u7e41\u7410\uff0c\u6211\u9700\u8981\u7b49\u5f85\u8bcd\u5178\u8f6f\u4ef6\u7684\u7f51\u7edc\u8bf7\u6c42\uff0c\u7528\u4eba\u8111\u5ffd\u7565\u8bcd\u5178\u8f6f\u4ef6\u4e2d\u7684\u5e7f\u544a\u3002\u6211\u8fd8\u8981\u4fdd\u8bc1\u81ea\u5df1\u5728\u7e41\u7410\u7684\u64cd\u4f5c\u6d41\u7a0b\u4e2d\u4e0d\u4f1a\u5206\u5fc3\\n- \u4f7f\u7528\u5168\u6587\u7ffb\u8bd1\u8f6f\u4ef6\uff0c\u4f1a\u5bfc\u81f4\u76f8\u5f53\u4e00\u90e8\u5206\u7684\u4fe1\u606f\u4e22\u5931\u6216\u504f\u79bb\u539f\u672c\u542b\u4e49\uff0c\u6211\u9700\u8981\u5bf9\u5168\u6587\u7ffb\u8bd1\u8f6f\u4ef6\u7ea0\u9519\uff0c\u6216\u8005\u76f4\u63a5\u67e5\u770b\u539f\u6587\u624d\u80fd\u4e86\u89e3\u5230\u771f\u6b63\u7684\u91ca\u4e49\\n\\n\u4e3e\u4e2a\u4f8b\u5b50\uff0c\u6211\u4e0d\u77e5\u9053\u201cbabel\u201d\u8fd9\u4e2a\u5355\u8bcd\u7684\u6c49\u8bed\u610f\u601d\u3002\u4f46\u662f\uff0c\u5f53\u8fd9\u4e2a\u8bcd\u5728\u82f1\u8bed\u8bed\u6599\u4e2d\u51fa\u73b0\u65f6\uff0c\u5b83\u5c06\u4f60\u7684\u601d\u7eea\u4ece\u201c\u601d\u8003\u4f5c\u8005\u8868\u8fbe\u7684\u542b\u4e49\u201d\uff0c\u8f6c\u53d8\u4e3a\u201c\u601d\u8003 babel \u662f\u4ec0\u4e48\u610f\u601d\u201d\u3002\u5f53\u6211\uff0c\u67e5\u8be2\u5230\u4e86\u201cbabel\u201d\u7684\u610f\u601d\u540e\uff0c\u5728\u8fd4\u56de\u6765\u5c06\u81ea\u5df1\u7684\u6ce8\u610f\u529b\u6295\u5165\u4e4b\u524d\u7684\u4e0a\u4e0b\u6587\u3002\u8fd9\u4e2a\u53d1\u751f\u5728\u4eba\u8111\u4e2d\u7684\u4e0a\u4e0b\u6587\u5207\u6362\uff0c\u8fc7\u7a0b\u8017\u65f6\uff0c\u5bb9\u6613\u51fa\u9519\uff0c\u6b63\u53cd\u9988\u51e0\u4e4e\u6ca1\u6709\u3002\\n\\n\u6211\u8ba4\u4e3a\uff0c\u8fd9\u4e9b\u56f0\u96be\u589e\u52a0\u4e86\u6211\u4ece\u4e8b\u8111\u529b\u6d3b\u52a8\u65f6\u7684\u5fc3\u667a\u8d1f\u62c5\uff0c\u964d\u4f4e\u4e86\u6211\u7684\u9605\u8bfb\u6548\u7387\u3002\u751a\u81f3\uff0c\u76f4\u63a5\u5bfc\u81f4\u6211\u5206\u5fc3\uff0c\u7cbe\u795e\u4e0d\u96c6\u4e2d\uff0c\u6700\u7ec8\u653e\u5f03\u9605\u8bfb\uff0c\u653e\u5f03\u83b7\u53d6\u77e5\u8bc6\u3002\\n\\n\u9762\u5bf9\u4e0a\u8ff0\u7684\u56f0\u96be\uff0c\u6211\u4eec\u901a\u5e38\u7684\u89e3\u51b3\u529e\u6cd5\u90fd\u662f\u201c\u80cc\u5355\u8bcd\u201d\u3002\u53ef\u662f\uff0c\u80cc\u5355\u8bcd\u5bf9\u4e8e\u6211\u6765\u8bf4\u662f\u4e2a\u75db\u82e6\u7684\u8fc7\u7a0b\uff1a\\n\\n- \u6211\u9700\u8981\u80cc\u8bf5\u5927\u91cf\u7684\uff0c\u4f7f\u7528\u9891\u7387\u6781\u4f4e\uff0c\u5e76\u4e14\u56e0\u4f7f\u7528\u9891\u7387\u4f4e\u800c\u975e\u5e38\u5bb9\u6613\u53d8\u5f97\u751f\u758f\u7684\u5355\u8bcd\\n- \u8003\u7eb2\u5185\u7684\u5355\u8bcd\u4ece\u6765\u90fd\u662f\u4e3a\u5927\u5bb6\u51c6\u5907\u7684\uff0c\u80cc\u8bf5\u8003\u7eb2\u5185\u7684\u5355\u8bcd\u4e0d\u80fd\u8986\u76d6\u5230\u4e13\u4e1a\u9886\u57df\u7684\u8bcd\u6c47\\n- \u5728\u6c49\u8bed\u73af\u5883\u4e2d\uff0c\u4fdd\u8bc1\u5927\u8111\u5bf9\u5927\u91cf\u82f1\u8bed\u5355\u8bcd\u7684\u719f\u6089\u548c\u5feb\u901f\u54cd\u5e94\uff0c\u9700\u8981\u82b1\u8d39\u5f88\u591a\u989d\u5916\u7684\u65f6\u95f4\u4e0e\u7cbe\u529b\\n\\n\u5bf9\u6211\u6765\u8bf4\uff0c\u80cc\u5355\u8bcd\u7684\u6548\u7387\u592a\u4f4e\u4e86\uff0c\u5956\u52b1\u6765\u7684\u592a\u665a\uff0c\u592a\u4e0d\u786e\u5b9a\uff0c\u9700\u8981\u5f3a\u5927\u7684\u610f\u5fd7\u529b\u624d\u80fd\u6301\u7eed\u4e0b\u53bb\u3002\u6295\u5165\u4ea7\u51fa\u6bd4\u592a\u9ad8\u3002\\n\\n### \u77db\u76fe\\n\\n\u4e0a\u8ff0\u8fd9\u4e9b\u52a8\u4f5c\u90fd\u4f1a\u5bf9\u4e8e\u6211\u6765\u8bf4\u662f\u8f83\u5927\u7684\u5fc3\u667a\u8d1f\u62c5\u3002\u4e0d\u65ad\u5730\u5e72\u6270\u7740\u6211\u7684**_\u9605\u8bfb - \u7406\u89e3_**\u8fc7\u7a0b\u3002\u963b\u788d\u6211\u901a\u8fc7\u82f1\u8bed\u83b7\u53d6\u4fe1\u606f\u4e0e\u77e5\u8bc6\uff0c\u800c\u6211\u5f53\u524d\u6240\u671f\u671b\u83b7\u53d6\u7684\u4fe1\u606f\u4e0e\u77e5\u8bc6\uff0c\u5927\u90fd\u7531\u82f1\u8bed\u4e66\u5199\u800c\u6210\u3002\\n\\n## \u89e3\u51b3\u529e\u6cd5\\n\\n\u9762\u5bf9\u8fd9\u4e9b\u56f0\u96be\u3002\u6211\u7ecf\u8fc7\u79cd\u79cd\u601d\u8003\uff0c\u63a2\u7d22\uff0c\u5b9e\u8df5\u3002\u5f00\u53d1\u4e86\u8fd9\u6b3e\u540d\u4e3a [Valo Reader](http://localhost:3000/valo-reader-doc/docs/intro) \u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u8fd9\u6b3e\u5e94\u7528\u7a0b\u5e8f\u901a\u8fc7 [OCR](https://zh.wikipedia.org/wiki/%E5%85%89%E5%AD%A6%E5%AD%97%E7%AC%A6%E8%AF%86%E5%88%AB) \u6280\u672f\uff0c\u5b9e\u65f6\u5730\u8bc6\u522b\u7528\u6237\u5f53\u524d\u6b63\u5728\u9605\u8bfb\u7684\u82f1\u6587\u6587\u672c\uff0c\u7ecf\u7531\u7a0b\u5e8f\u5185\u7f6e\u8bcd\u5178\u548c\u7528\u6237\u8bbe\u7f6e\uff0c\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u3002\u5728\u8fd9\u6b3e\u5e94\u7528\u7a0b\u4e2d\uff0c\u6211\u671f\u671b\u63a5\u8fd1\u53ef\u80fd\u5730\u964d\u4f4e\u7528\u6237\u5728\u9605\u8bfb\u82f1\u6587\u8bed\u6599\u65f6\uff0c\u56e0\u4e3a\u638c\u63e1\u7684\u82f1\u8bed\u8bcd\u6c47\u91cf\u8f83\u5c11\uff0c\u800c\u4ea7\u751f\u7684\u989d\u5916\u5fc3\u667a\u8d1f\u62c5\u3002\\n\\n\ud83c\udf89 \u6211\u5e0c\u671b\uff0c\u6211\u4eec\u5728\u4f7f\u7528\u8fd9\u6b3e\u8f6f\u4ef6\u9605\u8bfb\u82f1\u6587\uff0c\u4e0d\u9700\u8981\u6709\u5355\u8bcd\u57fa\u7840\u4e86\uff0c\u8fd9\u53ef\u4ee5**\u5927\u5e45\u51cf\u5c11**\u751f\u758f\u5355\u8bcd\u5e26\u6765\u7684\u5fc3\u667a\u8d1f\u62c5\u3002\u52a9\u529b\u6211\u4eec\u7684**_\u9605\u8bfb - \u7406\u89e3_**\u8fc7\u7a0b\u3002\\n\\n![AppIcon](/img/logo.png)\\n\\n:::note\\n\u60f3\u8981\u5728\u4e0d\u540c\u5e73\u53f0\u4e0a\u83b7\u53d6 Valo Reader\uff0c\u8bf7\u53c2\u9605[\u5b89\u88c5](/docs/installation)\\n:::\\n\\n## \u4e3a\u4ec0\u4e48\u4e0d\u662f macOS \u81ea\u5e26\u7684\u4e09\u6307\u8f7b\u70b9\\n\\n\u5f53\u7528\u6237\u5728\u4f7f\u7528 macOS \u65f6\u53ef\u4ee5\u901a\u8fc7\u89e6\u6478\u677f\u5feb\u901f\u5524\u8d77\u7cfb\u7edf\u81ea\u5e26\u7684\u5b57\u5178\u67e5\u8be2\u5355\u8bcd, \u8fd9\u4e2a\u529f\u80fd\u88ab macOS \u79f0\u4e3a\u67e5\u8be2\u4e0e\u6570\u636e\u68c0\u6d4b\u5668(Look up & data detectors)\\n\\n\u76f8\u6bd4\u4e8e\u7cfb\u7edf\u81ea\u5e26\u529f\u80fd, Valo Reader \u5f53\u524d\u4e3b\u8981\u6709\u4ee5\u4e0b\u4f18\u52bf:\\n\\n- \u6240\u89c1\u5373\u53ef\u67e5: \u53ef\u4ee5\u53ca\u65f6\u67e5\u8be2\u56fe\u7247, \u89c6\u9891\u4e43\u81f3\u7535\u5b50\u6e38\u620f\u4e2d\u7684\u82f1\u8bed\u5355\u8bcd, \u53ea\u8981\u8fd9\u4e2a\u82f1\u8bed\u5355\u8bcd\u663e\u793a\u5728\u4e86\u5c4f\u5e55\u4e0a, \u8fd9\u662f macOS \u7cfb\u7edf\u6240\u505a\u4e0d\u5230\u7684\\n- \u7f16\u7a0b\u53cb\u597d: \u53ef\u4ee5\u67e5\u8be2\u6e90\u4ee3\u7801\u4e2d\u7684\u6bcf\u4e2a\u5355\u8bcd: `PascalCase` \u5c06\u88ab Valo Reader \u8bc6\u522b\u4e3a `\\"pascal\\"` \u548c `\\"case\\"` \u5e76\u88ab\u5206\u522b\u67e5\u8be2\\n\\n## \u4e3a\u4ec0\u4e48\u4e0d\u662f\u5176\u4ed6\u8bcd\u5178\u8f6f\u4ef6\\n\\n\u4e5f\u6709\u5f88\u591a\u8f6f\u4ef6\u5177\u6709\u5b9e\u65f6\u7684\u529f\u80fd, \u4f46\u662f\u636e\u89c2\u5bdf, \u4ed6\u4eec\u7684\u5728\u5c4f\u5e55\u4e0a\u53d6\u8bcd\u7684\u529f\u80fd\u4f3c\u4e4e\u90fd\u6ca1\u6709\u76f4\u63a5\u8c03\u7528 [Vision](https://developer.apple.com/documentation/vision/recognizing_text_in_images) framework. \u5bfc\u81f4\u67e5\u8be2\u7f13\u6162, \u53d6\u8bcd\u4e0d\u7cbe\u786e\u7b49\u95ee\u9898"}]}')}}]); \ No newline at end of file diff --git a/assets/js/ebb011f4.6cf2d7e7.js b/assets/js/ebb011f4.6cf2d7e7.js new file mode 100644 index 0000000..c44bf24 --- /dev/null +++ b/assets/js/ebb011f4.6cf2d7e7.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdf_doc_source=self.webpackChunkdf_doc_source||[]).push([[649],{3905:(e,t,n)=>{n.d(t,{Zo:()=>d,kt:()=>k});var a=n(7294);function l(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(l[n]=e[n]);return l}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var p=a.createContext({}),s=function(e){var t=a.useContext(p),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},d=function(e){var t=s(e.components);return a.createElement(p.Provider,{value:t},e.children)},u="mdxType",c={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,l=e.mdxType,r=e.originalType,p=e.parentName,d=o(e,["components","mdxType","originalType","parentName"]),u=s(n),m=l,k=u["".concat(p,".").concat(m)]||u[m]||c[m]||r;return n?a.createElement(k,i(i({ref:t},d),{},{components:n})):a.createElement(k,i({ref:t},d))}));function k(e,t){var n=arguments,l=t&&t.mdxType;if("string"==typeof e||l){var r=n.length,i=new Array(r);i[0]=m;var o={};for(var p in t)hasOwnProperty.call(t,p)&&(o[p]=t[p]);o.originalType=e,o[u]="string"==typeof e?e:l,i[1]=o;for(var s=2;s{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>i,default:()=>c,frontMatter:()=>r,metadata:()=>o,toc:()=>s});var a=n(7462),l=(n(7294),n(3905));const r={slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},i=void 0,o={permalink:"/valo-reader-doc/blog/tech_detail_macos",editUrl:"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/tech_detail_macos.md",source:"@site/blog/tech_detail_macos.md",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",description:"\u6458\u8981",date:"2023-10-19T05:59:23.000Z",formattedDate:"2023\u5e7410\u670819\u65e5",tags:[{label:"technical",permalink:"/valo-reader-doc/blog/tags/technical"}],readingTime:23.58,hasTruncateMarker:!1,authors:[{name:"Ce Wang",title:"Solo dev",url:"https://github.com/haloWang",imageURL:"https://github.com/haloWang.png",key:"halowang"}],frontMatter:{slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},nextItem:{title:"\u5f00\u53d1\u76ee\u7684",permalink:"/valo-reader-doc/blog/motivation"}},p={authorsImageUrls:[void 0]},s=[{value:"\u6458\u8981",id:"\u6458\u8981",level:2},{value:"\u4ec0\u4e48\u662f Valo Reader?",id:"\u4ec0\u4e48\u662f-valo-reader",level:2},{value:"\u9879\u76ee\u5e03\u5c40",id:"\u9879\u76ee\u5e03\u5c40",level:2},{value:"\u5de5\u7a0b\u67b6\u6784",id:"\u5de5\u7a0b\u67b6\u6784",level:2},{value:"flutter \u4fa7",id:"flutter-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f",id:"\u4e3b\u7a0b\u5e8f",level:4},{value:"flutter package: \u6587\u672c\u5757\u8868\u5f81",id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81",level:4},{value:"flutter package: \u57fa\u7840\u5e93",id:"flutter-package-\u57fa\u7840\u5e93",level:4},{value:"native \u4fa7",id:"native-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",level:4},{value:"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93",id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93",level:4},{value:"CocoaPods dependency: OCR \u529f\u80fd\u5e93",id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93",level:4},{value:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",level:2},{value:"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)",id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud",level:3},{value:"\u5728 HUD \u4e2d\u7684\u72b6\u6001",id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001",level:4},{value:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)",id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard",level:3},{value:"\u539f\u751f\u4fa7 (native)",id:"\u539f\u751f\u4fa7-native",level:3},{value:"\u6280\u672f\u7ec6\u8282",id:"\u6280\u672f\u7ec6\u8282",level:2},{value:"\u591a\u5c4f\u5e55",id:"\u591a\u5c4f\u5e55",level:3},{value:"\u5c4f\u5e55\u6355\u6349",id:"\u5c4f\u5e55\u6355\u6349",level:3},{value:"OCR",id:"ocr",level:3},{value:"\u6027\u80fd\u8868\u73b0",id:"\u6027\u80fd\u8868\u73b0",level:4},{value:"\u8282\u80fd\u4f18\u5316",id:"\u8282\u80fd\u4f18\u5316",level:4},{value:"\u4e0d\u622a\u53d6\u81ea\u5df1",id:"\u4e0d\u622a\u53d6\u81ea\u5df1",level:4},{value:"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d",id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d",level:3},{value:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",level:3},{value:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",level:4},{value:"\u91ca\u4e49\u67e5\u8be2",id:"\u91ca\u4e49\u67e5\u8be2",level:3},{value:"\u53d1\u97f3",id:"\u53d1\u97f3",level:3}],d={toc:s},u="wrapper";function c(e){let{components:t,...r}=e;return(0,l.kt)(u,(0,a.Z)({},d,r,{components:t,mdxType:"MDXLayout"}),(0,l.kt)("h2",{id:"\u6458\u8981"},"\u6458\u8981"),(0,l.kt)("p",null,"\u672c\u6587\u4e3b\u8981\u4ecb\u7ecd\u4e86 Valo Reader for macOS\uff08\u4e0b\u6587\u7b80\u79f0\u4e3a\u201c\u672c\u7a0b\u5e8f\u201d\uff09\u7684",(0,l.kt)("a",{parentName:"p",href:"#%E9%A1%B9%E7%9B%AE%E5%B8%83%E5%B1%80"},"\u9879\u76ee\u5e03\u5c40"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E5%B7%A5%E7%A8%8B%E6%9E%B6%E6%9E%84"},"\u5de5\u7a0b\u67b6\u6784"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E8%BF%90%E8%A1%8C%E6%97%B6%E5%8A%9F%E8%83%BD%E6%A8%A1%E5%9D%97"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),"\u4e0e",(0,l.kt)("a",{parentName:"p",href:"#%E6%8A%80%E6%9C%AF%E7%BB%86%E8%8A%82"},"\u6280\u672f\u7ec6\u8282"),"\u3002\u4f60\u8fd8\u53ef\u4ee5\u5728",(0,l.kt)("a",{parentName:"p",href:"/docs/intro"},"\u672c\u6587\u6863"),"\u7684\u5176\u4ed6\u9875\u9762\u4e86\u89e3\u672c\u7a0b\u5e8f\u7684",(0,l.kt)("a",{parentName:"p",href:"/valo-reader-doc/blog/motivation"},"\u5f00\u53d1\u52a8\u673a"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/usage"},"\u4f7f\u7528\u65b9\u5f0f"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/privacy_security"},"\u9690\u79c1\u4e0e\u5b89\u5168"),"\u7b49\u5185\u5bb9"),(0,l.kt)("h2",{id:"\u4ec0\u4e48\u662f-valo-reader"},"\u4ec0\u4e48\u662f Valo Reader?"),(0,l.kt)("p",null,"Valo Reader \u8fd9\u4e2a\u9879\u76ee\u662f\u6211\u4e2a\u4eba\u5728\u65e5\u5e38\u751f\u6d3b\u4e2d\u9010\u6e10\u840c\u751f\uff0c\u6f14\u5316\u548c\u5b9e\u65bd\u7684\u60f3\u6cd5\u3002\u662f\u6211\u4e3a\u4e86\u5728\u81ea\u5df1\u7535\u5b50\u8bbe\u5907\u8bbe\u5907\u4e0a\u80fd\u8f7b\u677e\u9605\u8bfb\u82f1\u6587\u5185\u5bb9\uff0c\u5e76\u6e10\u8fd1\u5f0f\u5730\u63d0\u9ad8\u81ea\u5df1\u7684\u82f1\u6587\u6c34\u5e73\uff0c\u800c\u81ea\u884c\u8bbe\u8ba1\u548c\u7814\u53d1\u7684\u4e00\u6b3e\u4ea7\u54c1\uff0c\u5176\u6838\u5fc3\u529f\u80fd\u5c55\u793a\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"intro",src:n(2731).Z,width:"960",height:"268"})),(0,l.kt)("p",null,"\u5f53\u4f60\u5728\u4f7f\u7528\u672c\u7a0b\u5e8f\u65f6\uff0c\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u4f60\u611f\u5230\u751f\u758f\u7684\u82f1\u8bed\u5355\u8bcd\u4e0a\u3002\u6309\u4e0b\u6307\u5b9a\u7684\u6309\u952e\uff08\u9ed8\u8ba4\u4e3a Fn\uff09\uff0c\u672c\u7a0b\u5e8f\u5373\u4f1a\u5728\u54cd\u5e94\u7684\u4f4d\u7f6e\u5c55\u793a\u5bf9\u5e94\u7684\u91ca\u4e49\uff0c\u540c\u65f6\u4f7f\u7528\u8bed\u97f3\u5408\u6210\u6765\u53d1\u97f3\u3002"),(0,l.kt)("p",null,"\u76ee\u524d\uff0c\u4f60\u53ef\u4ee5\u901a\u8fc7 ",(0,l.kt)("a",{parentName:"p",href:"https://apps.apple.com/cn/app/valo-reader/id6448040931"},"Mac App Store")," \u6216",(0,l.kt)("a",{parentName:"p",href:"/docs/installation"},"\u5176\u4ed6\u65b9\u5f0f"),"\u6765\u83b7\u53d6\u672c\u7a0b\u5e8f"),(0,l.kt)("h2",{id:"\u9879\u76ee\u5e03\u5c40"},"\u9879\u76ee\u5e03\u5c40"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u9879\u76ee\u7684\u4e3b\u4f53\u7ed3\u6784\uff1a\n",(0,l.kt)("img",{alt:"project_layout",src:n(3642).Z,width:"1996",height:"960"})),(0,l.kt)("p",null,"\u672c\u6587\u7684\u4f4d\u4e8e\u4e0a\u56fe\u4e2d\u7684\u6d45\u9ec4\u8272\u90e8\u5206\uff0c\u7531 ",(0,l.kt)("a",{parentName:"p",href:"https://docusaurus.io/"},"docusaurus")," \u751f\u6210"),(0,l.kt)("p",null,"\u672c\u9879\u76ee\u7684\u60f3\u6cd5\u6700\u65e9\u5b9e\u73b0\u4e8e\u5728\u6d4f\u89c8\u5668\u4e2d\u8fd0\u884c\u7684 js \u811a\u672c\uff0c\u4f60\u53ef\u4ee5\u5728 github \u4e0a\u770b\u5230\u672c\u9879\u76ee\u5728\u6d4f\u89c8\u5668\u4e0a\u7684\u65e9\u671f\u5b9e\u73b0\u53ca\u5bf9\u5e94\u7684",(0,l.kt)("a",{parentName:"p",href:"https://github.com/HaloWang/english_flow#%E5%8A%9F%E8%83%BD%E5%B1%95%E7%A4%BA"},"\u529f\u80fd\u5c55\u793a"),"\uff0c\u5373\u56fe\u4e2d\u6d45\u7eff\u8272\u90e8\u5206"),(0,l.kt)("p",null,"\u800c\u672c\u6587\u4e3b\u8981\u8bf4\u660e\u4e86\u4e0a\u56fe\u4e2d macOS\uff08\u6d45\u84dd\u8272\u90e8\u5206\uff09\u7684\u8fd0\u884c\u65b9\u5f0f\u4e0e\u6280\u672f\u7ec6\u8282"),(0,l.kt)("h2",{id:"\u5de5\u7a0b\u67b6\u6784"},"\u5de5\u7a0b\u67b6\u6784"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create XXX --platform macos")," \u547d\u4ee4\u521b\u5efa\uff0c\u5176\u6e90\u4ee3\u7801\u4e3b\u8981\u5206\u4e3a\u4e24\u90e8\u5206"),(0,l.kt)("h3",{id:"flutter-\u4fa7"},"flutter \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 dart \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f"},"\u4e3b\u7a0b\u5e8f"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u7684\u4e3b\u8981\u529f\u80fd\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u901a\u8fc7 method channel \u5b9e\u73b0 flutter \u4e0e native \u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u7528\u6237\u8bbe\u7f6e\u9762\u677f\uff08Dashboard\uff09"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\u4e0e\u8bca\u65ad\u68c0\u67e5\u4fe1\u606f"),(0,l.kt)("li",{parentName:"ol"},"\u4f7f\u7528\u7b2c\u4e09\u65b9\u4f9d\u8d56\u63d0\u4f9b\u7684\u529f\u80fd")),(0,l.kt)("h4",{id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81"},"flutter package: \u6587\u672c\u5757\u8868\u5f81"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u662f\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u4ee3\u7801\uff0c\u5728 macOS / iOS / Android \u4e09\u7aef\u5171\u4eab\u90e8\u5206\u903b\u8f91\u4e0e UI\uff0c\u5176\u4e3b\u8981\u529f\u80fd\u7531\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u89e3\u6790\u7528\u6237\u8bbe\u5907\u5c4f\u5e55\u4e0a\u8bc6\u522b\u5230\u7684\u6587\u672c\u5757\uff0c\u7ed3\u6784\u5316\u6587\u672c\u5757\uff0c\u5e76\u5c06\u5176\u7ef4\u62a4\u5230\u672c\u7a0b\u5e8f\u7684\u5185\u5b58\u4e2d"),(0,l.kt)("li",{parentName:"ol"},"\u58f0\u660e\u5e76\u5b9e\u73b0\u7528\u6237\u89c6\u89c9\u7126\u70b9\u4e0e\u5df2\u77e5\u7684\u6587\u672c\u4fe1\u606f\u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09")),(0,l.kt)("h4",{id:"flutter-package-\u57fa\u7840\u5e93"},"flutter package: \u57fa\u7840\u5e93"),(0,l.kt)("p",null,"\u5728\u6211\u7684\u6240\u6709 flutter \u9879\u76ee\u4e2d\u5171\u4eab\u7684\u57fa\u7840\u4f9d\u8d56"),(0,l.kt)("h3",{id:"native-\u4fa7"},"native \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 swift \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206"},"\u4e3b\u7a0b\u5e8f\u90e8\u5206"),(0,l.kt)("p",null,"\u6301\u6709 flutter engine\uff0c\u901a\u8fc7 method channel \u6253\u901a flutter \u4e0e ",(0,l.kt)("inlineCode",{parentName:"p"},"Base"),"\u3001",(0,l.kt)("inlineCode",{parentName:"p"},"OCR")," \u4ee5\u53ca\u5e94\u7528\u7a0b\u5e8f\u672c\u8eab\u7684\u53cc\u5411\u8c03\u7528"),(0,l.kt)("p",null,"\u521b\u5efa\u9879\u76ee\u65f6\u9ed8\u8ba4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController")," \u5df2\u88ab\u5220\u9664"),(0,l.kt)("h4",{id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93"},"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e Cocoa framework \u4ea4\u4e92\uff0c\u5982\uff1a\u6743\u9650\u7533\u8bf7\uff0c\u952e\u76d8\u76d1\u542c\uff0c\u9f20\u6807\u76d1\u542c\uff0c\u622a\u53d6\u5c4f\u5e55\uff0c\u79fb\u52a8\u627f\u8f7d flutter \u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow"),"\u4ee5\u53ca\u63d0\u4f9b\u57fa\u672c\u82e5\u5e72\u539f\u751f\u80fd\u529b"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u6211\u4e5f\u4f1a\u5728\u8be5\u4f9d\u8d56\u4e2d\u5b66\u4e60\uff0c\u5c1d\u8bd5\u548c\u5b9e\u73b0 Cocoa/AppKit \u72ec\u6709\u7684 API \u4e0e\u529f\u80fd"),(0,l.kt)("h4",{id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93"},"CocoaPods dependency: OCR \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Apple Vision framework - Text Recognizing")," \u4ea4\u4e92\uff0c\u5e76\u5b9e\u73b0\u90e8\u5206\u903b\u8f91\uff0c\u5982\uff1a\u53d1\u8d77\u6587\u672c\u8bc6\u522b\u8bf7\u6c42\uff0c\u7ef4\u62a4\u6587\u672c\u8bc6\u522b\u54cd\u5e94\u7f13\u5b58"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u5728 macOS \u548c iOS \u9879\u76ee\u540c\u65f6\u4f9d\u8d56\u5e76\u5171\u4eab\u4ee3\u7801"),(0,l.kt)("h2",{id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0c\u672c\u7a0b\u5e8f\u4e3b\u8981\u53ef\u5206\u4e3a\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\u3001\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD) \u548c\u539f\u751f\u4fa7 (native) \u4e09\u4e2a\u6a21\u5757"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u7684\u4e3b\u8981\u6a21\u5757\u53ca\u5176\u901a\u8baf\uff1a\n",(0,l.kt)("img",{alt:"modules_light",src:n(1974).Z+"#gh-light-mode-only",width:"1926",height:"894"}),"\n",(0,l.kt)("img",{alt:"modules_dark",src:n(8555).Z+"#gh-dark-mode-only",width:"1926",height:"894"})),(0,l.kt)("h3",{id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud"},"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)"),(0,l.kt)("p",null,"HUD \u662f\u4e00\u4e2a\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u81ea\u5df1\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterEngine")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")),(0,l.kt)("p",null,"HUD \u7684\u4e3b\u8981\u4efb\u52a1\u5305\u542b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u5411 native \u6d3e\u53d1 ocr \u8bf7\u6c42"),(0,l.kt)("li",{parentName:"ul"},"\u89e3\u6790 native \u5bf9\u5c4f\u5e55\u6307\u5b9a\u533a\u57df\u7684 ocr \u7ed3\u679c\uff0c\u5e76\u5728\u5185\u5b58\u4e2d\u7ef4\u62a4\u8be5\u7ed3\u679c\u4f9b\u540e\u7ee7\u7a0b\u5e8f\u903b\u8f91\u4f7f\u7528"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u4ee5\u5728\u9700\u8981\u65f6\u8ba1\u7b97\u91ca\u4e49\u5c55\u793a\u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u952e\u76d8\u6309\u952e\u72b6\u6001\uff0c\u5728\u53d8\u5316\u65f6\u6d3e\u53d1 ocr \u8bf7\u6c42\u5e76\u5c55\u793a\u5355\u8bcd\u91ca\u4e49"),(0,l.kt)("li",{parentName:"ul"},"\u5b9e\u65f6\u540c\u6b65 Dashboard engine \u4f20\u9012\u8fc7\u6765\u7684\u7528\u6237\u8bbe\u7f6e\uff0c\u5e76\u4fee\u6539\u5c55\u793a\u903b\u8f91")),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u8be5\u6a21\u5757\u6240\u7ef4\u62a4\u7684\u6570\u636e\u6620\u5c04\u5230\u5c4f\u5e55\u4e0a\u65f6\u7684\u53ef\u89c6\u5316\u6548\u679c\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation",src:n(9462).Z,width:"1898",height:"222"})),(0,l.kt)("h4",{id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001"},"\u5728 HUD \u4e2d\u7684\u72b6\u6001"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u7ba1\u7406\u7edd\u5927\u90e8\u5206\u7684\u72b6\u6001\u3002\u5728\u6784\u7b51\u672c\u7a0b\u5e8f\u65f6\uff0c\u6211\u7684\u601d\u8003\u8fc7\u7a0b\u4e3b\u8981\u57fa\u4e8e\u72b6\u6001\u4e0e\u72b6\u6001\u53d8\u5316\uff0c\u4ece\u5c55\u793a\u91ca\u4e49\u952e\u70b9\u51fb\u5230\u6e32\u67d3\u91ca\u4e49\u9762\u677f\u52a8\u753b"),(0,l.kt)("p",null,"\u5728\u5927\u91cf\u4f7f\u7528 riverpod \u4e2d\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"Provider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"ProviderContainer.listen")," \u540e\uff0c\u6211\u53ef\u4ee5\u6784\u5efa\u4e00\u4e2a\u9ad8\u5ea6\u5f02\u6b65\uff0c\u8c03\u7528\u987a\u5e8f\u4e0d\u654f\u611f\uff0c\u72b6\u6001\u53d8\u5316\u9a71\u52a8\u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u5728\u7f16\u7801\u8fc7\u7a0b\u4e2d\uff0c\u8fd9\u79cd\u65b9\u5f0f\u8ba9\u7a0b\u5e8f\u5458\u53ef\u4ee5\u4ece\u4e00\u5927\u4e32\u7684\u547d\u4ee4\u5f0f(Imperative)\u65b9\u6cd5\u8c03\u7528\u4e2d\u89e3\u653e\u51fa\u6765\uff0c\u4ec5\u4ec5\u9700\u8981\u5173\u6ce8\u548c\u786e\u4fdd\u6bcf\u4e2a\u6700\u5c0f\u903b\u8f91\u5355\u5143\u2014\u2014",(0,l.kt)("inlineCode",{parentName:"p"},"Provider"),"\uff0c\u786e\u4fdd\u5176\u5b9e\u73b0\u662f\u6b63\u786e\u7684\u5373\u53ef\uff0c\u8fd9\u51cf\u5c11\u4e86\u6784\u5efa\u590d\u6742\u548c\u957f\u4e32\u903b\u8f91\u51fa\u9519\u7684\u53ef\u80fd"),(0,l.kt)("p",null,"\u4e0b\u9762\u7684\u4e24\u5f20\u56fe\u8868\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u7684\u4e3b\u8981\u529f\u80fd\u6d41\u7a0b"),(0,l.kt)("p",null,"\u5207\u6362\u5c55\u793a\u91ca\u4e49\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:'graph LR\n A1[\u76d1\u542c\u952e\u76d8\u72b6\u6001];\n A[\u952e\u76d8\u72b6\u6001\u53d1\u751f\u6539\u53d8];\n query_button_down{\u5c55\u793a\u91ca\u4e49\u952e\u6309\u4e0b?};\n C[\u505c\u6b62\u5b9a\u65f6\u5668];\n D[\u542f\u52a8\u5b9a\u65f6\u5668\u5e76\u91cd\u7f6e\u6587\u672c\u5757\u72b6\u6001];\n mouse_state[\u9f20\u6807\u72b6\u6001];\n screen_state[\u5c4f\u5e55\u72b6\u6001];\n send_request_to_native["\u5411 native \u53d1\u9001\u622a\u56fe\u4e0e OCR \u8bf7\u6c42"];\n F[native OCR \u7ed3\u679c\u5df2\u83b7\u5f97];\n G[\u89e3\u6790 OCR \u7ed3\u679c];\n H1[\u66f4\u65b0\u6587\u672c\u5757\u72b6\u6001];\n\n mouse_state--\x3esend_request_to_native\n screen_state--\x3esend_request_to_native\n send_request_to_native-.->F;\n F--\x3eG;\n G--\x3eH1;\n A1-.->A;\n A--\x3equery_button_down;\n query_button_down--\x3e|false|C;\n query_button_down--\x3e|true|D;\n D-.->send_request_to_native;'}),(0,l.kt)("p",null,"\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u53d8\u66f4\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:"graph LR;\n H2[\u6587\u672c\u5757\u72b6\u6001\u53d8\u66f4];\n most_wanted_state_logic[\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u76d1\u542c\u903b\u8f91];\n most_wanted_state_changed[\u72b6\u6001\u6539\u53d8];\n mouse_change0[\u76d1\u542c\u9f20\u6807\u72b6\u6001];\n mouse_change[\u9f20\u6807\u72b6\u6001\u6539\u53d8];\n sync_mouse_to_flutter[\u540c\u6b65\u9f20\u6807\u72b6\u6001\u81f3 flutter];\n sync_current_screen[\u540c\u6b65\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u6570\u636e];\n sync_current_screen_to_flutter[\u540c\u6b65\u5c4f\u5e55\u6570\u636e\u81f3 flutter];\n hasTextBlock{\u72b6\u6001\u7684\u503c\u662f\u5426\u4e3a\u7a7a\\n\u5373\u9f20\u6807\u5f53\u524d\u4f4d\u7f6e\u6ca1\u6709\u8bc6\u522b\u5230\u6587\u672c};\n queryDB[\u67e5\u8be2\u6570\u636e\u5e93];\n dbHasData{\u6570\u636e\u5e93\u5305\u542b\u91ca\u4e49};\n hideAllFlow[\u9690\u85cf\u6240\u6709\u91ca\u4e49\u9762\u677f];\n asParam[\u4f5c\u4e3aOCR\u8bf7\u6c42\u7684\u53c2\u6570];\n screen_changed{\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u53d1\u751f\u53d8\u52a8?}\n\n \u5207\u6362\u5c55\u793a\u91ca\u4e49-.->H2\n mouse_change0-.->mouse_change--\x3esync_mouse_to_flutter;\n mouse_change --\x3e sync_current_screen --\x3e sync_current_screen_to_flutter;\n sync_current_screen_to_flutter --\x3e \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001\n sync_current_screen --\x3e screen_changed --\x3e |true| \u6539\u53d8\u627f\u8f7dHUD\u7684NSPanel\u4f4d\u7f6e;\n screen_changed --\x3e |false| return_0;\n H2 --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter -.-> asParam;\n \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001 -.-> asParam;\n most_wanted_state_logic --\x3e most_wanted_state_changed --\x3e hasTextBlock;\n \u5c55\u793a\u91ca\u4e49\u952e\u72b6\u6001 --\x3e most_wanted_state_logic;\n most_wanted_state_changed --\x3e hideAllFlow;\n hasTextBlock --\x3e |false| return_1;\n hasTextBlock --\x3e |true| queryDB;\n queryDB --\x3e dbHasData;\n dbHasData --\x3e |true| \u6e32\u67d3\u5c55\u793a\u91ca\u4e49\u9762\u677f\u52a8\u753b\n dbHasData --\x3e |false| return_2;"}),(0,l.kt)("h3",{id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard"},"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)"),(0,l.kt)("p",null,"Dashboard \u662f\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPanel")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u4e0d\u540c\u4e8e HUD \u7684\u72ec\u7acb engine \u4e0e method channel"),(0,l.kt)("p",null,"Dashboard \u7684\u4e3b\u8981\u4efb\u52a1\u662f\u4e3a\u7528\u6237\u63d0\u4f9b\u63a7\u5236\u672c\u7a0b\u5e8f\u7684 UI\uff0c\u5305\u62ec\u5feb\u6377\u952e\u8bbe\u7f6e\uff0c\u5f00\u673a\u81ea\u542f\uff0c\u5173\u4e8e\u672c\u7a0b\u5e8f\u7b49\u5e38\u89c1\u7528\u6237\u4ea4\u4e92"),(0,l.kt)("p",null,"\u540c\u65f6\uff0cDashboard \u4f1a\u7ef4\u62a4\u672c\u7a0b\u5e8f\u5728\u5185\u5b58\u548c\u786c\u76d8\u4e2d\u7684\u72b6\u6001\uff0c\u5e76\u5c06\u8fd9\u4e2a\u72b6\u6001\u901a\u8fc7 method channel \u540c\u6b65\u81f3\u4e0b\u9762\u5373\u5c06\u8981\u8bb2\u5230\u7684\u91ca\u4e49\u5c55\u793a\u9762\u677f"),(0,l.kt)("h3",{id:"\u539f\u751f\u4fa7-native"},"\u539f\u751f\u4fa7 (native)"),(0,l.kt)("p",null,"\u539f\u751f\u4fa7\u662f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create")," \u547d\u4ee4\u521b\u5efa\u51fa\u6765\u7684\u4f20\u7edf Xcode \u5de5\u7a0b"),(0,l.kt)("p",null,"\u5728\u539f\u751f\u4fa7\uff0c\u6211\u4e3b\u8981\u5173\u6ce8\u7684\u95ee\u9898\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u5904\u7406\u6765\u81ea ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u7684\u8c03\u7528\uff0c\u5e76\u5728\u9700\u8981\u65f6\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterResult")," \u54cd\u5e94\u5bf9\u5e94\u7684 flutter engine"),(0,l.kt)("li",{parentName:"ul"},"\u7ef4\u62a4\u627f\u8f7d HUD \u548c Dashboard \u7684\u4e24\u4e2a ",(0,l.kt)("inlineCode",{parentName:"li"},"NSWindow"),"\uff0c\u5c24\u5176\u662f\u5f53\u7528\u6237\u7684\u8bbe\u5907\u540c\u65f6\u94fe\u63a5\u591a\u4e2a\u5c4f\u5e55\u65f6\uff0c\u6b63\u786e\u5730\u8bbe\u7f6e HUD \u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u5c01\u88c5\u5e76\u5411 flutter \u63d0\u4f9b\u80fd\u529b\uff0c\u5982\uff1a",(0,l.kt)("ul",{parentName:"li"},(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u53d8\u5316\u548c\u952e\u76d8\u72b6\u6001\u53d8\u5316\uff0c\u5e76\u5c06\u72b6\u6001\u540c\u6b65\u81f3 flutter"),(0,l.kt)("li",{parentName:"ul"},"\u4e3a flutter \u63d0\u4f9b OCR \u548c\u8bed\u97f3\u5408\u6210\u80fd\u529b"),(0,l.kt)("li",{parentName:"ul"},"\u5728\u6267\u884c OCR \u65f6\u4f7f\u7528 swift \u5b9e\u73b0\u5fc5\u8981\u7684\u6027\u80fd\u4f18\u5316")))),(0,l.kt)("h2",{id:"\u6280\u672f\u7ec6\u8282"},"\u6280\u672f\u7ec6\u8282"),(0,l.kt)("p",null,"\u5f97\u76ca\u4e8e Apple \u4e00\u8109\u76f8\u627f\u7684 API \u8bbe\u8ba1\u4ee5\u53ca\u540c\u6837\u7684\u7f16\u7a0b\u8bed\u8a00\uff0c\u719f\u7ec3\u4e8e iOS \u5e94\u7528\u5f00\u53d1(Cocoa Touch & UIKit)\u7684\u7a0b\u5e8f\u5458\u5728\u9762\u5bf9\u4e0e macOS \u4ea4\u4e92\u7684 Cocoa \u548c AppKit \u6846\u67b6\u65f6\uff0c\u53ef\u4ee5\u590d\u7528\u5f88\u591a\u601d\u60f3\u4e0e\u903b\u8f91"),(0,l.kt)("p",null,"\u4f46\u76f8\u6bd4\u4e8e iOS\uff0cmacOS \u7ed9\u4e86\u7528\u6237\u66f4\u5927\u7684\u821e\u53f0\uff0c\u4e5f\u8ba9\u5f00\u53d1\u8005\u9762\u5bf9\u4e86\u66f4\u591a\u7684\u6311\u6218: \u9f20\u6807\uff0c\u952e\u76d8\uff0c\u7a97\u53e3\u548c\u5c4f\u5e55"),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u672c\u7a0b\u5e8f\u7684\u8fc7\u7a0b\u4e2d\uff0c\u6211\u9047\u5230\u4e86\u8bf8\u591a\u7684\u56f0\u96be\uff0c\u4e5f\u5728\u8305\u585e\u987f\u5f00\u65f6\u6536\u83b7\u4e86\u5f88\u591a\u5feb\u4e50\uff0c\u6211\u5728\u8fd9\u4e00\u7ae0\u8282\u4f1a\u8bb0\u5f55\u4e00\u4e0b"),(0,l.kt)("h3",{id:"\u591a\u5c4f\u5e55"},"\u591a\u5c4f\u5e55"),(0,l.kt)("p",null,"macOS \u53ca\u5176\u8fd0\u884c\u7684\u786c\u4ef6\u8bbe\u5907\u5e38\u5e38\u8fde\u63a5\u7740\u591a\u5757\u5c4f\u5e55\uff0c\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u529f\u80fd\u5c31\u662f\u5728\u4efb\u610f\u7684\u5c4f\u5e55\u4f4d\u7f6e\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\uff0c\u8fd9\u5c31\u8981\u786e\u4fdd\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\u653e\u7f6e\u4e8e\u6b63\u786e\u7684\u4f4d\u7f6e\u4e0a"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u8bf7\u6c42 native \u6355\u6349\u5c4f\u5e55\u65f6\uff0c\u6355\u6349\u7684\u77e9\u5f62\u6846\u7684\u4f4d\u7f6e\u6b63\u786e"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u89e3\u6790 ocr response \u540e\uff0c\u7ed3\u679c\u53ef\u4ee5\u6b63\u786e\u5730\u548c\u5c4f\u5e55\u4e0a\u771f\u6b63\u7684\u5185\u5bb9\u5efa\u7acb\u6620\u5c04")),(0,l.kt)("p",null,"\u548c UIKit \u4e2d\u5e38\u7528\u7684 CGRect \u4e0d\u540c\uff0cAppKit \u5bf9\u8bbe\u5907\u5c4f\u5e55\u7684\u62bd\u8c61 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen"),"\uff0c\u5176\u5750\u6807\u4ee5 NSRect \u8ba1\u7b97\uff0c\u5750\u6807\u7cfb\u539f\u70b9\u4e3a macOS \u539f\u59cb\u5c4f\u5e55\u7684\u5de6\u4e0b\u89d2\u3002\u800c\u6b64\u65f6\uff0c\u5982\u679c\u4f60\u7ed9\u4f60\u7684\u8bbe\u5907\u8fde\u63a5\u4e0a\u4e86\u5176\u4ed6\u7684\u5c4f\u5e55\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen.screens")," \u6240\u5448\u73b0\u5c4f\u5e55\u5e03\u5c40\uff0c\u53ef\u80fd\u5c31\u4f1a\u53d8\u6210\u4e0b\u56fe\u6240\u793a\u7684\u6837\u5b50\uff1a\n",(0,l.kt)("img",{alt:"appkit_screens_coordinate",src:n(1589).Z,width:"1630",height:"734"})),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u7684\u5de6\u4e0b\u89d2\u5750\u6807\u503c\u5e76\u975e\u662f (0, 0)\uff0c\u800c\u662f ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 0")," \u7684 (0, 0) \u70b9\u7684\u76f8\u5bf9\u4f4d\u7f6e (w",(0,l.kt)("sub",null,"1"),", d",(0,l.kt)("sub",null,"2"),")\u3002\u901a\u8fc7\u76d1\u542c\u9f20\u6807\u79fb\u52a8\u4e8b\u4ef6\u83b7\u53d6\u7684\u9f20\u6807\u4f4d\u7f6e ",(0,l.kt)("inlineCode",{parentName:"p"},"event.locationInWindow"),"\uff0c\u4e5f\u662f\u76f8\u5bf9\u4e8e\u539f\u70b9\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPoint")),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u65f6\uff0c\u4ec5\u4ec5\u5c06\u81ea\u5df1\u7684\u601d\u7ef4\u4ece iOS \u7684 CGRect \u5750\u6807\u7cfb\u8f6c\u5316\u81f3 NSRect \u5750\u6807\u7cfb\u8fd8\u7b97\u7b80\u5355\u3002\u4f46\u5728\u540e\u7ee7\u7684\u903b\u8f91\u4e2d\uff0c\u56e0\u4e3a OCR \u7ed3\u679c\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRectangleObservation.boundingBox")," \u53c8\u4f1a\u56de\u5230 CGRect \u5750\u6807\u7cfb\uff0c\u5750\u6807\u7cfb\u7684\u9891\u7e41\u8f6c\u5316\u786e\u5b9e\u4f1a\u7ed9\u4eba\u5e26\u6765\u4e00\u5b9a\u7684\u56f0\u6270"),(0,l.kt)("h3",{id:"\u5c4f\u5e55\u6355\u6349"},"\u5c4f\u5e55\u6355\u6349"),(0,l.kt)("p",null,"\u60f3\u6355\u6349 macOS \u7684\u5c4f\u5e55\uff0c\u4f60\u53ef\u4ee5\u8c03\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u51fd\u6570"),(0,l.kt)("p",null,"\u4f46\u503c\u5f97\u6ce8\u610f\u7684\u662f\uff0c\u60f3\u8981\u622a\u53d6\u5c4f\u5e55\u4e0a\u5176\u4ed6\u8fdb\u7a0b\u7684\u5185\u5bb9(\u6bd4\u5982 IDE \u6216\u8005\u6d4f\u89c8\u5668)\uff0c\u9700\u8981\u9884\u5148\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGRequestScreenCaptureAccess"),"\uff0c\u7533\u8bf7\u5230\u622a\u53d6\u5176\u4ed6\u8fdb\u7a0b UI \u7684\u6743\u9650\uff0c\u5426\u5219 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u53ea\u80fd\u62ff\u5230 macOS \u7684\u684c\u9762\uff08\u4ee5\u53ca\u7a0b\u5e8f\u672c\u8eab\uff09"),(0,l.kt)("h3",{id:"ocr"},"OCR"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 Apple \u4e3a\u5f00\u53d1\u8005\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Vision - Recognizing Text")," \u8fdb\u884c\u5c4f\u5e55\u6587\u672c\u7684\u63d0\u53d6"),(0,l.kt)("p",null,"OCR \u8bf7\u6c42\u7684\u8c03\u7528\u7531 flutter side \u5524\u8d77\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'void dispatchOCRRequestToNative() async {\n final x = 100;\n final y = 100;\n final width = 300;\n final height = 80;\n final dimensions = [x, y, width, height];\n final result = await methodChannel.invokeMethod("captureAndOCR", dimensions);\n\n // parse the result from native\n // ...\n}\n')),(0,l.kt)("p",null,"Native side \u5728\u63a5\u6536\u5230\u8bf7\u6c42\u540e\uff0c\u4f1a\u6267\u884c\u622a\u5c4f\u548c\u8bc6\u522b\u64cd\u4f5c\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"import Vision\n// ...\nlet result : FlutterResult = ...\nlet image = captureScreen()\nlet handler = VNImageRequestHandler(cgImage: image)\nlet request = VNRecognizeTextRequest { request, error in\n // ocr finished\n let parsedResult = parse(request.results)\n result(parsedResult)\n}\nhandler.perform([request])\n")),(0,l.kt)("h4",{id:"\u6027\u80fd\u8868\u73b0"},"\u6027\u80fd\u8868\u73b0"),(0,l.kt)("p",null,"\u5728 macbook 2021 \u7684 m1 pro \u4e0a\uff0c\u4ee5 320\u271575 \u7684\u8bbe\u8ba1\u5206\u8fa8\u7387\u622a\u56fe(\u5b9e\u9645\u5206\u8fa8\u7387\u4e3a 640\u2715150)\uff0c\u6bcf\u79d2 11 \u5e27\u7684\u60c5\u51b5\u4e0b\u8fdb\u884c\u957f\u65f6\u95f4\u7684\u622a\u56fe\u548c OCR \u64cd\u4f5c\uff0c\u6574\u4e2a\u6d41\u7a0b\u7684\u5ef6\u8fdf\u5e73\u5747\u7ea6\u4e3a 65ms\uff0c\u6211\u8ba4\u4e3a\u8fd8\u7b97\u662f\u4e00\u4e2a\u53ef\u63a5\u53d7\u7684\u72b6\u6001"),(0,l.kt)("h4",{id:"\u8282\u80fd\u4f18\u5316"},"\u8282\u80fd\u4f18\u5316"),(0,l.kt)("p",null,"\u5373\u4fbf\u662f\u6027\u80fd\u5141\u8bb8\uff0c\u4f18\u5316\u4e5f\u662f\u5e94\u8be5\u505a\u7684\uff0c\u7528\u6237\u4e00\u65e6\u4f7f\u7528\u4f60\u7684 App\uff0c\u5c31\u611f\u89c9 macbook \u7684 C \u9762\u53d1\u70ed\uff0c\u8fd9\u662f\u65e0\u6cd5\u5bb9\u5fcd\u7684"),(0,l.kt)("p",null,"\u5f53\u524d\u5728 OCR \u6d41\u7a0b\u4e2d\u4e3b\u8981\u7684\u6027\u80fd\u4f18\u5316\u6b65\u9aa4\u662f\u5728 native \u7ef4\u62a4\u4e00\u4e2a\u5148\u8fdb\u5148\u51fa\uff0c\u6700\u5927\u5bb9\u91cf\u4e3a 40 \u5e27\u7684\u5b57\u5178\uff0c\u4ee5\u56fe\u7247\u6570\u636e\u4e3a key \u7f13\u5b58 OCR \u7684\u7ed3\u679c\uff0c\u4e0b\u9762\u662f\u7b80\u5316\u540e\u7684\u4ee3\u7801\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"let cacheManager = CacheManager()\n// ...\nlet image = captureScreen()\nlet key = image.dataProvider.data\nif let cachedResult = cacheManager[key] {\n delegate.onOCRResult(cachedResult)\n return\n}\n// ...\ndispatchOCRRequestToDeviceGPU(image) { result\n cacheManager[key] = result\n // ... other logic\n}\n// ...\n")),(0,l.kt)("p",null,"\u5f53\u9f20\u6807\u6307\u9488\u4f4d\u7f6e\u4e0d\u53d8\uff0c\u622a\u56fe\u83b7\u53d6\u7684\u56fe\u7247\u4e0d\u53d8\u65f6\uff0cnative \u5728\u5904\u7406 OCR \u8bf7\u6c42\u65f6\u4f1a\u5148\u547d\u4e2d\u7f13\u5b58\uff0c\u5e76\u76f4\u63a5\u8fd4\u56de\u7ed3\u679c\uff0c\u4ee5\u51cf\u5c11\u975e\u5fc5\u9700\u7684 GPU \u8c03\u7528"),(0,l.kt)("p",null,"\u5728\u7cfb\u7edf\u81ea\u5e26\u7684\u6d3b\u52a8\u76d1\u89c6\u5668\u4e2d\u67e5\u770b\u8fdb\u7a0b\u3002\u53d1\u73b0\uff0c\u5728\u5e94\u7528\u7684\u7f13\u5b58\u7b56\u7565\u540e\uff0c\u5728\u672c\u7a0b\u5e8f\u6d3b\u52a8\u65f6\uff0c\u5176 CPU/GPU \u5360\u7528\u7387\u786e\u5b9e\u964d\u4f4e\u4e86\u5f88\u591a\uff0c\u540c\u65f6\uff0cOCR \u7684\u7f13\u5b58\u7ed3\u679c\u88ab\u964d\u4e3a\u4e86 5ms"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u611f\u89c9\uff0c\u4ee5 ",(0,l.kt)("inlineCode",{parentName:"p"},"CFData")," \u4f5c\u4e3a key \u67e5\u8be2\u5b57\u5178\u8fd8\u4e0d\u662f\u6548\u7387\u6700\u9ad8\u7684\u7b97\u6cd5\uff0c\u5e94\u8be5\u53ef\u4ee5\u7ee7\u7eed\u63a2\u7d22\u4e00\u4e0b")),(0,l.kt)("h4",{id:"\u4e0d\u622a\u53d6\u81ea\u5df1"},"\u4e0d\u622a\u53d6\u81ea\u5df1"),(0,l.kt)("p",null,"\u5728\u6211\u8fdb\u884c\u5f00\u53d1\u65f6\uff0c\u53d1\u73b0\u5f53\u672c\u7a0b\u5e8f\u5728\u5c55\u793a\u5355\u8bcd\u91ca\u4e49 UI (HUD)\u65f6\uff0c\u56e0\u4e3a HUD \u672c\u8eab\u4e5f\u4f1a\u5728\u4e00\u5b9a\u8303\u56f4\u5185\u88ab\u622a\u5c4f\u51fd\u6570\u6355\u83b7\uff0c\u5bfc\u81f4 HUD \u4f1a\u5f71\u54cd OCR \u7684\u7ed3\u679c\uff0c\u800c\u5728 OCR \u7ed3\u679c\u53d8\u52a8\u540e\uff0cHUD \u53c8\u4f1a\u968f\u4e4b\u53d1\u751f\u53d8\u52a8\uff0c\u518d\u6b21\u5f71\u54cd OCR \u7ed3\u679c\uff0c\u5c31\u8fd9\u6837\u5faa\u73af\u5f80\u590d\uff0c\u8fde\u7f13\u5b58\u4e5f\u90fd\u5931\u6548\u4e86\u3002\u5728\u67e5\u9605\u548c\u5c1d\u8bd5\u5927\u91cf\u7684 API \u540e\uff0c\u6211\u7ec8\u4e8e\u5728 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u4e2d\u627e\u5230\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},"sharingType")," \u8fd9\u4e2a\u5c5e\u6027\uff0c\u5c4f\u5e55\u6355\u6349\u65b9\u6cd5\u6355\u83b7 HUD \u81ea\u8eab\uff0c\u603b\u7b97\u662f\u6253\u7834\u4e86\u8fd9\u4e2a\u94fe\u6761"),(0,l.kt)("h3",{id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d"},"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d"),(0,l.kt)("p",null,"\u5bf9 Vision - VNRecognizeTextRequest \u7684\u7ed3\u679c\u89e3\u6790\u662f\u4e2a\u590d\u6742\u7684\u4efb\u52a1"),(0,l.kt)("p",null,"\u6267\u884c\u8fd9\u4e2a\u590d\u6742\u4efb\u52a1\u7684\u539f\u56e0\u6709\u4e09\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u8bbe\u8ba1\u7684\u4ea4\u4e92\u662f\u201c\u7528\u6237\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u611f\u5174\u8da3\u7684\u5355\u8bcd\u4e0a\u65f6\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u201d\uff0c\u8fd9\u5c31\u8981\u6c42\u6211\u4eec\u77e5\u9053\u5c4f\u5e55\u4e0a\u6bcf\u4e2a\u5355\u8bcd\u7684\u5177\u4f53\u4f4d\u7f6e\u548c\u5185\u5bb9"),(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u9488\u5bf9\u6e90\u4ee3\u7801\u7279\u5316\uff0c\u5728\u9762\u5bf9",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E9%A9%BC%E5%B3%B0%E5%BC%8F%E5%A4%A7%E5%B0%8F%E5%86%99"},"\u9a7c\u5cf0\u547d\u540d\u6cd5"),"\u65f6\uff0c\u9700\u8981\u77e5\u9053\u6784\u6210\u4e00\u4e2a symbol \u7684\u6bcf\u4e2a\u5355\u8bcd\u7684\u610f\u601d\uff1a",(0,l.kt)("inlineCode",{parentName:"li"},"ThisIsAVeryVeryLongClassName")," -> ",(0,l.kt)("inlineCode",{parentName:"li"},"This"),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"Is")),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"A")),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Long"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Class"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Name")," (\u56e0\u8fc7\u4e8e\u7b80\u5355\uff0c\u4e22\u5f03\u957f\u5ea6\u5c0f\u4e8e 3 \u7684\u82f1\u8bed\u5355\u8bcd)"),(0,l.kt)("li",{parentName:"ol"},"\u6211\u4eec\u8981\u4ee5\u5355\u8bcd\u7684\u6587\u672c\u5185\u5bb9\u4e3a key\uff0c\u67e5\u8be2\u6570\u636e\u5e93\u3002\u5728\u6587\u672c\u5e8f\u5217\u5305\u542b\u7279\u6b8a\u5b57\u7b26\u65f6\uff0c\u6211\u4eec\u662f\u65e0\u6cd5\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u5230\u5355\u8bcd\u91ca\u4e49\u7684\u3002\u6240\u4ee5\u8981\u79fb\u9664\u975e\u5b57\u6bcd\u5b57\u7b26\u3002\u540c\u65f6\uff0c\u8fd9\u4e00\u64cd\u4f5c\u4e5f\u53ef\u4ee5\u81ea\u7136\u800c\u7136\u5730\u9002\u914d\u7f16\u7a0b\u8bed\u8a00\u4e2d\u7684\u5176\u4ed6\u547d\u540d\u65b9\u5f0f\uff0c\u6bd4\u5982\u79fb\u9664\u4e0b\u5212\u7ebf\u5c31\u53ef\u4ee5\u9002\u5e94",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E8%9B%87%E5%BD%A2%E5%91%BD%E5%90%8D%E6%B3%95"},"\u86c7\u5f62\u547d\u540d\u6cd5"))),(0,l.kt)("p",null,"\u5047\u8bbe\u6211\u4eec\u5728\u5bf9\u4e0b\u9762\u7684\u56fe\u7247\u6267\u884c OCR \u8bf7\u6c42"),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"ocr_text_source",src:n(3591).Z,width:"1038",height:"198"})),(0,l.kt)("p",null,"\u5728 Native \u7aef\u53d6 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizeTextRequest.results.topCandidates(1)")," \u540e\uff0c\u7ed3\u679c\u5982\u4e0b"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision provides its text-recognition capabilities through VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request, an image-based request type that finds and extracts text in images. The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following example shows how to use VNImageRequestHandler to perform a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest for recognizing text in the specified CGImage.",\n]\n')),(0,l.kt)("p",null,"\u5728 Native \u7aef\u4ee5",(0,l.kt)("strong",{parentName:"p"},"\u7a7a\u683c"),"\u5b57\u7b26\u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizedText.boundingBox(for:)")," \u5bf9 OCR \u7ed3\u679c\u8fdb\u884c\u7b2c\u4e00\u6b21\u6574\u7406\uff0c\u4e0a\u8ff0\u7684\u7ed3\u679c\u4f1a\u53d8\u4e3a\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text-recognition", "capabilities", "through", "VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request,", "an", "image-based", "request", "type", "that", "finds", "and", "extracts", "text", "in", "images.", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "to", "use", "VNImageRequestHandler", "to", "perform", "a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest", "for", "recognizing", "text", "in", "the", "specified", "CGImage.",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\u5c06 boundingBox \u7684\u8fd4\u56de\u503c\u548c\u622a\u5c4f\u5c3a\u5bf8\u8fdb\u884c\u4e58\u7b97\uff0c\u8fd4\u56de\u7ed9 flutter \u7aef\u3002Flutter \u7aef\u5728\u62ff\u5230\u7ed3\u679c\u540e\u4e3b\u8981\u8fdb\u884c\u4e24\u4e2a\u6b65\u9aa4\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u79fb\u9664\u975e\u82f1\u6587\u5b57\u7b26\uff0c\u8ba1\u7b97\u65b0\u7684 rect"),(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u5bf9\u5305\u542b\u5927\u5199\u5b57\u6bcd\u7684\u5b57\u7b26\u4e32\u8fdb\u884c\u5206\u5272\uff0c\u8ba1\u7b97\u65b0\u7684 rect")),(0,l.kt)("p",null,"\u8fd9\u6837\uff0c\u6587\u672c\u5757\u7684\u5185\u5bb9\u5728 flutter \u7aef\u5c31\u4f1a\u53d8\u6210\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text", "recognition", "capabilities", "through", "Recognize", "Text",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request","image", "based", "request", "type", "that", "finds", "and", "extracts", "text", "images", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "use", "Image", "Request", "Handler", "perform",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Recognize", "Text", "Request", "for", "recognizing", "text", "the", "specified", "Image",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\uff0cflutter \u4fa7\u4f1a\u4ee5\u4e0b\u9762\u7684\u4ee3\u7801\u5bf9\u8fd9\u4e9b OCR \u4fe1\u606f\u8fdb\u884c\u62bd\u8c61\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},"/// The representation of block\n///\n/// \u6587\u672c\u5757\u8868\u5f81\nclass Block {\n String text;\n double x;\n double y;\n double w;\n double h;\n}\n\n// ...\n\n/// All text representations in memory\n///\n/// \u6240\u6709\u6587\u672c\u5757\u8868\u5f81\nfinal blocks = StateProvider>((_) => []);\n")),(0,l.kt)("p",null,"\u4e0a\u9762\u7684\u6570\u636e\u5728\u5c4f\u5e55\u4e0a\u7684\u53ef\u89c6\u5316\u6548\u679c\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation_2",src:n(6607).Z,width:"1050",height:"184"})),(0,l.kt)("p",null,"\u5176\u4e2d\u5916\u56f4\u7eff\u6846\u4ee3\u8868\u622a\u5c4f\u8303\u56f4\uff0c\u5185\u90e8\u7684\u5c0f\u7eff\u6846\u4ee3\u8868 native \u7aef\u8fd4\u56de\u7684\u7ed3\u679c\uff0c\u5c0f\u7eff\u6846\u5185\u90e8\u7684\u84dd\u8272\u6846\u4ee3\u8868 flutter side \u7b2c\u4e8c\u6b21\u89e3\u6790\u7ed3\u679c\u3002\u53ef\u4ee5\u770b\u5230\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},'"VNImageRequestHandler"')," \u8fd9\u4e2a\u5b57\u7b26\u4e32\u88ab\u5206\u5272\u4e3a\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},'"Image"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Request"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Handler"')," \u8fd9\u4e09\u4e2a\u6587\u672c\u5757\u3002\u8fd9\u6837\uff0c\u6211\u5c31\u53ef\u4ee5\u77e5\u9053\u7528\u6237\u5230\u5e95\u5bf9\u54ea\u4e00\u6bb5\u5b57\u6bcd\u5e8f\u5217\uff08\u5355\u8bcd\uff09\u611f\u5174\u8da3\u4e86\uff08\u5047\u8bbe\u6e90\u4ee3\u7801\u7b26\u53f7\u7684\u547d\u540d\u662f\u89c4\u8303\u7684 \ud83d\ude07\uff09\u3002"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u4f60\u4e5f\u53ef\u4ee5\u6253\u5f00\u672c\u7a0b\u5e8f\u7684\u7528\u6237\u8bbe\u7f6e\u9762\u677f\u6765\u76f4\u63a5\u89c2\u5bdf\u5176\u8fd0\u884c\u65f6\u8868\u73b0\uff1a\n",(0,l.kt)("img",{alt:"dashboard_inspect",src:n(6521).Z,width:"970",height:"192"}))),(0,l.kt)("h3",{id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"},"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,'\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u4f1a\u521b\u5efa\u4e24\u4e2a flutter engine\uff0c\u5206\u522b\u7528\u4e8e\u6e32\u67d3"\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)"\u548c"\u6b22\u8fce\u4e0e\u8bbe\u7f6e\u9875\u9762(Dashboard)"'),(0,l.kt)("p",null,"\u8fd9\u4e24\u4e2a\u5f15\u64ce\u5747\u901a\u8fc7\u57fa\u672c\u65b9\u5f0f\u521b\u5efa\uff0c\u5747\u91c7\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")," \u4e0e native \u8fdb\u884c\u901a\u8baf"),(0,l.kt)("p",null,"\u53cc\u5f15\u64ce\u5171\u4eab\u540c\u4e00\u4efd\u4ee3\u7801\uff0c\u7ecf\u7531\u4e0d\u540c\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/dart-lang/sdk/blob/master/runtime/docs/compiler/aot/entry_point_pragma.md"},"dart \u5165\u53e3\u51fd\u6570"),"\u542f\u52a8\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'// \u8bbe\u7f6e\u9875\u9762 (Dashboard)\ndashboardEngine = FlutterEngine(name: "dashboard", project: nil)\ndashboardChannel = FlutterMethodChannel(name: "dashboard", binaryMessenger: dashboardEngine.binaryMessenger)\ndashboardEngine.run(withEntrypoint: "_dashboard")\n//...\n\n// \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\nhudEngine = FlutterEngine(name: "hud", project: nil)\nhudChannel = FlutterMethodChannel(name: "hud", binaryMessenger: hudEngine.binaryMessenger)\nhudChannel.run(withEntrypoint: "_hud")\n')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u4e2a\u4eba\u6ca1\u6709\u4f7f\u7528\u5b98\u65b9\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://docs.flutter.dev/add-to-app/multiple-flutters"},"FlutterEngineGroup \u65b9\u6848"),"\uff0c\u56e0\u4e3a\u5728\u5f53\u521d\u8fdb\u884c\u5c1d\u8bd5\u65f6\u53d1\u73b0\u4e86\u5f15\u64ce\u65e0\u54cd\u5e94\u7684\u95ee\u9898\uff0c\u81f3\u4eca\u4ecd\u7136\u662f P2 \u7ea7\u522b\u7684 open ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/flutter/flutter/issues/119403"},"issue"))),(0,l.kt)("h4",{id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"},"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u6765\u7ba1\u7406\u7edd\u5927\u90e8\u5206\u72b6\u6001\uff0c\u6240\u4ee5\u5728\u540c\u6b65\u72b6\u6001\u65f6\uff0c\u6211\u4e5f\u671f\u671b\u81ea\u5df1\u7684\u81ea\u5df1\u7684\u5fc3\u667a\u6a21\u578b\u53ef\u4ee5\u66f4\u8d34\u8fd1 riverpod"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0cDashboard engine \u9700\u8981\u5c06\u7528\u6237\u8bbe\u7f6e\u540c\u6b65\u81f3 HUD engine (\u6bd4\u5982\u66f4\u6539\u67e5\u8bcd\u5feb\u6377\u952e)\uff0c\u6211\u5728\u8fd9\u91cc\u76d1\u542c\u4e86 Dashboard engine \u5bf9\u5e94\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," \u7684\u53d8\u66f4\uff0c\u5e76\u901a\u8fc7 method channel\uff0c\u7ecf\u7531 native \u8f6c\u53d1\u81f3 HUD engine\uff0cHUD engine \u5728\u6536\u5230\u4e86\u901a\u77e5\u540e\uff0c\u518d\u66f4\u65b0\u81ea\u5df1\u7ef4\u62a4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider"),"\u3002\u4ece\u800c\u5b9e\u73b0\u4e86\u8de8\u5f15\u64ce\u7684\u72b6\u6001\u540c\u6b65"),(0,l.kt)("h3",{id:"\u91ca\u4e49\u67e5\u8be2"},"\u91ca\u4e49\u67e5\u8be2"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u5728\u67e5\u8be2\u5355\u8bcd\u91ca\u4e49\u65f6\u4f7f\u7528\u672c\u5730 sqlite \u6570\u636e\u5e93\uff0c\u6570\u636e\u5e93\u6e90\u4e8e ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/skywind3000/ECDICT-ultimate"},"ecdict-ultimate")),(0,l.kt)("p",null,"\u9274\u4e8e\u5f00\u6e90\u6570\u636e\u5e93\u8fc7\u4e8e\u5de8\u5927\uff0c\u4e14\u5b58\u5728\u4e00\u4e9b\u672c\u7a0b\u5e8f\u65e0\u9700\u4f7f\u7528\u7684 columns\uff0c\u6211\u53c8\u4f7f\u7528 dart \u548c ",(0,l.kt)("a",{parentName:"p",href:"https://pub.dev/packages/drift"},"drift")," \u5bf9\u5176\u8fdb\u884c\u4e86\u6e05\u6d17\u548c\u526a\u88c1"),(0,l.kt)("p",null,"\u5728\u67e5\u8be2\u65f6\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5c06\u67e5\u8be2\u7684\u672c\u6587\u4e32\u201c\u6253\u6563\u201d\uff0c\u5e76\u4e00\u9f50\u8bf7\u6c42\u6570\u636e\u5e93\uff0c\u4ee5\u671f\u80fd\u547d\u4e2d\u4e0d\u65ad\u5343\u53d8\u4e07\u5316\u7684\u82f1\u8bed\u5355\u8bcd\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'final sequence = "qwertyuiopasdfghjkl";\nfinal sequences = ["qwe","wer","ert",...,"qwer","wert",...,"qwert",...,"qwertyuiopasdfghjkl"];\nfinal List results = await queryDB(keys:sequences); // average latency: ~6ms\n')),(0,l.kt)("p",null,"\u5728\u8bf7\u6c42\u5b8c\u6210\u540e\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5bf9\u8bf7\u6c42\u7ed3\u679c\u8fdb\u884c\u53bb\u91cd\uff0c\u903b\u8f91\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b owl (\u732b\u5934\u9e70)"),(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b knowledge(\u77e5\u8bc6)"),(0,l.kt)("li",{parentName:"ul"},'\u5e8f\u5217 "knowledge" \u5305\u542b "owl"\uff0c\u79fb\u9664\u67e5\u8be2\u7ed3\u679c\u4e2d\u7684 owl')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u76ee\u524d\uff0c\u8be5\u903b\u8f91\u8fd8\u6ca1\u6709\u5b9e\u73b0\u5bf9\u51b2\u7a81\u7684\u5224\u5b9a\uff1a\u5982\u5411\u6570\u636e\u5e93\u67e5\u8be2\u7684 key \u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'"gitignore"'),"\uff0c\u67e5\u8be2\u7ed3\u679c\u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'["git", "tig", "ignore"]'),"\uff0c\u76ee\u524d\u8fd9\u4e09\u4e2a\u7ed3\u679c\u90fd\u4f1a\u88ab\u6e32\u67d3\u5230 HUD \u4e0a\u3002\u4f46\u663e\u7136\uff0c\u9009\u53d6 git \u548c ignore \u8fd9\u4e24\u4e2a\u67e5\u8be2\u7ed3\u679c\u9020\u6210\u7684 \u201c\u51b2\u7a81\u201d \u662f\u6700\u5c0f\u7684")),(0,l.kt)("h3",{id:"\u53d1\u97f3"},"\u53d1\u97f3"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/avfoundation/speech_synthesis"},"AVFoundation - Speech synthesis")," API \u6765\u5b9e\u65f6\u5408\u6210\u8bed\u97f3"),(0,l.kt)("p",null,"\u8be5 API \u53ef\u4ee5\u540c\u65f6\u5728 iOS/macOS \u4e0a\u4f7f\u7528\uff0c\u4e14\u53d1\u97f3\u8f83\u4e3a\u51c6\u786e\uff0c\u4ee5\u6211\u81ea\u8eab\u7684\u80fd\u529b(\u5168\u56fd\u5927\u5b66\u82f1\u8bed\u56db\u7ea7)\u6765\u770b\uff0c\u6548\u679c\u8fd8\u7b97\u6ee1\u610f\uff0c\u786e\u5b9e\u6bd4\u6211\u7684\u53d1\u97f3\u51c6 \ud83e\udd23"),(0,l.kt)("p",null,"\u5f53\u7136\uff0c\u540e\u7ee7\u5982\u679c\u6709\u66f4\u9ad8\u7684\u9700\u6c42\uff0c\u4e5f\u6709\u5f88\u591a\u5176\u4ed6\u53ef\u9009\u65b9\u6848"))}c.isMDXComponent=!0},3591:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/ocr_text_source-9af654c22d465590f3079fd8d4f2f24f.png"},1589:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/appkit_screens_coordinate-94563877ba8462b8412c6e945e1f7b5e.png"},6521:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/dashboard_inspect-0a3c0a7bba8e62a06a697e1555bb3ea4.png"},2731:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/intro-663e889360c7a0e605df52af33af3b11.gif"},8555:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_dark-ede87b48b4273a15fc0974c77b5586cf.png"},1974:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_light-980bb1be0d553350c1dcf22e0b4325e2.png"},3642:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/project_layout-b9e52932db96541ab24f4ed06bc28bc0.png"},9462:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation-e7dc02491d5b436f2ef66bd928e65e17.png"},6607:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation_1-7c6db883d1b12b61355048e553101f21.png"}}]); \ No newline at end of file diff --git a/assets/js/ebb011f4.c91967e2.js b/assets/js/ebb011f4.c91967e2.js deleted file mode 100644 index e4a10bd..0000000 --- a/assets/js/ebb011f4.c91967e2.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdf_doc_source=self.webpackChunkdf_doc_source||[]).push([[649],{3905:(e,t,n)=>{n.d(t,{Zo:()=>d,kt:()=>k});var a=n(7294);function l(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(l[n]=e[n]);return l}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var p=a.createContext({}),s=function(e){var t=a.useContext(p),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},d=function(e){var t=s(e.components);return a.createElement(p.Provider,{value:t},e.children)},u="mdxType",c={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,l=e.mdxType,r=e.originalType,p=e.parentName,d=o(e,["components","mdxType","originalType","parentName"]),u=s(n),m=l,k=u["".concat(p,".").concat(m)]||u[m]||c[m]||r;return n?a.createElement(k,i(i({ref:t},d),{},{components:n})):a.createElement(k,i({ref:t},d))}));function k(e,t){var n=arguments,l=t&&t.mdxType;if("string"==typeof e||l){var r=n.length,i=new Array(r);i[0]=m;var o={};for(var p in t)hasOwnProperty.call(t,p)&&(o[p]=t[p]);o.originalType=e,o[u]="string"==typeof e?e:l,i[1]=o;for(var s=2;s{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>i,default:()=>c,frontMatter:()=>r,metadata:()=>o,toc:()=>s});var a=n(7462),l=(n(7294),n(3905));const r={slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},i=void 0,o={permalink:"/valo-reader-doc/blog/tech_detail_macos",editUrl:"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/tech_detail_macos.md",source:"@site/blog/tech_detail_macos.md",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",description:"\u6458\u8981",date:"2023-10-19T05:59:23.000Z",formattedDate:"2023\u5e7410\u670819\u65e5",tags:[{label:"technical",permalink:"/valo-reader-doc/blog/tags/technical"}],readingTime:23.595,hasTruncateMarker:!1,authors:[{name:"Ce Wang",title:"Solo dev",url:"https://github.com/haloWang",imageURL:"https://github.com/haloWang.png",key:"halowang"}],frontMatter:{slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},nextItem:{title:"\u5f00\u53d1\u76ee\u7684",permalink:"/valo-reader-doc/blog/motivation"}},p={authorsImageUrls:[void 0]},s=[{value:"\u6458\u8981",id:"\u6458\u8981",level:2},{value:"\u4ec0\u4e48\u662f Valo Reader?",id:"\u4ec0\u4e48\u662f-valo-reader",level:2},{value:"\u9879\u76ee\u5e03\u5c40",id:"\u9879\u76ee\u5e03\u5c40",level:2},{value:"\u5de5\u7a0b\u67b6\u6784",id:"\u5de5\u7a0b\u67b6\u6784",level:2},{value:"flutter \u4fa7",id:"flutter-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f",id:"\u4e3b\u7a0b\u5e8f",level:4},{value:"flutter package: \u6587\u672c\u5757\u8868\u5f81",id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81",level:4},{value:"flutter package: \u57fa\u7840\u5e93",id:"flutter-package-\u57fa\u7840\u5e93",level:4},{value:"native \u4fa7",id:"native-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",level:4},{value:"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93",id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93",level:4},{value:"CocoaPods dependency: OCR \u529f\u80fd\u5e93",id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93",level:4},{value:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",level:2},{value:"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)",id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud",level:3},{value:"\u5728 HUD \u4e2d\u7684\u72b6\u6001",id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001",level:4},{value:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)",id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard",level:3},{value:"\u539f\u751f\u4fa7 (native)",id:"\u539f\u751f\u4fa7-native",level:3},{value:"\u6280\u672f\u7ec6\u8282",id:"\u6280\u672f\u7ec6\u8282",level:2},{value:"\u591a\u5c4f\u5e55",id:"\u591a\u5c4f\u5e55",level:3},{value:"\u5c4f\u5e55\u6355\u6349",id:"\u5c4f\u5e55\u6355\u6349",level:3},{value:"OCR",id:"ocr",level:3},{value:"\u6027\u80fd\u8868\u73b0",id:"\u6027\u80fd\u8868\u73b0",level:4},{value:"\u8282\u80fd\u4f18\u5316",id:"\u8282\u80fd\u4f18\u5316",level:4},{value:"\u4e0d\u622a\u53d6\u81ea\u5df1",id:"\u4e0d\u622a\u53d6\u81ea\u5df1",level:4},{value:"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d",id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d",level:3},{value:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",level:3},{value:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",level:4},{value:"\u91ca\u4e49\u67e5\u8be2",id:"\u91ca\u4e49\u67e5\u8be2",level:3},{value:"\u53d1\u97f3",id:"\u53d1\u97f3",level:3}],d={toc:s},u="wrapper";function c(e){let{components:t,...r}=e;return(0,l.kt)(u,(0,a.Z)({},d,r,{components:t,mdxType:"MDXLayout"}),(0,l.kt)("h2",{id:"\u6458\u8981"},"\u6458\u8981"),(0,l.kt)("p",null,"\u672c\u6587\u4e3b\u8981\u4ecb\u7ecd\u4e86 Valo Reader for macOS\uff08\u4e0b\u6587\u7b80\u79f0\u4e3a\u201c\u672c\u7a0b\u5e8f\u201d\uff09\u7684",(0,l.kt)("a",{parentName:"p",href:"#%E9%A1%B9%E7%9B%AE%E5%B8%83%E5%B1%80"},"\u9879\u76ee\u5e03\u5c40"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E5%B7%A5%E7%A8%8B%E6%9E%B6%E6%9E%84"},"\u5de5\u7a0b\u67b6\u6784"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E8%BF%90%E8%A1%8C%E6%97%B6%E5%8A%9F%E8%83%BD%E6%A8%A1%E5%9D%97"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),"\u4e0e",(0,l.kt)("a",{parentName:"p",href:"#%E6%8A%80%E6%9C%AF%E7%BB%86%E8%8A%82"},"\u6280\u672f\u7ec6\u8282"),"\u3002\u4f60\u8fd8\u53ef\u4ee5\u5728",(0,l.kt)("a",{parentName:"p",href:"/docs/intro"},"\u672c\u6587\u6863"),"\u7684\u5176\u4ed6\u9875\u9762\u4e86\u89e3\u672c\u7a0b\u5e8f\u7684",(0,l.kt)("a",{parentName:"p",href:"/valo-reader-doc/blog/motivation"},"\u5f00\u53d1\u52a8\u673a"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/usage"},"\u4f7f\u7528\u65b9\u5f0f"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/privacy_security"},"\u9690\u79c1\u4e0e\u5b89\u5168"),"\u7b49\u5185\u5bb9"),(0,l.kt)("h2",{id:"\u4ec0\u4e48\u662f-valo-reader"},"\u4ec0\u4e48\u662f Valo Reader?"),(0,l.kt)("p",null,"Valo Reader \u8fd9\u4e2a\u9879\u76ee\u662f\u6211\u4e2a\u4eba\u5728\u65e5\u5e38\u751f\u6d3b\u4e2d\u9010\u6e10\u840c\u751f\uff0c\u6f14\u5316\u548c\u5b9e\u65bd\u7684\u60f3\u6cd5\u3002\u662f\u6211\u4e3a\u4e86\u5728\u81ea\u5df1\u7535\u5b50\u8bbe\u5907\u8bbe\u5907\u4e0a\u80fd\u8f7b\u677e\u9605\u8bfb\u82f1\u6587\u5185\u5bb9\uff0c\u5e76\u6e10\u8fd1\u5f0f\u5730\u63d0\u9ad8\u81ea\u5df1\u7684\u82f1\u6587\u6c34\u5e73\uff0c\u800c\u81ea\u884c\u8bbe\u8ba1\u548c\u7814\u53d1\u7684\u4e00\u6b3e\u4ea7\u54c1\uff0c\u5176\u6838\u5fc3\u529f\u80fd\u5c55\u793a\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"intro",src:n(2731).Z,width:"960",height:"268"})),(0,l.kt)("p",null,"\u5f53\u4f60\u5728\u4f7f\u7528\u672c\u7a0b\u5e8f\u65f6\uff0c\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u4f60\u611f\u5230\u751f\u758f\u7684\u82f1\u8bed\u5355\u8bcd\u4e0a\u3002\u6309\u4e0b\u6307\u5b9a\u7684\u6309\u952e\uff08\u9ed8\u8ba4\u4e3a Fn\uff09\uff0c\u672c\u7a0b\u5e8f\u5373\u4f1a\u5728\u54cd\u5e94\u7684\u4f4d\u7f6e\u5c55\u793a\u5bf9\u5e94\u7684\u91ca\u4e49\uff0c\u540c\u65f6\u4f7f\u7528\u8bed\u97f3\u5408\u6210\u6765\u53d1\u97f3\u3002"),(0,l.kt)("p",null,"\u76ee\u524d\uff0c\u4f60\u53ef\u4ee5\u901a\u8fc7 ",(0,l.kt)("a",{parentName:"p",href:"https://apps.apple.com/cn/app/valo-reader/id6448040931"},"Mac App Store")," \u6216",(0,l.kt)("a",{parentName:"p",href:"/docs/installation"},"\u5176\u4ed6\u65b9\u5f0f"),"\u6765\u83b7\u53d6\u672c\u7a0b\u5e8f"),(0,l.kt)("h2",{id:"\u9879\u76ee\u5e03\u5c40"},"\u9879\u76ee\u5e03\u5c40"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u9879\u76ee\u7684\u4e3b\u4f53\u7ed3\u6784\uff1a\n",(0,l.kt)("img",{alt:"project_layout",src:n(3642).Z,width:"1996",height:"960"})),(0,l.kt)("p",null,"\u672c\u6587\u7684\u4f4d\u7f6e\u5219\u5904\u4e8e\u4e0a\u56fe\u4e2d\u7684\u6d45\u9ec4\u8272\u90e8\u5206\uff0c\u7531 ",(0,l.kt)("a",{parentName:"p",href:"https://docusaurus.io/"},"docusaurus")," \u751f\u6210"),(0,l.kt)("p",null,"\u672c\u9879\u76ee\u7684\u60f3\u6cd5\u6700\u65e9\u5b9e\u73b0\u4e8e\u5728\u6d4f\u89c8\u5668\u4e2d\u8fd0\u884c\u7684 js \u811a\u672c\uff0c\u4f60\u53ef\u4ee5\u5728 github \u4e0a\u770b\u5230\u672c\u9879\u76ee\u5728\u6d4f\u89c8\u5668\u4e0a\u7684\u65e9\u671f\u5b9e\u73b0\u53ca\u5bf9\u5e94\u7684",(0,l.kt)("a",{parentName:"p",href:"https://github.com/HaloWang/english_flow#%E5%8A%9F%E8%83%BD%E5%B1%95%E7%A4%BA"},"\u529f\u80fd\u5c55\u793a"),"\uff0c\u5373\u56fe\u4e2d\u6d45\u7eff\u8272\u90e8\u5206"),(0,l.kt)("p",null,"\u800c\u672c\u6587\u4e3b\u8981\u8bf4\u660e\u4e86\u4e0a\u56fe\u4e2d macOS\uff08\u6d45\u84dd\u8272\u90e8\u5206\uff09\u7684\u8fd0\u884c\u65b9\u5f0f\u4e0e\u6280\u672f\u7ec6\u8282"),(0,l.kt)("h2",{id:"\u5de5\u7a0b\u67b6\u6784"},"\u5de5\u7a0b\u67b6\u6784"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create XXX --platform macos")," \u547d\u4ee4\u521b\u5efa\uff0c\u5176\u6e90\u4ee3\u7801\u4e3b\u8981\u5206\u4e3a\u4e24\u90e8\u5206"),(0,l.kt)("h3",{id:"flutter-\u4fa7"},"flutter \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 dart \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f"},"\u4e3b\u7a0b\u5e8f"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u7684\u4e3b\u8981\u529f\u80fd\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u901a\u8fc7 method channel \u5b9e\u73b0 flutter \u4e0e native \u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u7528\u6237\u8bbe\u7f6e\u9762\u677f\uff08Dashboard\uff09"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\u4e0e\u8bca\u65ad\u68c0\u67e5\u4fe1\u606f"),(0,l.kt)("li",{parentName:"ol"},"\u4f7f\u7528\u7b2c\u4e09\u65b9\u4f9d\u8d56\u63d0\u4f9b\u7684\u529f\u80fd")),(0,l.kt)("h4",{id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81"},"flutter package: \u6587\u672c\u5757\u8868\u5f81"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u662f\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u4ee3\u7801\uff0c\u5728 macOS / iOS / Android \u4e09\u7aef\u5171\u4eab\u90e8\u5206\u903b\u8f91\u4e0e UI\uff0c\u5176\u4e3b\u8981\u529f\u80fd\u7531\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u89e3\u6790\u7528\u6237\u8bbe\u5907\u5c4f\u5e55\u4e0a\u8bc6\u522b\u5230\u7684\u6587\u672c\u5757\uff0c\u7ed3\u6784\u5316\u6587\u672c\u5757\uff0c\u5e76\u5c06\u5176\u7ef4\u62a4\u5230\u672c\u7a0b\u5e8f\u7684\u5185\u5b58\u4e2d"),(0,l.kt)("li",{parentName:"ol"},"\u58f0\u660e\u5e76\u5b9e\u73b0\u7528\u6237\u89c6\u89c9\u7126\u70b9\u4e0e\u5df2\u77e5\u7684\u6587\u672c\u4fe1\u606f\u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09")),(0,l.kt)("h4",{id:"flutter-package-\u57fa\u7840\u5e93"},"flutter package: \u57fa\u7840\u5e93"),(0,l.kt)("p",null,"\u5728\u6211\u7684\u6240\u6709 flutter \u9879\u76ee\u4e2d\u5171\u4eab\u7684\u57fa\u7840\u4f9d\u8d56"),(0,l.kt)("h3",{id:"native-\u4fa7"},"native \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 swift \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206"},"\u4e3b\u7a0b\u5e8f\u90e8\u5206"),(0,l.kt)("p",null,"\u6301\u6709 flutter engine\uff0c\u901a\u8fc7 method channel \u6253\u901a flutter \u4e0e ",(0,l.kt)("inlineCode",{parentName:"p"},"Base"),"\u3001",(0,l.kt)("inlineCode",{parentName:"p"},"OCR")," \u4ee5\u53ca\u5e94\u7528\u7a0b\u5e8f\u672c\u8eab\u7684\u53cc\u5411\u8c03\u7528"),(0,l.kt)("p",null,"\u521b\u5efa\u9879\u76ee\u65f6\u9ed8\u8ba4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController")," \u5df2\u88ab\u5220\u9664"),(0,l.kt)("h4",{id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93"},"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e Cocoa framework \u4ea4\u4e92\uff0c\u5982\uff1a\u6743\u9650\u7533\u8bf7\uff0c\u952e\u76d8\u76d1\u542c\uff0c\u9f20\u6807\u76d1\u542c\uff0c\u622a\u53d6\u5c4f\u5e55\uff0c\u79fb\u52a8\u627f\u8f7d flutter \u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow"),"\u4ee5\u53ca\u63d0\u4f9b\u57fa\u672c\u82e5\u5e72\u539f\u751f\u80fd\u529b"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u6211\u4e5f\u4f1a\u5728\u8be5\u4f9d\u8d56\u4e2d\u5b66\u4e60\uff0c\u5c1d\u8bd5\u548c\u5b9e\u73b0 Cocoa/AppKit \u72ec\u6709\u7684 API \u4e0e\u529f\u80fd"),(0,l.kt)("h4",{id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93"},"CocoaPods dependency: OCR \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Apple Vision framework - Text Recognizing")," \u4ea4\u4e92\uff0c\u5e76\u5b9e\u73b0\u90e8\u5206\u903b\u8f91\uff0c\u5982\uff1a\u53d1\u8d77\u6587\u672c\u8bc6\u522b\u8bf7\u6c42\uff0c\u7ef4\u62a4\u6587\u672c\u8bc6\u522b\u54cd\u5e94\u7f13\u5b58"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u5728 macOS \u548c iOS \u9879\u76ee\u540c\u65f6\u4f9d\u8d56\u5e76\u5171\u4eab\u4ee3\u7801"),(0,l.kt)("h2",{id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0c\u672c\u7a0b\u5e8f\u4e3b\u8981\u53ef\u5206\u4e3a\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\u3001\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD) \u548c\u539f\u751f\u4fa7 (native) \u4e09\u4e2a\u6a21\u5757"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u7684\u4e3b\u8981\u6a21\u5757\u53ca\u5176\u901a\u8baf\uff1a\n",(0,l.kt)("img",{alt:"modules_light",src:n(1974).Z+"#gh-light-mode-only",width:"1926",height:"894"}),"\n",(0,l.kt)("img",{alt:"modules_dark",src:n(8555).Z+"#gh-dark-mode-only",width:"1926",height:"894"})),(0,l.kt)("h3",{id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud"},"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)"),(0,l.kt)("p",null,"HUD \u662f\u4e00\u4e2a\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u81ea\u5df1\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterEngine")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")),(0,l.kt)("p",null,"HUD \u7684\u4e3b\u8981\u4efb\u52a1\u5305\u542b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u5411 native \u6d3e\u53d1 ocr \u8bf7\u6c42"),(0,l.kt)("li",{parentName:"ul"},"\u89e3\u6790 native \u5bf9\u5c4f\u5e55\u6307\u5b9a\u533a\u57df\u7684 ocr \u7ed3\u679c\uff0c\u5e76\u5728\u5185\u5b58\u4e2d\u7ef4\u62a4\u8be5\u7ed3\u679c\u4f9b\u540e\u7ee7\u7a0b\u5e8f\u903b\u8f91\u4f7f\u7528"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u4ee5\u5728\u9700\u8981\u65f6\u8ba1\u7b97\u91ca\u4e49\u5c55\u793a\u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u952e\u76d8\u6309\u952e\u72b6\u6001\uff0c\u5728\u53d8\u5316\u65f6\u6d3e\u53d1 ocr \u8bf7\u6c42\u5e76\u5c55\u793a\u5355\u8bcd\u91ca\u4e49"),(0,l.kt)("li",{parentName:"ul"},"\u5b9e\u65f6\u540c\u6b65 Dashboard engine \u4f20\u9012\u8fc7\u6765\u7684\u7528\u6237\u8bbe\u7f6e\uff0c\u5e76\u4fee\u6539\u5c55\u793a\u903b\u8f91")),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u8be5\u6a21\u5757\u6240\u7ef4\u62a4\u7684\u6570\u636e\u6620\u5c04\u5230\u5c4f\u5e55\u4e0a\u65f6\u7684\u53ef\u89c6\u5316\u6548\u679c\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation",src:n(9462).Z,width:"1898",height:"222"})),(0,l.kt)("h4",{id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001"},"\u5728 HUD \u4e2d\u7684\u72b6\u6001"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u7ba1\u7406\u7edd\u5927\u90e8\u5206\u7684\u72b6\u6001\u3002\u5728\u6784\u7b51\u672c\u7a0b\u5e8f\u65f6\uff0c\u6211\u7684\u601d\u8003\u8fc7\u7a0b\u4e3b\u8981\u57fa\u4e8e\u72b6\u6001\u4e0e\u72b6\u6001\u53d8\u5316\uff0c\u4ece\u5c55\u793a\u91ca\u4e49\u952e\u70b9\u51fb\u5230\u6e32\u67d3\u91ca\u4e49\u9762\u677f\u52a8\u753b"),(0,l.kt)("p",null,"\u5728\u5927\u91cf\u4f7f\u7528 riverpod \u4e2d\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"Provider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"ProviderContainer.listen")," \u540e\uff0c\u6211\u53ef\u4ee5\u6784\u5efa\u4e00\u4e2a\u9ad8\u5ea6\u5f02\u6b65\uff0c\u8c03\u7528\u987a\u5e8f\u4e0d\u654f\u611f\uff0c\u72b6\u6001\u53d8\u5316\u9a71\u52a8\u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u5728\u7f16\u7801\u8fc7\u7a0b\u4e2d\uff0c\u8fd9\u79cd\u65b9\u5f0f\u8ba9\u7a0b\u5e8f\u5458\u53ef\u4ee5\u4ece\u4e00\u5927\u4e32\u7684\u547d\u4ee4\u5f0f(Imperative)\u65b9\u6cd5\u8c03\u7528\u4e2d\u89e3\u653e\u51fa\u6765\uff0c\u4ec5\u4ec5\u9700\u8981\u5173\u6ce8\u548c\u786e\u4fdd\u6bcf\u4e2a\u6700\u5c0f\u903b\u8f91\u5355\u5143\u2014\u2014",(0,l.kt)("inlineCode",{parentName:"p"},"Provider"),"\uff0c\u786e\u4fdd\u5176\u5b9e\u73b0\u662f\u6b63\u786e\u7684\u5373\u53ef\uff0c\u8fd9\u51cf\u5c11\u4e86\u6784\u5efa\u590d\u6742\u548c\u957f\u4e32\u903b\u8f91\u51fa\u9519\u7684\u53ef\u80fd"),(0,l.kt)("p",null,"\u4e0b\u9762\u7684\u4e24\u5f20\u56fe\u8868\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u7684\u4e3b\u8981\u529f\u80fd\u6d41\u7a0b"),(0,l.kt)("p",null,"\u5207\u6362\u5c55\u793a\u91ca\u4e49\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:'graph LR\n A1[\u76d1\u542c\u952e\u76d8\u72b6\u6001];\n A[\u952e\u76d8\u72b6\u6001\u53d1\u751f\u6539\u53d8];\n query_button_down{\u5c55\u793a\u91ca\u4e49\u952e\u6309\u4e0b?};\n C[\u505c\u6b62\u5b9a\u65f6\u5668];\n D[\u542f\u52a8\u5b9a\u65f6\u5668\u5e76\u91cd\u7f6e\u6587\u672c\u5757\u72b6\u6001];\n mouse_state[\u9f20\u6807\u72b6\u6001];\n screen_state[\u5c4f\u5e55\u72b6\u6001];\n send_request_to_native["\u5411 native \u53d1\u9001\u622a\u56fe\u4e0e OCR \u8bf7\u6c42"];\n F[native OCR \u7ed3\u679c\u5df2\u83b7\u5f97];\n G[\u89e3\u6790 OCR \u7ed3\u679c];\n H1[\u66f4\u65b0\u6587\u672c\u5757\u72b6\u6001];\n\n mouse_state--\x3esend_request_to_native\n screen_state--\x3esend_request_to_native\n send_request_to_native-.->F;\n F--\x3eG;\n G--\x3eH1;\n A1-.->A;\n A--\x3equery_button_down;\n query_button_down--\x3e|false|C;\n query_button_down--\x3e|true|D;\n D-.->send_request_to_native;'}),(0,l.kt)("p",null,"\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u53d8\u66f4\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:"graph LR;\n H2[\u6587\u672c\u5757\u72b6\u6001\u53d8\u66f4];\n most_wanted_state_logic[\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u76d1\u542c\u903b\u8f91];\n most_wanted_state_changed[\u72b6\u6001\u6539\u53d8];\n mouse_change0[\u76d1\u542c\u9f20\u6807\u72b6\u6001];\n mouse_change[\u9f20\u6807\u72b6\u6001\u6539\u53d8];\n sync_mouse_to_flutter[\u540c\u6b65\u9f20\u6807\u72b6\u6001\u81f3 flutter];\n sync_current_screen[\u540c\u6b65\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u6570\u636e];\n sync_current_screen_to_flutter[\u540c\u6b65\u5c4f\u5e55\u6570\u636e\u81f3 flutter];\n hasTextBlock{\u72b6\u6001\u7684\u503c\u662f\u5426\u4e3a\u7a7a\\n\u5373\u9f20\u6807\u5f53\u524d\u4f4d\u7f6e\u6ca1\u6709\u8bc6\u522b\u5230\u6587\u672c};\n queryDB[\u67e5\u8be2\u6570\u636e\u5e93];\n dbHasData{\u6570\u636e\u5e93\u5305\u542b\u91ca\u4e49};\n hideAllFlow[\u9690\u85cf\u6240\u6709\u91ca\u4e49\u9762\u677f];\n asParam[\u4f5c\u4e3aOCR\u8bf7\u6c42\u7684\u53c2\u6570];\n screen_changed{\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u53d1\u751f\u53d8\u52a8?}\n\n \u5207\u6362\u5c55\u793a\u91ca\u4e49-.->H2\n mouse_change0-.->mouse_change--\x3esync_mouse_to_flutter;\n mouse_change --\x3e sync_current_screen --\x3e sync_current_screen_to_flutter;\n sync_current_screen_to_flutter --\x3e \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001\n sync_current_screen --\x3e screen_changed --\x3e |true| \u6539\u53d8\u627f\u8f7dHUD\u7684NSPanel\u4f4d\u7f6e;\n screen_changed --\x3e |false| return_0;\n H2 --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter -.-> asParam;\n \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001 -.-> asParam;\n most_wanted_state_logic --\x3e most_wanted_state_changed --\x3e hasTextBlock;\n \u5c55\u793a\u91ca\u4e49\u952e\u72b6\u6001 --\x3e most_wanted_state_logic;\n most_wanted_state_changed --\x3e hideAllFlow;\n hasTextBlock --\x3e |false| return_1;\n hasTextBlock --\x3e |true| queryDB;\n queryDB --\x3e dbHasData;\n dbHasData --\x3e |true| \u6e32\u67d3\u5c55\u793a\u91ca\u4e49\u9762\u677f\u52a8\u753b\n dbHasData --\x3e |false| return_2;"}),(0,l.kt)("h3",{id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard"},"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)"),(0,l.kt)("p",null,"Dashboard \u662f\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPanel")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u4e0d\u540c\u4e8e HUD \u7684\u72ec\u7acb engine \u4e0e method channel"),(0,l.kt)("p",null,"Dashboard \u7684\u4e3b\u8981\u4efb\u52a1\u662f\u4e3a\u7528\u6237\u63d0\u4f9b\u63a7\u5236\u672c\u7a0b\u5e8f\u7684 UI\uff0c\u5305\u62ec\u5feb\u6377\u952e\u8bbe\u7f6e\uff0c\u5f00\u673a\u81ea\u542f\uff0c\u5173\u4e8e\u672c\u7a0b\u5e8f\u7b49\u5e38\u89c1\u7528\u6237\u4ea4\u4e92"),(0,l.kt)("p",null,"\u540c\u65f6\uff0cDashboard \u4f1a\u7ef4\u62a4\u672c\u7a0b\u5e8f\u5728\u5185\u5b58\u548c\u786c\u76d8\u4e2d\u7684\u72b6\u6001\uff0c\u5e76\u5c06\u8fd9\u4e2a\u72b6\u6001\u901a\u8fc7 method channel \u540c\u6b65\u81f3\u4e0b\u9762\u5373\u5c06\u8981\u8bb2\u5230\u7684\u91ca\u4e49\u5c55\u793a\u9762\u677f"),(0,l.kt)("h3",{id:"\u539f\u751f\u4fa7-native"},"\u539f\u751f\u4fa7 (native)"),(0,l.kt)("p",null,"\u539f\u751f\u4fa7\u662f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create")," \u547d\u4ee4\u521b\u5efa\u51fa\u6765\u7684\u4f20\u7edf Xcode \u5de5\u7a0b"),(0,l.kt)("p",null,"\u5728\u539f\u751f\u4fa7\uff0c\u6211\u4e3b\u8981\u5173\u6ce8\u7684\u95ee\u9898\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u5904\u7406\u6765\u81ea ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u7684\u8c03\u7528\uff0c\u5e76\u5728\u9700\u8981\u65f6\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterResult")," \u54cd\u5e94\u5bf9\u5e94\u7684 flutter engine"),(0,l.kt)("li",{parentName:"ul"},"\u7ef4\u62a4\u627f\u8f7d HUD \u548c Dashboard \u7684\u4e24\u4e2a ",(0,l.kt)("inlineCode",{parentName:"li"},"NSWindow"),"\uff0c\u5c24\u5176\u662f\u5f53\u7528\u6237\u7684\u8bbe\u5907\u540c\u65f6\u94fe\u63a5\u591a\u4e2a\u5c4f\u5e55\u65f6\uff0c\u6b63\u786e\u5730\u8bbe\u7f6e HUD \u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u5c01\u88c5\u5e76\u5411 flutter \u63d0\u4f9b\u80fd\u529b\uff0c\u5982\uff1a",(0,l.kt)("ul",{parentName:"li"},(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u53d8\u5316\u548c\u952e\u76d8\u72b6\u6001\u53d8\u5316\uff0c\u5e76\u5c06\u72b6\u6001\u540c\u6b65\u81f3 flutter"),(0,l.kt)("li",{parentName:"ul"},"\u4e3a flutter \u63d0\u4f9b OCR \u548c\u8bed\u97f3\u5408\u6210\u80fd\u529b"),(0,l.kt)("li",{parentName:"ul"},"\u5728\u6267\u884c OCR \u65f6\u4f7f\u7528 swift \u5b9e\u73b0\u5fc5\u8981\u7684\u6027\u80fd\u4f18\u5316")))),(0,l.kt)("h2",{id:"\u6280\u672f\u7ec6\u8282"},"\u6280\u672f\u7ec6\u8282"),(0,l.kt)("p",null,"\u5f97\u76ca\u4e8e Apple \u4e00\u8109\u76f8\u627f\u7684 API \u8bbe\u8ba1\u4ee5\u53ca\u540c\u6837\u7684\u7f16\u7a0b\u8bed\u8a00\uff0c\u719f\u7ec3\u4e8e iOS \u5e94\u7528\u5f00\u53d1(Cocoa Touch & UIKit)\u7684\u7a0b\u5e8f\u5458\u5728\u9762\u5bf9\u4e0e macOS \u4ea4\u4e92\u7684 Cocoa \u548c AppKit \u6846\u67b6\u65f6\uff0c\u53ef\u4ee5\u590d\u7528\u5f88\u591a\u601d\u60f3\u4e0e\u903b\u8f91"),(0,l.kt)("p",null,"\u4f46\u76f8\u6bd4\u4e8e iOS\uff0cmacOS \u7ed9\u4e86\u7528\u6237\u66f4\u5927\u7684\u821e\u53f0\uff0c\u4e5f\u8ba9\u5f00\u53d1\u8005\u9762\u5bf9\u4e86\u66f4\u591a\u7684\u6311\u6218: \u9f20\u6807\uff0c\u952e\u76d8\uff0c\u7a97\u53e3\u548c\u5c4f\u5e55"),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u672c\u7a0b\u5e8f\u7684\u8fc7\u7a0b\u4e2d\uff0c\u6211\u9047\u5230\u4e86\u8bf8\u591a\u7684\u56f0\u96be\uff0c\u4e5f\u5728\u8305\u585e\u987f\u5f00\u65f6\u6536\u83b7\u4e86\u5f88\u591a\u5feb\u4e50\uff0c\u6211\u5728\u8fd9\u4e00\u7ae0\u8282\u4f1a\u8bb0\u5f55\u4e00\u4e0b"),(0,l.kt)("h3",{id:"\u591a\u5c4f\u5e55"},"\u591a\u5c4f\u5e55"),(0,l.kt)("p",null,"macOS \u53ca\u5176\u8fd0\u884c\u7684\u786c\u4ef6\u8bbe\u5907\u5e38\u5e38\u8fde\u63a5\u7740\u591a\u5757\u5c4f\u5e55\uff0c\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u529f\u80fd\u5c31\u662f\u5728\u4efb\u610f\u7684\u5c4f\u5e55\u4f4d\u7f6e\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\uff0c\u8fd9\u5c31\u8981\u786e\u4fdd\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\u653e\u7f6e\u4e8e\u6b63\u786e\u7684\u4f4d\u7f6e\u4e0a"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u8bf7\u6c42 native \u6355\u6349\u5c4f\u5e55\u65f6\uff0c\u6355\u6349\u7684\u77e9\u5f62\u6846\u7684\u4f4d\u7f6e\u6b63\u786e"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u89e3\u6790 ocr response \u540e\uff0c\u7ed3\u679c\u53ef\u4ee5\u6b63\u786e\u5730\u548c\u5c4f\u5e55\u4e0a\u771f\u6b63\u7684\u5185\u5bb9\u5efa\u7acb\u6620\u5c04")),(0,l.kt)("p",null,"\u548c UIKit \u4e2d\u5e38\u7528\u7684 CGRect \u4e0d\u540c\uff0cAppKit \u5bf9\u8bbe\u5907\u5c4f\u5e55\u7684\u62bd\u8c61 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen"),"\uff0c\u5176\u5750\u6807\u4ee5 NSRect \u8ba1\u7b97\uff0c\u5750\u6807\u7cfb\u539f\u70b9\u4e3a macOS \u539f\u59cb\u5c4f\u5e55\u7684\u5de6\u4e0b\u89d2\u3002\u800c\u6b64\u65f6\uff0c\u5982\u679c\u4f60\u7ed9\u4f60\u7684\u8bbe\u5907\u8fde\u63a5\u4e0a\u4e86\u5176\u4ed6\u7684\u5c4f\u5e55\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen.screens")," \u6240\u5448\u73b0\u5c4f\u5e55\u5e03\u5c40\uff0c\u53ef\u80fd\u5c31\u4f1a\u53d8\u6210\u4e0b\u56fe\u6240\u793a\u7684\u6837\u5b50\uff1a\n",(0,l.kt)("img",{alt:"appkit_screens_coordinate",src:n(1589).Z,width:"1630",height:"734"})),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u7684\u5de6\u4e0b\u89d2\u5750\u6807\u503c\u5e76\u975e\u662f (0, 0)\uff0c\u800c\u662f ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 0")," \u7684 (0, 0) \u70b9\u7684\u76f8\u5bf9\u4f4d\u7f6e (w",(0,l.kt)("sub",null,"1"),", d",(0,l.kt)("sub",null,"2"),")\u3002\u901a\u8fc7\u76d1\u542c\u9f20\u6807\u79fb\u52a8\u4e8b\u4ef6\u83b7\u53d6\u7684\u9f20\u6807\u4f4d\u7f6e ",(0,l.kt)("inlineCode",{parentName:"p"},"event.locationInWindow"),"\uff0c\u4e5f\u662f\u76f8\u5bf9\u4e8e\u539f\u70b9\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPoint")),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u65f6\uff0c\u4ec5\u4ec5\u5c06\u81ea\u5df1\u7684\u601d\u7ef4\u4ece iOS \u7684 CGRect \u5750\u6807\u7cfb\u8f6c\u5316\u81f3 NSRect \u5750\u6807\u7cfb\u8fd8\u7b97\u7b80\u5355\u3002\u4f46\u5728\u540e\u7ee7\u7684\u903b\u8f91\u4e2d\uff0c\u56e0\u4e3a OCR \u7ed3\u679c\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRectangleObservation.boundingBox")," \u53c8\u4f1a\u56de\u5230 CGRect \u5750\u6807\u7cfb\uff0c\u5750\u6807\u7cfb\u7684\u9891\u7e41\u8f6c\u5316\u786e\u5b9e\u4f1a\u7ed9\u4eba\u5e26\u6765\u4e00\u5b9a\u7684\u56f0\u6270"),(0,l.kt)("h3",{id:"\u5c4f\u5e55\u6355\u6349"},"\u5c4f\u5e55\u6355\u6349"),(0,l.kt)("p",null,"\u60f3\u6355\u6349 macOS \u7684\u5c4f\u5e55\uff0c\u4f60\u53ef\u4ee5\u8c03\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u51fd\u6570"),(0,l.kt)("p",null,"\u4f46\u503c\u5f97\u6ce8\u610f\u7684\u662f\uff0c\u60f3\u8981\u622a\u53d6\u5c4f\u5e55\u4e0a\u5176\u4ed6\u8fdb\u7a0b\u7684\u5185\u5bb9(\u6bd4\u5982 IDE \u6216\u8005\u6d4f\u89c8\u5668)\uff0c\u9700\u8981\u9884\u5148\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGRequestScreenCaptureAccess"),"\uff0c\u7533\u8bf7\u5230\u622a\u53d6\u5176\u4ed6\u8fdb\u7a0b UI \u7684\u6743\u9650\uff0c\u5426\u5219 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u53ea\u80fd\u62ff\u5230 macOS \u7684\u684c\u9762\uff08\u4ee5\u53ca\u7a0b\u5e8f\u672c\u8eab\uff09"),(0,l.kt)("h3",{id:"ocr"},"OCR"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 Apple \u4e3a\u5f00\u53d1\u8005\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Vision - Recognizing Text")," \u8fdb\u884c\u5c4f\u5e55\u6587\u672c\u7684\u63d0\u53d6"),(0,l.kt)("p",null,"OCR \u8bf7\u6c42\u7684\u8c03\u7528\u7531 flutter side \u5524\u8d77\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'void dispatchOCRRequestToNative() async {\n final x = 100;\n final y = 100;\n final width = 300;\n final height = 80;\n final dimensions = [x, y, width, height];\n final result = await methodChannel.invokeMethod("captureAndOCR", dimensions);\n\n // parse the result from native\n // ...\n}\n')),(0,l.kt)("p",null,"Native side \u5728\u63a5\u6536\u5230\u8bf7\u6c42\u540e\uff0c\u4f1a\u6267\u884c\u622a\u5c4f\u548c\u8bc6\u522b\u64cd\u4f5c\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"import Vision\n// ...\nlet result : FlutterResult = ...\nlet image = captureScreen()\nlet handler = VNImageRequestHandler(cgImage: image)\nlet request = VNRecognizeTextRequest { request, error in\n // ocr finished\n let parsedResult = parse(request.results)\n result(parsedResult)\n}\nhandler.perform([request])\n")),(0,l.kt)("h4",{id:"\u6027\u80fd\u8868\u73b0"},"\u6027\u80fd\u8868\u73b0"),(0,l.kt)("p",null,"\u5728 macbook 2021 \u7684 m1 pro \u4e0a\uff0c\u4ee5 320\u271575 \u7684\u8bbe\u8ba1\u5206\u8fa8\u7387\u622a\u56fe(\u5b9e\u9645\u5206\u8fa8\u7387\u4e3a 640\u2715150)\uff0c\u6bcf\u79d2 11 \u5e27\u7684\u60c5\u51b5\u4e0b\u8fdb\u884c\u957f\u65f6\u95f4\u7684\u622a\u56fe\u548c OCR \u64cd\u4f5c\uff0c\u6574\u4e2a\u6d41\u7a0b\u7684\u5ef6\u8fdf\u5e73\u5747\u7ea6\u4e3a 65ms\uff0c\u6211\u8ba4\u4e3a\u8fd8\u7b97\u662f\u4e00\u4e2a\u53ef\u63a5\u53d7\u7684\u72b6\u6001"),(0,l.kt)("h4",{id:"\u8282\u80fd\u4f18\u5316"},"\u8282\u80fd\u4f18\u5316"),(0,l.kt)("p",null,"\u5373\u4fbf\u662f\u6027\u80fd\u5141\u8bb8\uff0c\u4f18\u5316\u4e5f\u662f\u5e94\u8be5\u505a\u7684\uff0c\u7528\u6237\u4e00\u65e6\u4f7f\u7528\u4f60\u7684 App\uff0c\u5c31\u611f\u89c9 macbook \u7684 C \u9762\u53d1\u70ed\uff0c\u8fd9\u662f\u65e0\u6cd5\u5bb9\u5fcd\u7684"),(0,l.kt)("p",null,"\u5f53\u524d\u5728 OCR \u6d41\u7a0b\u4e2d\u4e3b\u8981\u7684\u6027\u80fd\u4f18\u5316\u6b65\u9aa4\u662f\u5728 native \u7ef4\u62a4\u4e00\u4e2a\u5148\u8fdb\u5148\u51fa\uff0c\u6700\u5927\u5bb9\u91cf\u4e3a 40 \u5e27\u7684\u5b57\u5178\uff0c\u4ee5\u56fe\u7247\u6570\u636e\u4e3a key \u7f13\u5b58 OCR \u7684\u7ed3\u679c\uff0c\u4e0b\u9762\u662f\u7b80\u5316\u540e\u7684\u4ee3\u7801\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"let cacheManager = CacheManager()\n// ...\nlet image = captureScreen()\nlet key = image.dataProvider.data\nif let cachedResult = cacheManager[key] {\n delegate.onOCRResult(cachedResult)\n return\n}\n// ...\ndispatchOCRRequestToDeviceGPU(image) { result\n cacheManager[key] = result\n // ... other logic\n}\n// ...\n")),(0,l.kt)("p",null,"\u5f53\u9f20\u6807\u6307\u9488\u4f4d\u7f6e\u4e0d\u53d8\uff0c\u622a\u56fe\u83b7\u53d6\u7684\u56fe\u7247\u4e0d\u53d8\u65f6\uff0cnative \u5728\u5904\u7406 OCR \u8bf7\u6c42\u65f6\u4f1a\u5148\u547d\u4e2d\u7f13\u5b58\uff0c\u5e76\u76f4\u63a5\u8fd4\u56de\u7ed3\u679c\uff0c\u4ee5\u51cf\u5c11\u975e\u5fc5\u9700\u7684 GPU \u8c03\u7528"),(0,l.kt)("p",null,"\u5728\u7cfb\u7edf\u81ea\u5e26\u7684\u6d3b\u52a8\u76d1\u89c6\u5668\u4e2d\u67e5\u770b\u8fdb\u7a0b\u3002\u53d1\u73b0\uff0c\u5728\u5e94\u7528\u7684\u7f13\u5b58\u7b56\u7565\u540e\uff0c\u5728\u672c\u7a0b\u5e8f\u6d3b\u52a8\u65f6\uff0c\u5176 CPU/GPU \u5360\u7528\u7387\u786e\u5b9e\u964d\u4f4e\u4e86\u5f88\u591a\uff0c\u540c\u65f6\uff0cOCR \u7684\u7f13\u5b58\u7ed3\u679c\u88ab\u964d\u4e3a\u4e86 5ms"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u611f\u89c9\uff0c\u4ee5 ",(0,l.kt)("inlineCode",{parentName:"p"},"CFData")," \u4f5c\u4e3a key \u67e5\u8be2\u5b57\u5178\u8fd8\u4e0d\u662f\u6548\u7387\u6700\u9ad8\u7684\u7b97\u6cd5\uff0c\u5e94\u8be5\u53ef\u4ee5\u7ee7\u7eed\u63a2\u7d22\u4e00\u4e0b")),(0,l.kt)("h4",{id:"\u4e0d\u622a\u53d6\u81ea\u5df1"},"\u4e0d\u622a\u53d6\u81ea\u5df1"),(0,l.kt)("p",null,"\u5728\u6211\u8fdb\u884c\u5f00\u53d1\u65f6\uff0c\u53d1\u73b0\u5f53\u672c\u7a0b\u5e8f\u5728\u5c55\u793a\u5355\u8bcd\u91ca\u4e49 UI (HUD)\u65f6\uff0c\u56e0\u4e3a HUD \u672c\u8eab\u4e5f\u4f1a\u5728\u4e00\u5b9a\u8303\u56f4\u5185\u88ab\u622a\u5c4f\u51fd\u6570\u6355\u83b7\uff0c\u5bfc\u81f4 HUD \u4f1a\u5f71\u54cd OCR \u7684\u7ed3\u679c\uff0c\u800c\u5728 OCR \u7ed3\u679c\u53d8\u52a8\u540e\uff0cHUD \u53c8\u4f1a\u968f\u4e4b\u53d1\u751f\u53d8\u52a8\uff0c\u518d\u6b21\u5f71\u54cd OCR \u7ed3\u679c\uff0c\u5c31\u8fd9\u6837\u5faa\u73af\u5f80\u590d\uff0c\u8fde\u7f13\u5b58\u4e5f\u90fd\u5931\u6548\u4e86\u3002\u5728\u67e5\u9605\u548c\u5c1d\u8bd5\u5927\u91cf\u7684 API \u540e\uff0c\u6211\u7ec8\u4e8e\u5728 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u4e2d\u627e\u5230\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},"sharingType")," \u8fd9\u4e2a\u5c5e\u6027\uff0c\u5c4f\u5e55\u6355\u6349\u65b9\u6cd5\u6355\u83b7 HUD \u81ea\u8eab\uff0c\u603b\u7b97\u662f\u6253\u7834\u4e86\u8fd9\u4e2a\u94fe\u6761"),(0,l.kt)("h3",{id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d"},"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d"),(0,l.kt)("p",null,"\u5bf9 Vision - VNRecognizeTextRequest \u7684\u7ed3\u679c\u89e3\u6790\u662f\u4e2a\u590d\u6742\u7684\u4efb\u52a1"),(0,l.kt)("p",null,"\u6267\u884c\u8fd9\u4e2a\u590d\u6742\u4efb\u52a1\u7684\u539f\u56e0\u6709\u4e09\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u8bbe\u8ba1\u7684\u4ea4\u4e92\u662f\u201c\u7528\u6237\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u611f\u5174\u8da3\u7684\u5355\u8bcd\u4e0a\u65f6\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u201d\uff0c\u8fd9\u5c31\u8981\u6c42\u6211\u4eec\u77e5\u9053\u5c4f\u5e55\u4e0a\u6bcf\u4e2a\u5355\u8bcd\u7684\u5177\u4f53\u4f4d\u7f6e\u548c\u5185\u5bb9"),(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u9488\u5bf9\u6e90\u4ee3\u7801\u7279\u5316\uff0c\u5728\u9762\u5bf9",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E9%A9%BC%E5%B3%B0%E5%BC%8F%E5%A4%A7%E5%B0%8F%E5%86%99"},"\u9a7c\u5cf0\u547d\u540d\u6cd5"),"\u65f6\uff0c\u9700\u8981\u77e5\u9053\u6784\u6210\u4e00\u4e2a symbol \u7684\u6bcf\u4e2a\u5355\u8bcd\u7684\u610f\u601d\uff1a",(0,l.kt)("inlineCode",{parentName:"li"},"ThisIsAVeryVeryLongClassName")," -> ",(0,l.kt)("inlineCode",{parentName:"li"},"This"),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"Is")),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"A")),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Long"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Class"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Name")," (\u56e0\u8fc7\u4e8e\u7b80\u5355\uff0c\u4e22\u5f03\u957f\u5ea6\u5c0f\u4e8e 3 \u7684\u82f1\u8bed\u5355\u8bcd)"),(0,l.kt)("li",{parentName:"ol"},"\u6211\u4eec\u8981\u4ee5\u5355\u8bcd\u7684\u6587\u672c\u5185\u5bb9\u4e3a key\uff0c\u67e5\u8be2\u6570\u636e\u5e93\u3002\u5728\u6587\u672c\u5e8f\u5217\u5305\u542b\u7279\u6b8a\u5b57\u7b26\u65f6\uff0c\u6211\u4eec\u662f\u65e0\u6cd5\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u5230\u5355\u8bcd\u91ca\u4e49\u7684\u3002\u6240\u4ee5\u8981\u79fb\u9664\u975e\u5b57\u6bcd\u5b57\u7b26\u3002\u540c\u65f6\uff0c\u8fd9\u4e00\u64cd\u4f5c\u4e5f\u53ef\u4ee5\u81ea\u7136\u800c\u7136\u5730\u9002\u914d\u7f16\u7a0b\u8bed\u8a00\u4e2d\u7684\u5176\u4ed6\u547d\u540d\u65b9\u5f0f\uff0c\u6bd4\u5982\u79fb\u9664\u4e0b\u5212\u7ebf\u5c31\u53ef\u4ee5\u9002\u5e94",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E8%9B%87%E5%BD%A2%E5%91%BD%E5%90%8D%E6%B3%95"},"\u86c7\u5f62\u547d\u540d\u6cd5"))),(0,l.kt)("p",null,"\u5047\u8bbe\u6211\u4eec\u5728\u5bf9\u4e0b\u9762\u7684\u56fe\u7247\u6267\u884c OCR \u8bf7\u6c42"),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"ocr_text_source",src:n(3591).Z,width:"1038",height:"198"})),(0,l.kt)("p",null,"\u5728 Native \u7aef\u53d6 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizeTextRequest.results.topCandidates(1)")," \u540e\uff0c\u7ed3\u679c\u5982\u4e0b"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision provides its text-recognition capabilities through VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request, an image-based request type that finds and extracts text in images. The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following example shows how to use VNImageRequestHandler to perform a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest for recognizing text in the specified CGImage.",\n]\n')),(0,l.kt)("p",null,"\u5728 Native \u7aef\u4ee5",(0,l.kt)("strong",{parentName:"p"},"\u7a7a\u683c"),"\u5b57\u7b26\u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizedText.boundingBox(for:)")," \u5bf9 OCR \u7ed3\u679c\u8fdb\u884c\u7b2c\u4e00\u6b21\u6574\u7406\uff0c\u4e0a\u8ff0\u7684\u7ed3\u679c\u4f1a\u53d8\u4e3a\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text-recognition", "capabilities", "through", "VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request,", "an", "image-based", "request", "type", "that", "finds", "and", "extracts", "text", "in", "images.", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "to", "use", "VNImageRequestHandler", "to", "perform", "a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest", "for", "recognizing", "text", "in", "the", "specified", "CGImage.",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\u5c06 boundingBox \u7684\u8fd4\u56de\u503c\u548c\u622a\u5c4f\u5c3a\u5bf8\u8fdb\u884c\u4e58\u7b97\uff0c\u8fd4\u56de\u7ed9 flutter \u7aef\u3002Flutter \u7aef\u5728\u62ff\u5230\u7ed3\u679c\u540e\u4e3b\u8981\u8fdb\u884c\u4e24\u4e2a\u6b65\u9aa4\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u79fb\u9664\u975e\u82f1\u6587\u5b57\u7b26\uff0c\u8ba1\u7b97\u65b0\u7684 rect"),(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u5bf9\u5305\u542b\u5927\u5199\u5b57\u6bcd\u7684\u5b57\u7b26\u4e32\u8fdb\u884c\u5206\u5272\uff0c\u8ba1\u7b97\u65b0\u7684 rect")),(0,l.kt)("p",null,"\u8fd9\u6837\uff0c\u6587\u672c\u5757\u7684\u5185\u5bb9\u5728 flutter \u7aef\u5c31\u4f1a\u53d8\u6210\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text", "recognition", "capabilities", "through", "Recognize", "Text",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request","image", "based", "request", "type", "that", "finds", "and", "extracts", "text", "images", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "use", "Image", "Request", "Handler", "perform",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Recognize", "Text", "Request", "for", "recognizing", "text", "the", "specified", "Image",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\uff0cflutter \u4fa7\u4f1a\u4ee5\u4e0b\u9762\u7684\u4ee3\u7801\u5bf9\u8fd9\u4e9b OCR \u4fe1\u606f\u8fdb\u884c\u62bd\u8c61\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},"/// The representation of block\n///\n/// \u6587\u672c\u5757\u8868\u5f81\nclass Block {\n String text;\n double x;\n double y;\n double w;\n double h;\n}\n\n// ...\n\n/// All text representations in memory\n///\n/// \u6240\u6709\u6587\u672c\u5757\u8868\u5f81\nfinal blocks = StateProvider>((_) => []);\n")),(0,l.kt)("p",null,"\u4e0a\u9762\u7684\u6570\u636e\u5728\u5c4f\u5e55\u4e0a\u7684\u53ef\u89c6\u5316\u6548\u679c\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation_2",src:n(6607).Z,width:"1050",height:"184"})),(0,l.kt)("p",null,"\u5176\u4e2d\u5916\u56f4\u7eff\u6846\u4ee3\u8868\u622a\u5c4f\u8303\u56f4\uff0c\u5185\u90e8\u7684\u5c0f\u7eff\u6846\u4ee3\u8868 native \u7aef\u8fd4\u56de\u7684\u7ed3\u679c\uff0c\u5c0f\u7eff\u6846\u5185\u90e8\u7684\u84dd\u8272\u6846\u4ee3\u8868 flutter side \u7b2c\u4e8c\u6b21\u89e3\u6790\u7ed3\u679c\u3002\u53ef\u4ee5\u770b\u5230\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},'"VNImageRequestHandler"')," \u8fd9\u4e2a\u5b57\u7b26\u4e32\u88ab\u5206\u5272\u4e3a\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},'"Image"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Request"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Handler"')," \u8fd9\u4e09\u4e2a\u6587\u672c\u5757\u3002\u8fd9\u6837\uff0c\u6211\u5c31\u53ef\u4ee5\u77e5\u9053\u7528\u6237\u5230\u5e95\u5bf9\u54ea\u4e00\u6bb5\u5b57\u6bcd\u5e8f\u5217\uff08\u5355\u8bcd\uff09\u611f\u5174\u8da3\u4e86\uff08\u5047\u8bbe\u6e90\u4ee3\u7801\u7b26\u53f7\u7684\u547d\u540d\u662f\u89c4\u8303\u7684 \ud83d\ude07\uff09\u3002"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u4f60\u4e5f\u53ef\u4ee5\u6253\u5f00\u672c\u7a0b\u5e8f\u7684\u7528\u6237\u8bbe\u7f6e\u9762\u677f\u6765\u76f4\u63a5\u89c2\u5bdf\u5176\u8fd0\u884c\u65f6\u8868\u73b0\uff1a\n",(0,l.kt)("img",{alt:"dashboard_inspect",src:n(6521).Z,width:"970",height:"192"}))),(0,l.kt)("h3",{id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"},"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,'\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u4f1a\u521b\u5efa\u4e24\u4e2a flutter engine\uff0c\u5206\u522b\u7528\u4e8e\u6e32\u67d3"\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)"\u548c"\u6b22\u8fce\u4e0e\u8bbe\u7f6e\u9875\u9762(Dashboard)"'),(0,l.kt)("p",null,"\u8fd9\u4e24\u4e2a\u5f15\u64ce\u5747\u901a\u8fc7\u57fa\u672c\u65b9\u5f0f\u521b\u5efa\uff0c\u5747\u91c7\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")," \u4e0e native \u8fdb\u884c\u901a\u8baf"),(0,l.kt)("p",null,"\u53cc\u5f15\u64ce\u5171\u4eab\u540c\u4e00\u4efd\u4ee3\u7801\uff0c\u7ecf\u7531\u4e0d\u540c\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/dart-lang/sdk/blob/master/runtime/docs/compiler/aot/entry_point_pragma.md"},"dart \u5165\u53e3\u51fd\u6570"),"\u542f\u52a8\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'// \u8bbe\u7f6e\u9875\u9762 (Dashboard)\ndashboardEngine = FlutterEngine(name: "dashboard", project: nil)\ndashboardChannel = FlutterMethodChannel(name: "dashboard", binaryMessenger: dashboardEngine.binaryMessenger)\ndashboardEngine.run(withEntrypoint: "_dashboard")\n//...\n\n// \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\nhudEngine = FlutterEngine(name: "hud", project: nil)\nhudChannel = FlutterMethodChannel(name: "hud", binaryMessenger: hudEngine.binaryMessenger)\nhudChannel.run(withEntrypoint: "_hud")\n')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u4e2a\u4eba\u6ca1\u6709\u4f7f\u7528\u5b98\u65b9\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://docs.flutter.dev/add-to-app/multiple-flutters"},"FlutterEngineGroup \u65b9\u6848"),"\uff0c\u56e0\u4e3a\u5728\u5f53\u521d\u8fdb\u884c\u5c1d\u8bd5\u65f6\u53d1\u73b0\u4e86\u5f15\u64ce\u65e0\u54cd\u5e94\u7684\u95ee\u9898\uff0c\u81f3\u4eca\u4ecd\u7136\u662f P2 \u7ea7\u522b\u7684 open ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/flutter/flutter/issues/119403"},"issue"))),(0,l.kt)("h4",{id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"},"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u6765\u7ba1\u7406\u7edd\u5927\u90e8\u5206\u72b6\u6001\uff0c\u6240\u4ee5\u5728\u540c\u6b65\u72b6\u6001\u65f6\uff0c\u6211\u4e5f\u671f\u671b\u81ea\u5df1\u7684\u81ea\u5df1\u7684\u5fc3\u667a\u6a21\u578b\u53ef\u4ee5\u66f4\u8d34\u8fd1 riverpod"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0cDashboard engine \u9700\u8981\u5c06\u7528\u6237\u8bbe\u7f6e\u540c\u6b65\u81f3 HUD engine (\u6bd4\u5982\u66f4\u6539\u67e5\u8bcd\u5feb\u6377\u952e)\uff0c\u6211\u5728\u8fd9\u91cc\u76d1\u542c\u4e86 Dashboard engine \u5bf9\u5e94\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," \u7684\u53d8\u66f4\uff0c\u5e76\u901a\u8fc7 method channel\uff0c\u7ecf\u7531 native \u8f6c\u53d1\u81f3 HUD engine\uff0cHUD engine \u5728\u6536\u5230\u4e86\u901a\u77e5\u540e\uff0c\u518d\u66f4\u65b0\u81ea\u5df1\u7ef4\u62a4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider"),"\u3002\u4ece\u800c\u5b9e\u73b0\u4e86\u8de8\u5f15\u64ce\u7684\u72b6\u6001\u540c\u6b65"),(0,l.kt)("h3",{id:"\u91ca\u4e49\u67e5\u8be2"},"\u91ca\u4e49\u67e5\u8be2"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u5728\u67e5\u8be2\u5355\u8bcd\u91ca\u4e49\u65f6\u4f7f\u7528\u672c\u5730 sqlite \u6570\u636e\u5e93\uff0c\u6570\u636e\u5e93\u6e90\u4e8e ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/skywind3000/ECDICT-ultimate"},"ecdict-ultimate")),(0,l.kt)("p",null,"\u9274\u4e8e\u5f00\u6e90\u6570\u636e\u5e93\u8fc7\u4e8e\u5de8\u5927\uff0c\u4e14\u5b58\u5728\u4e00\u4e9b\u672c\u7a0b\u5e8f\u65e0\u9700\u4f7f\u7528\u7684 columns\uff0c\u6211\u53c8\u4f7f\u7528 dart \u548c ",(0,l.kt)("a",{parentName:"p",href:"https://pub.dev/packages/drift"},"drift")," \u5bf9\u5176\u8fdb\u884c\u4e86\u6e05\u6d17\u548c\u526a\u88c1"),(0,l.kt)("p",null,"\u5728\u67e5\u8be2\u65f6\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5c06\u67e5\u8be2\u7684\u672c\u6587\u4e32\u201c\u6253\u6563\u201d\uff0c\u5e76\u4e00\u9f50\u8bf7\u6c42\u6570\u636e\u5e93\uff0c\u4ee5\u671f\u80fd\u547d\u4e2d\u4e0d\u65ad\u5343\u53d8\u4e07\u5316\u7684\u82f1\u8bed\u5355\u8bcd\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'final sequence = "qwertyuiopasdfghjkl";\nfinal sequences = ["qwe","wer","ert",...,"qwer","wert",...,"qwert",...,"qwertyuiopasdfghjkl"];\nfinal List results = await queryDB(keys:sequences); // average latency: ~6ms\n')),(0,l.kt)("p",null,"\u5728\u8bf7\u6c42\u5b8c\u6210\u540e\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5bf9\u8bf7\u6c42\u7ed3\u679c\u8fdb\u884c\u53bb\u91cd\uff0c\u903b\u8f91\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b owl (\u732b\u5934\u9e70)"),(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b knowledge(\u77e5\u8bc6)"),(0,l.kt)("li",{parentName:"ul"},'\u5e8f\u5217 "knowledge" \u5305\u542b "owl"\uff0c\u79fb\u9664\u67e5\u8be2\u7ed3\u679c\u4e2d\u7684 owl')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u76ee\u524d\uff0c\u8be5\u903b\u8f91\u8fd8\u6ca1\u6709\u5b9e\u73b0\u5bf9\u51b2\u7a81\u7684\u5224\u5b9a\uff1a\u5982\u5411\u6570\u636e\u5e93\u67e5\u8be2\u7684 key \u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'"gitignore"'),"\uff0c\u67e5\u8be2\u7ed3\u679c\u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'["git", "tig", "ignore"]'),"\uff0c\u76ee\u524d\u8fd9\u4e09\u4e2a\u7ed3\u679c\u90fd\u4f1a\u88ab\u6e32\u67d3\u5230 HUD \u4e0a\u3002\u4f46\u663e\u7136\uff0c\u9009\u53d6 git \u548c ignore \u8fd9\u4e24\u4e2a\u67e5\u8be2\u7ed3\u679c\u9020\u6210\u7684 \u201c\u51b2\u7a81\u201d \u662f\u6700\u5c0f\u7684")),(0,l.kt)("h3",{id:"\u53d1\u97f3"},"\u53d1\u97f3"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/avfoundation/speech_synthesis"},"AVFoundation - Speech synthesis")," API \u6765\u5b9e\u65f6\u5408\u6210\u8bed\u97f3"),(0,l.kt)("p",null,"\u8be5 API \u53ef\u4ee5\u540c\u65f6\u5728 iOS/macOS \u4e0a\u4f7f\u7528\uff0c\u4e14\u53d1\u97f3\u8f83\u4e3a\u51c6\u786e\uff0c\u4ee5\u6211\u81ea\u8eab\u7684\u80fd\u529b(\u5168\u56fd\u5927\u5b66\u82f1\u8bed\u56db\u7ea7)\u6765\u770b\uff0c\u6548\u679c\u8fd8\u7b97\u6ee1\u610f\uff0c\u786e\u5b9e\u6bd4\u6211\u7684\u53d1\u97f3\u51c6 \ud83e\udd23"),(0,l.kt)("p",null,"\u5f53\u7136\uff0c\u540e\u7ee7\u5982\u679c\u6709\u66f4\u9ad8\u7684\u9700\u6c42\uff0c\u4e5f\u6709\u5f88\u591a\u5176\u4ed6\u53ef\u9009\u65b9\u6848"))}c.isMDXComponent=!0},3591:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/ocr_text_source-9af654c22d465590f3079fd8d4f2f24f.png"},1589:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/appkit_screens_coordinate-94563877ba8462b8412c6e945e1f7b5e.png"},6521:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/dashboard_inspect-0a3c0a7bba8e62a06a697e1555bb3ea4.png"},2731:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/intro-663e889360c7a0e605df52af33af3b11.gif"},8555:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_dark-ede87b48b4273a15fc0974c77b5586cf.png"},1974:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_light-980bb1be0d553350c1dcf22e0b4325e2.png"},3642:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/project_layout-b9e52932db96541ab24f4ed06bc28bc0.png"},9462:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation-e7dc02491d5b436f2ef66bd928e65e17.png"},6607:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation_1-7c6db883d1b12b61355048e553101f21.png"}}]); \ No newline at end of file diff --git a/assets/js/f812cb1f.26bfec5d.js b/assets/js/f812cb1f.26bfec5d.js deleted file mode 100644 index ae0a3f8..0000000 --- a/assets/js/f812cb1f.26bfec5d.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdf_doc_source=self.webpackChunkdf_doc_source||[]).push([[594],{3905:(e,t,n)=>{n.d(t,{Zo:()=>d,kt:()=>k});var a=n(7294);function l(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(l[n]=e[n]);return l}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var p=a.createContext({}),s=function(e){var t=a.useContext(p),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},d=function(e){var t=s(e.components);return a.createElement(p.Provider,{value:t},e.children)},u="mdxType",c={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,l=e.mdxType,r=e.originalType,p=e.parentName,d=o(e,["components","mdxType","originalType","parentName"]),u=s(n),m=l,k=u["".concat(p,".").concat(m)]||u[m]||c[m]||r;return n?a.createElement(k,i(i({ref:t},d),{},{components:n})):a.createElement(k,i({ref:t},d))}));function k(e,t){var n=arguments,l=t&&t.mdxType;if("string"==typeof e||l){var r=n.length,i=new Array(r);i[0]=m;var o={};for(var p in t)hasOwnProperty.call(t,p)&&(o[p]=t[p]);o.originalType=e,o[u]="string"==typeof e?e:l,i[1]=o;for(var s=2;s{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>i,default:()=>c,frontMatter:()=>r,metadata:()=>o,toc:()=>s});var a=n(7462),l=(n(7294),n(3905));const r={slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},i=void 0,o={permalink:"/valo-reader-doc/blog/tech_detail_macos",editUrl:"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/tech_detail_macos.md",source:"@site/blog/tech_detail_macos.md",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",description:"\u6458\u8981",date:"2023-10-19T05:59:23.000Z",formattedDate:"2023\u5e7410\u670819\u65e5",tags:[{label:"technical",permalink:"/valo-reader-doc/blog/tags/technical"}],readingTime:23.595,hasTruncateMarker:!1,authors:[{name:"Ce Wang",title:"Solo dev",url:"https://github.com/haloWang",imageURL:"https://github.com/haloWang.png",key:"halowang"}],frontMatter:{slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},nextItem:{title:"\u5f00\u53d1\u76ee\u7684",permalink:"/valo-reader-doc/blog/motivation"}},p={authorsImageUrls:[void 0]},s=[{value:"\u6458\u8981",id:"\u6458\u8981",level:2},{value:"\u4ec0\u4e48\u662f Valo Reader?",id:"\u4ec0\u4e48\u662f-valo-reader",level:2},{value:"\u9879\u76ee\u5e03\u5c40",id:"\u9879\u76ee\u5e03\u5c40",level:2},{value:"\u5de5\u7a0b\u67b6\u6784",id:"\u5de5\u7a0b\u67b6\u6784",level:2},{value:"flutter \u4fa7",id:"flutter-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f",id:"\u4e3b\u7a0b\u5e8f",level:4},{value:"flutter package: \u6587\u672c\u5757\u8868\u5f81",id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81",level:4},{value:"flutter package: \u57fa\u7840\u5e93",id:"flutter-package-\u57fa\u7840\u5e93",level:4},{value:"native \u4fa7",id:"native-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",level:4},{value:"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93",id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93",level:4},{value:"CocoaPods dependency: OCR \u529f\u80fd\u5e93",id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93",level:4},{value:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",level:2},{value:"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)",id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud",level:3},{value:"\u5728 HUD \u4e2d\u7684\u72b6\u6001",id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001",level:4},{value:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)",id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard",level:3},{value:"\u539f\u751f\u4fa7 (native)",id:"\u539f\u751f\u4fa7-native",level:3},{value:"\u6280\u672f\u7ec6\u8282",id:"\u6280\u672f\u7ec6\u8282",level:2},{value:"\u591a\u5c4f\u5e55",id:"\u591a\u5c4f\u5e55",level:3},{value:"\u5c4f\u5e55\u6355\u6349",id:"\u5c4f\u5e55\u6355\u6349",level:3},{value:"OCR",id:"ocr",level:3},{value:"\u6027\u80fd\u8868\u73b0",id:"\u6027\u80fd\u8868\u73b0",level:4},{value:"\u8282\u80fd\u4f18\u5316",id:"\u8282\u80fd\u4f18\u5316",level:4},{value:"\u4e0d\u622a\u53d6\u81ea\u5df1",id:"\u4e0d\u622a\u53d6\u81ea\u5df1",level:4},{value:"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d",id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d",level:3},{value:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",level:3},{value:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",level:4},{value:"\u91ca\u4e49\u67e5\u8be2",id:"\u91ca\u4e49\u67e5\u8be2",level:3},{value:"\u53d1\u97f3",id:"\u53d1\u97f3",level:3}],d={toc:s},u="wrapper";function c(e){let{components:t,...r}=e;return(0,l.kt)(u,(0,a.Z)({},d,r,{components:t,mdxType:"MDXLayout"}),(0,l.kt)("h2",{id:"\u6458\u8981"},"\u6458\u8981"),(0,l.kt)("p",null,"\u672c\u6587\u4e3b\u8981\u4ecb\u7ecd\u4e86 Valo Reader for macOS\uff08\u4e0b\u6587\u7b80\u79f0\u4e3a\u201c\u672c\u7a0b\u5e8f\u201d\uff09\u7684",(0,l.kt)("a",{parentName:"p",href:"#%E9%A1%B9%E7%9B%AE%E5%B8%83%E5%B1%80"},"\u9879\u76ee\u5e03\u5c40"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E5%B7%A5%E7%A8%8B%E6%9E%B6%E6%9E%84"},"\u5de5\u7a0b\u67b6\u6784"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E8%BF%90%E8%A1%8C%E6%97%B6%E5%8A%9F%E8%83%BD%E6%A8%A1%E5%9D%97"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),"\u4e0e",(0,l.kt)("a",{parentName:"p",href:"#%E6%8A%80%E6%9C%AF%E7%BB%86%E8%8A%82"},"\u6280\u672f\u7ec6\u8282"),"\u3002\u4f60\u8fd8\u53ef\u4ee5\u5728",(0,l.kt)("a",{parentName:"p",href:"/docs/intro"},"\u672c\u6587\u6863"),"\u7684\u5176\u4ed6\u9875\u9762\u4e86\u89e3\u672c\u7a0b\u5e8f\u7684",(0,l.kt)("a",{parentName:"p",href:"/valo-reader-doc/blog/motivation"},"\u5f00\u53d1\u52a8\u673a"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/usage"},"\u4f7f\u7528\u65b9\u5f0f"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/privacy_security"},"\u9690\u79c1\u4e0e\u5b89\u5168"),"\u7b49\u5185\u5bb9"),(0,l.kt)("h2",{id:"\u4ec0\u4e48\u662f-valo-reader"},"\u4ec0\u4e48\u662f Valo Reader?"),(0,l.kt)("p",null,"Valo Reader \u8fd9\u4e2a\u9879\u76ee\u662f\u6211\u4e2a\u4eba\u5728\u65e5\u5e38\u751f\u6d3b\u4e2d\u9010\u6e10\u840c\u751f\uff0c\u6f14\u5316\u548c\u5b9e\u65bd\u7684\u60f3\u6cd5\u3002\u662f\u6211\u4e3a\u4e86\u5728\u81ea\u5df1\u7535\u5b50\u8bbe\u5907\u8bbe\u5907\u4e0a\u80fd\u8f7b\u677e\u9605\u8bfb\u82f1\u6587\u5185\u5bb9\uff0c\u5e76\u6e10\u8fd1\u5f0f\u5730\u63d0\u9ad8\u81ea\u5df1\u7684\u82f1\u6587\u6c34\u5e73\uff0c\u800c\u81ea\u884c\u8bbe\u8ba1\u548c\u7814\u53d1\u7684\u4e00\u6b3e\u4ea7\u54c1\uff0c\u5176\u6838\u5fc3\u529f\u80fd\u5c55\u793a\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"intro",src:n(2731).Z,width:"960",height:"268"})),(0,l.kt)("p",null,"\u5f53\u4f60\u5728\u4f7f\u7528\u672c\u7a0b\u5e8f\u65f6\uff0c\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u4f60\u611f\u5230\u751f\u758f\u7684\u82f1\u8bed\u5355\u8bcd\u4e0a\u3002\u6309\u4e0b\u6307\u5b9a\u7684\u6309\u952e\uff08\u9ed8\u8ba4\u4e3a Fn\uff09\uff0c\u672c\u7a0b\u5e8f\u5373\u4f1a\u5728\u54cd\u5e94\u7684\u4f4d\u7f6e\u5c55\u793a\u5bf9\u5e94\u7684\u91ca\u4e49\uff0c\u540c\u65f6\u4f7f\u7528\u8bed\u97f3\u5408\u6210\u6765\u53d1\u97f3\u3002"),(0,l.kt)("p",null,"\u76ee\u524d\uff0c\u4f60\u53ef\u4ee5\u901a\u8fc7 ",(0,l.kt)("a",{parentName:"p",href:"https://apps.apple.com/cn/app/valo-reader/id6448040931"},"Mac App Store")," \u6216",(0,l.kt)("a",{parentName:"p",href:"/docs/installation"},"\u5176\u4ed6\u65b9\u5f0f"),"\u6765\u83b7\u53d6\u672c\u7a0b\u5e8f"),(0,l.kt)("h2",{id:"\u9879\u76ee\u5e03\u5c40"},"\u9879\u76ee\u5e03\u5c40"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u9879\u76ee\u7684\u4e3b\u4f53\u7ed3\u6784\uff1a\n",(0,l.kt)("img",{alt:"project_layout",src:n(3642).Z,width:"1996",height:"960"})),(0,l.kt)("p",null,"\u672c\u6587\u7684\u4f4d\u7f6e\u5219\u5904\u4e8e\u4e0a\u56fe\u4e2d\u7684\u6d45\u9ec4\u8272\u90e8\u5206\uff0c\u7531 ",(0,l.kt)("a",{parentName:"p",href:"https://docusaurus.io/"},"docusaurus")," \u751f\u6210"),(0,l.kt)("p",null,"\u672c\u9879\u76ee\u7684\u60f3\u6cd5\u6700\u65e9\u5b9e\u73b0\u4e8e\u5728\u6d4f\u89c8\u5668\u4e2d\u8fd0\u884c\u7684 js \u811a\u672c\uff0c\u4f60\u53ef\u4ee5\u5728 github \u4e0a\u770b\u5230\u672c\u9879\u76ee\u5728\u6d4f\u89c8\u5668\u4e0a\u7684\u65e9\u671f\u5b9e\u73b0\u53ca\u5bf9\u5e94\u7684",(0,l.kt)("a",{parentName:"p",href:"https://github.com/HaloWang/english_flow#%E5%8A%9F%E8%83%BD%E5%B1%95%E7%A4%BA"},"\u529f\u80fd\u5c55\u793a"),"\uff0c\u5373\u56fe\u4e2d\u6d45\u7eff\u8272\u90e8\u5206"),(0,l.kt)("p",null,"\u800c\u672c\u6587\u4e3b\u8981\u8bf4\u660e\u4e86\u4e0a\u56fe\u4e2d macOS\uff08\u6d45\u84dd\u8272\u90e8\u5206\uff09\u7684\u8fd0\u884c\u65b9\u5f0f\u4e0e\u6280\u672f\u7ec6\u8282"),(0,l.kt)("h2",{id:"\u5de5\u7a0b\u67b6\u6784"},"\u5de5\u7a0b\u67b6\u6784"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create XXX --platform macos")," \u547d\u4ee4\u521b\u5efa\uff0c\u5176\u6e90\u4ee3\u7801\u4e3b\u8981\u5206\u4e3a\u4e24\u90e8\u5206"),(0,l.kt)("h3",{id:"flutter-\u4fa7"},"flutter \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 dart \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f"},"\u4e3b\u7a0b\u5e8f"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u7684\u4e3b\u8981\u529f\u80fd\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u901a\u8fc7 method channel \u5b9e\u73b0 flutter \u4e0e native \u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u7528\u6237\u8bbe\u7f6e\u9762\u677f\uff08Dashboard\uff09"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\u4e0e\u8bca\u65ad\u68c0\u67e5\u4fe1\u606f"),(0,l.kt)("li",{parentName:"ol"},"\u4f7f\u7528\u7b2c\u4e09\u65b9\u4f9d\u8d56\u63d0\u4f9b\u7684\u529f\u80fd")),(0,l.kt)("h4",{id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81"},"flutter package: \u6587\u672c\u5757\u8868\u5f81"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u662f\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u4ee3\u7801\uff0c\u5728 macOS / iOS / Android \u4e09\u7aef\u5171\u4eab\u90e8\u5206\u903b\u8f91\u4e0e UI\uff0c\u5176\u4e3b\u8981\u529f\u80fd\u7531\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u89e3\u6790\u7528\u6237\u8bbe\u5907\u5c4f\u5e55\u4e0a\u8bc6\u522b\u5230\u7684\u6587\u672c\u5757\uff0c\u7ed3\u6784\u5316\u6587\u672c\u5757\uff0c\u5e76\u5c06\u5176\u7ef4\u62a4\u5230\u672c\u7a0b\u5e8f\u7684\u5185\u5b58\u4e2d"),(0,l.kt)("li",{parentName:"ol"},"\u58f0\u660e\u5e76\u5b9e\u73b0\u7528\u6237\u89c6\u89c9\u7126\u70b9\u4e0e\u5df2\u77e5\u7684\u6587\u672c\u4fe1\u606f\u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09")),(0,l.kt)("h4",{id:"flutter-package-\u57fa\u7840\u5e93"},"flutter package: \u57fa\u7840\u5e93"),(0,l.kt)("p",null,"\u5728\u6211\u7684\u6240\u6709 flutter \u9879\u76ee\u4e2d\u5171\u4eab\u7684\u57fa\u7840\u4f9d\u8d56"),(0,l.kt)("h3",{id:"native-\u4fa7"},"native \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 swift \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206"},"\u4e3b\u7a0b\u5e8f\u90e8\u5206"),(0,l.kt)("p",null,"\u6301\u6709 flutter engine\uff0c\u901a\u8fc7 method channel \u6253\u901a flutter \u4e0e ",(0,l.kt)("inlineCode",{parentName:"p"},"Base"),"\u3001",(0,l.kt)("inlineCode",{parentName:"p"},"OCR")," \u4ee5\u53ca\u5e94\u7528\u7a0b\u5e8f\u672c\u8eab\u7684\u53cc\u5411\u8c03\u7528"),(0,l.kt)("p",null,"\u521b\u5efa\u9879\u76ee\u65f6\u9ed8\u8ba4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController")," \u5df2\u88ab\u5220\u9664"),(0,l.kt)("h4",{id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93"},"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e Cocoa framework \u4ea4\u4e92\uff0c\u5982\uff1a\u6743\u9650\u7533\u8bf7\uff0c\u952e\u76d8\u76d1\u542c\uff0c\u9f20\u6807\u76d1\u542c\uff0c\u622a\u53d6\u5c4f\u5e55\uff0c\u79fb\u52a8\u627f\u8f7d flutter \u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow"),"\u4ee5\u53ca\u63d0\u4f9b\u57fa\u672c\u82e5\u5e72\u539f\u751f\u80fd\u529b"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u6211\u4e5f\u4f1a\u5728\u8be5\u4f9d\u8d56\u4e2d\u5b66\u4e60\uff0c\u5c1d\u8bd5\u548c\u5b9e\u73b0 Cocoa/AppKit \u72ec\u6709\u7684 API \u4e0e\u529f\u80fd"),(0,l.kt)("h4",{id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93"},"CocoaPods dependency: OCR \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Apple Vision framework - Text Recognizing")," \u4ea4\u4e92\uff0c\u5e76\u5b9e\u73b0\u90e8\u5206\u903b\u8f91\uff0c\u5982\uff1a\u53d1\u8d77\u6587\u672c\u8bc6\u522b\u8bf7\u6c42\uff0c\u7ef4\u62a4\u6587\u672c\u8bc6\u522b\u54cd\u5e94\u7f13\u5b58"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u5728 macOS \u548c iOS \u9879\u76ee\u540c\u65f6\u4f9d\u8d56\u5e76\u5171\u4eab\u4ee3\u7801"),(0,l.kt)("h2",{id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0c\u672c\u7a0b\u5e8f\u4e3b\u8981\u53ef\u5206\u4e3a\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\u3001\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD) \u548c\u539f\u751f\u4fa7 (native) \u4e09\u4e2a\u6a21\u5757"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u7684\u4e3b\u8981\u6a21\u5757\u53ca\u5176\u901a\u8baf\uff1a\n",(0,l.kt)("img",{alt:"modules_light",src:n(1974).Z+"#gh-light-mode-only",width:"1926",height:"894"}),"\n",(0,l.kt)("img",{alt:"modules_dark",src:n(8555).Z+"#gh-dark-mode-only",width:"1926",height:"894"})),(0,l.kt)("h3",{id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud"},"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)"),(0,l.kt)("p",null,"HUD \u662f\u4e00\u4e2a\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u81ea\u5df1\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterEngine")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")),(0,l.kt)("p",null,"HUD \u7684\u4e3b\u8981\u4efb\u52a1\u5305\u542b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u5411 native \u6d3e\u53d1 ocr \u8bf7\u6c42"),(0,l.kt)("li",{parentName:"ul"},"\u89e3\u6790 native \u5bf9\u5c4f\u5e55\u6307\u5b9a\u533a\u57df\u7684 ocr \u7ed3\u679c\uff0c\u5e76\u5728\u5185\u5b58\u4e2d\u7ef4\u62a4\u8be5\u7ed3\u679c\u4f9b\u540e\u7ee7\u7a0b\u5e8f\u903b\u8f91\u4f7f\u7528"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u4ee5\u5728\u9700\u8981\u65f6\u8ba1\u7b97\u91ca\u4e49\u5c55\u793a\u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u952e\u76d8\u6309\u952e\u72b6\u6001\uff0c\u5728\u53d8\u5316\u65f6\u6d3e\u53d1 ocr \u8bf7\u6c42\u5e76\u5c55\u793a\u5355\u8bcd\u91ca\u4e49"),(0,l.kt)("li",{parentName:"ul"},"\u5b9e\u65f6\u540c\u6b65 Dashboard engine \u4f20\u9012\u8fc7\u6765\u7684\u7528\u6237\u8bbe\u7f6e\uff0c\u5e76\u4fee\u6539\u5c55\u793a\u903b\u8f91")),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u8be5\u6a21\u5757\u6240\u7ef4\u62a4\u7684\u6570\u636e\u6620\u5c04\u5230\u5c4f\u5e55\u4e0a\u65f6\u7684\u53ef\u89c6\u5316\u6548\u679c\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation",src:n(9462).Z,width:"1898",height:"222"})),(0,l.kt)("h4",{id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001"},"\u5728 HUD \u4e2d\u7684\u72b6\u6001"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u7ba1\u7406\u7edd\u5927\u90e8\u5206\u7684\u72b6\u6001\u3002\u5728\u6784\u7b51\u672c\u7a0b\u5e8f\u65f6\uff0c\u6211\u7684\u601d\u8003\u8fc7\u7a0b\u4e3b\u8981\u57fa\u4e8e\u72b6\u6001\u4e0e\u72b6\u6001\u53d8\u5316\uff0c\u4ece\u5c55\u793a\u91ca\u4e49\u952e\u70b9\u51fb\u5230\u6e32\u67d3\u91ca\u4e49\u9762\u677f\u52a8\u753b"),(0,l.kt)("p",null,"\u5728\u5927\u91cf\u4f7f\u7528 riverpod \u4e2d\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"Provider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"ProviderContainer.listen")," \u540e\uff0c\u6211\u53ef\u4ee5\u6784\u5efa\u4e00\u4e2a\u9ad8\u5ea6\u5f02\u6b65\uff0c\u8c03\u7528\u987a\u5e8f\u4e0d\u654f\u611f\uff0c\u72b6\u6001\u53d8\u5316\u9a71\u52a8\u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u5728\u7f16\u7801\u8fc7\u7a0b\u4e2d\uff0c\u8fd9\u79cd\u65b9\u5f0f\u8ba9\u7a0b\u5e8f\u5458\u53ef\u4ee5\u4ece\u4e00\u5927\u4e32\u7684\u547d\u4ee4\u5f0f(Imperative)\u65b9\u6cd5\u8c03\u7528\u4e2d\u89e3\u653e\u51fa\u6765\uff0c\u4ec5\u4ec5\u9700\u8981\u5173\u6ce8\u548c\u786e\u4fdd\u6bcf\u4e2a\u6700\u5c0f\u903b\u8f91\u5355\u5143\u2014\u2014",(0,l.kt)("inlineCode",{parentName:"p"},"Provider"),"\uff0c\u786e\u4fdd\u5176\u5b9e\u73b0\u662f\u6b63\u786e\u7684\u5373\u53ef\uff0c\u8fd9\u51cf\u5c11\u4e86\u6784\u5efa\u590d\u6742\u548c\u957f\u4e32\u903b\u8f91\u51fa\u9519\u7684\u53ef\u80fd"),(0,l.kt)("p",null,"\u4e0b\u9762\u7684\u4e24\u5f20\u56fe\u8868\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u7684\u4e3b\u8981\u529f\u80fd\u6d41\u7a0b"),(0,l.kt)("p",null,"\u5207\u6362\u5c55\u793a\u91ca\u4e49\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:'graph LR\n A1[\u76d1\u542c\u952e\u76d8\u72b6\u6001];\n A[\u952e\u76d8\u72b6\u6001\u53d1\u751f\u6539\u53d8];\n query_button_down{\u5c55\u793a\u91ca\u4e49\u952e\u6309\u4e0b?};\n C[\u505c\u6b62\u5b9a\u65f6\u5668];\n D[\u542f\u52a8\u5b9a\u65f6\u5668\u5e76\u91cd\u7f6e\u6587\u672c\u5757\u72b6\u6001];\n mouse_state[\u9f20\u6807\u72b6\u6001];\n screen_state[\u5c4f\u5e55\u72b6\u6001];\n send_request_to_native["\u5411 native \u53d1\u9001\u622a\u56fe\u4e0e OCR \u8bf7\u6c42"];\n F[native OCR \u7ed3\u679c\u5df2\u83b7\u5f97];\n G[\u89e3\u6790 OCR \u7ed3\u679c];\n H1[\u66f4\u65b0\u6587\u672c\u5757\u72b6\u6001];\n\n mouse_state--\x3esend_request_to_native\n screen_state--\x3esend_request_to_native\n send_request_to_native-.->F;\n F--\x3eG;\n G--\x3eH1;\n A1-.->A;\n A--\x3equery_button_down;\n query_button_down--\x3e|false|C;\n query_button_down--\x3e|true|D;\n D-.->send_request_to_native;'}),(0,l.kt)("p",null,"\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u53d8\u66f4\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:"graph LR;\n H2[\u6587\u672c\u5757\u72b6\u6001\u53d8\u66f4];\n most_wanted_state_logic[\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u76d1\u542c\u903b\u8f91];\n most_wanted_state_changed[\u72b6\u6001\u6539\u53d8];\n mouse_change0[\u76d1\u542c\u9f20\u6807\u72b6\u6001];\n mouse_change[\u9f20\u6807\u72b6\u6001\u6539\u53d8];\n sync_mouse_to_flutter[\u540c\u6b65\u9f20\u6807\u72b6\u6001\u81f3 flutter];\n sync_current_screen[\u540c\u6b65\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u6570\u636e];\n sync_current_screen_to_flutter[\u540c\u6b65\u5c4f\u5e55\u6570\u636e\u81f3 flutter];\n hasTextBlock{\u72b6\u6001\u7684\u503c\u662f\u5426\u4e3a\u7a7a\\n\u5373\u9f20\u6807\u5f53\u524d\u4f4d\u7f6e\u6ca1\u6709\u8bc6\u522b\u5230\u6587\u672c};\n queryDB[\u67e5\u8be2\u6570\u636e\u5e93];\n dbHasData{\u6570\u636e\u5e93\u5305\u542b\u91ca\u4e49};\n hideAllFlow[\u9690\u85cf\u6240\u6709\u91ca\u4e49\u9762\u677f];\n asParam[\u4f5c\u4e3aOCR\u8bf7\u6c42\u7684\u53c2\u6570];\n screen_changed{\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u53d1\u751f\u53d8\u52a8?}\n\n \u5207\u6362\u5c55\u793a\u91ca\u4e49-.->H2\n mouse_change0-.->mouse_change--\x3esync_mouse_to_flutter;\n mouse_change --\x3e sync_current_screen --\x3e sync_current_screen_to_flutter;\n sync_current_screen_to_flutter --\x3e \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001\n sync_current_screen --\x3e screen_changed --\x3e |true| \u6539\u53d8\u627f\u8f7dHUD\u7684NSPanel\u4f4d\u7f6e;\n screen_changed --\x3e |false| return_0;\n H2 --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter -.-> asParam;\n \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001 -.-> asParam;\n most_wanted_state_logic --\x3e most_wanted_state_changed --\x3e hasTextBlock;\n \u5c55\u793a\u91ca\u4e49\u952e\u72b6\u6001 --\x3e most_wanted_state_logic;\n most_wanted_state_changed --\x3e hideAllFlow;\n hasTextBlock --\x3e |false| return_1;\n hasTextBlock --\x3e |true| queryDB;\n queryDB --\x3e dbHasData;\n dbHasData --\x3e |true| \u6e32\u67d3\u5c55\u793a\u91ca\u4e49\u9762\u677f\u52a8\u753b\n dbHasData --\x3e |false| return_2;"}),(0,l.kt)("h3",{id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard"},"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)"),(0,l.kt)("p",null,"Dashboard \u662f\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPanel")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u4e0d\u540c\u4e8e HUD \u7684\u72ec\u7acb engine \u4e0e method channel"),(0,l.kt)("p",null,"Dashboard \u7684\u4e3b\u8981\u4efb\u52a1\u662f\u4e3a\u7528\u6237\u63d0\u4f9b\u63a7\u5236\u672c\u7a0b\u5e8f\u7684 UI\uff0c\u5305\u62ec\u5feb\u6377\u952e\u8bbe\u7f6e\uff0c\u5f00\u673a\u81ea\u542f\uff0c\u5173\u4e8e\u672c\u7a0b\u5e8f\u7b49\u5e38\u89c1\u7528\u6237\u4ea4\u4e92"),(0,l.kt)("p",null,"\u540c\u65f6\uff0cDashboard \u4f1a\u7ef4\u62a4\u672c\u7a0b\u5e8f\u5728\u5185\u5b58\u548c\u786c\u76d8\u4e2d\u7684\u72b6\u6001\uff0c\u5e76\u5c06\u8fd9\u4e2a\u72b6\u6001\u901a\u8fc7 method channel \u540c\u6b65\u81f3\u4e0b\u9762\u5373\u5c06\u8981\u8bb2\u5230\u7684\u91ca\u4e49\u5c55\u793a\u9762\u677f"),(0,l.kt)("h3",{id:"\u539f\u751f\u4fa7-native"},"\u539f\u751f\u4fa7 (native)"),(0,l.kt)("p",null,"\u539f\u751f\u4fa7\u662f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create")," \u547d\u4ee4\u521b\u5efa\u51fa\u6765\u7684\u4f20\u7edf Xcode \u5de5\u7a0b"),(0,l.kt)("p",null,"\u5728\u539f\u751f\u4fa7\uff0c\u6211\u4e3b\u8981\u5173\u6ce8\u7684\u95ee\u9898\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u5904\u7406\u6765\u81ea ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u7684\u8c03\u7528\uff0c\u5e76\u5728\u9700\u8981\u65f6\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterResult")," \u54cd\u5e94\u5bf9\u5e94\u7684 flutter engine"),(0,l.kt)("li",{parentName:"ul"},"\u7ef4\u62a4\u627f\u8f7d HUD \u548c Dashboard \u7684\u4e24\u4e2a ",(0,l.kt)("inlineCode",{parentName:"li"},"NSWindow"),"\uff0c\u5c24\u5176\u662f\u5f53\u7528\u6237\u7684\u8bbe\u5907\u540c\u65f6\u94fe\u63a5\u591a\u4e2a\u5c4f\u5e55\u65f6\uff0c\u6b63\u786e\u5730\u8bbe\u7f6e HUD \u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u5c01\u88c5\u5e76\u5411 flutter \u63d0\u4f9b\u80fd\u529b\uff0c\u5982\uff1a",(0,l.kt)("ul",{parentName:"li"},(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u53d8\u5316\u548c\u952e\u76d8\u72b6\u6001\u53d8\u5316\uff0c\u5e76\u5c06\u72b6\u6001\u540c\u6b65\u81f3 flutter"),(0,l.kt)("li",{parentName:"ul"},"\u4e3a flutter \u63d0\u4f9b OCR \u548c\u8bed\u97f3\u5408\u6210\u80fd\u529b"),(0,l.kt)("li",{parentName:"ul"},"\u5728\u6267\u884c OCR \u65f6\u4f7f\u7528 swift \u5b9e\u73b0\u5fc5\u8981\u7684\u6027\u80fd\u4f18\u5316")))),(0,l.kt)("h2",{id:"\u6280\u672f\u7ec6\u8282"},"\u6280\u672f\u7ec6\u8282"),(0,l.kt)("p",null,"\u5f97\u76ca\u4e8e Apple \u4e00\u8109\u76f8\u627f\u7684 API \u8bbe\u8ba1\u4ee5\u53ca\u540c\u6837\u7684\u7f16\u7a0b\u8bed\u8a00\uff0c\u719f\u7ec3\u4e8e iOS \u5e94\u7528\u5f00\u53d1(Cocoa Touch & UIKit)\u7684\u7a0b\u5e8f\u5458\u5728\u9762\u5bf9\u4e0e macOS \u4ea4\u4e92\u7684 Cocoa \u548c AppKit \u6846\u67b6\u65f6\uff0c\u53ef\u4ee5\u590d\u7528\u5f88\u591a\u601d\u60f3\u4e0e\u903b\u8f91"),(0,l.kt)("p",null,"\u4f46\u76f8\u6bd4\u4e8e iOS\uff0cmacOS \u7ed9\u4e86\u7528\u6237\u66f4\u5927\u7684\u821e\u53f0\uff0c\u4e5f\u8ba9\u5f00\u53d1\u8005\u9762\u5bf9\u4e86\u66f4\u591a\u7684\u6311\u6218: \u9f20\u6807\uff0c\u952e\u76d8\uff0c\u7a97\u53e3\u548c\u5c4f\u5e55"),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u672c\u7a0b\u5e8f\u7684\u8fc7\u7a0b\u4e2d\uff0c\u6211\u9047\u5230\u4e86\u8bf8\u591a\u7684\u56f0\u96be\uff0c\u4e5f\u5728\u8305\u585e\u987f\u5f00\u65f6\u6536\u83b7\u4e86\u5f88\u591a\u5feb\u4e50\uff0c\u6211\u5728\u8fd9\u4e00\u7ae0\u8282\u4f1a\u8bb0\u5f55\u4e00\u4e0b"),(0,l.kt)("h3",{id:"\u591a\u5c4f\u5e55"},"\u591a\u5c4f\u5e55"),(0,l.kt)("p",null,"macOS \u53ca\u5176\u8fd0\u884c\u7684\u786c\u4ef6\u8bbe\u5907\u5e38\u5e38\u8fde\u63a5\u7740\u591a\u5757\u5c4f\u5e55\uff0c\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u529f\u80fd\u5c31\u662f\u5728\u4efb\u610f\u7684\u5c4f\u5e55\u4f4d\u7f6e\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\uff0c\u8fd9\u5c31\u8981\u786e\u4fdd\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\u653e\u7f6e\u4e8e\u6b63\u786e\u7684\u4f4d\u7f6e\u4e0a"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u8bf7\u6c42 native \u6355\u6349\u5c4f\u5e55\u65f6\uff0c\u6355\u6349\u7684\u77e9\u5f62\u6846\u7684\u4f4d\u7f6e\u6b63\u786e"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u89e3\u6790 ocr response \u540e\uff0c\u7ed3\u679c\u53ef\u4ee5\u6b63\u786e\u5730\u548c\u5c4f\u5e55\u4e0a\u771f\u6b63\u7684\u5185\u5bb9\u5efa\u7acb\u6620\u5c04")),(0,l.kt)("p",null,"\u548c UIKit \u4e2d\u5e38\u7528\u7684 CGRect \u4e0d\u540c\uff0cAppKit \u5bf9\u8bbe\u5907\u5c4f\u5e55\u7684\u62bd\u8c61 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen"),"\uff0c\u5176\u5750\u6807\u4ee5 NSRect \u8ba1\u7b97\uff0c\u5750\u6807\u7cfb\u539f\u70b9\u4e3a macOS \u539f\u59cb\u5c4f\u5e55\u7684\u5de6\u4e0b\u89d2\u3002\u800c\u6b64\u65f6\uff0c\u5982\u679c\u4f60\u7ed9\u4f60\u7684\u8bbe\u5907\u8fde\u63a5\u4e0a\u4e86\u5176\u4ed6\u7684\u5c4f\u5e55\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen.screens")," \u6240\u5448\u73b0\u5c4f\u5e55\u5e03\u5c40\uff0c\u53ef\u80fd\u5c31\u4f1a\u53d8\u6210\u4e0b\u56fe\u6240\u793a\u7684\u6837\u5b50\uff1a\n",(0,l.kt)("img",{alt:"appkit_screens_coordinate",src:n(1589).Z,width:"1630",height:"734"})),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u7684\u5de6\u4e0b\u89d2\u5750\u6807\u503c\u5e76\u975e\u662f (0, 0)\uff0c\u800c\u662f ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 0")," \u7684 (0, 0) \u70b9\u7684\u76f8\u5bf9\u4f4d\u7f6e (w",(0,l.kt)("sub",null,"1"),", d",(0,l.kt)("sub",null,"2"),")\u3002\u901a\u8fc7\u76d1\u542c\u9f20\u6807\u79fb\u52a8\u4e8b\u4ef6\u83b7\u53d6\u7684\u9f20\u6807\u4f4d\u7f6e ",(0,l.kt)("inlineCode",{parentName:"p"},"event.locationInWindow"),"\uff0c\u4e5f\u662f\u76f8\u5bf9\u4e8e\u539f\u70b9\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPoint")),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u65f6\uff0c\u4ec5\u4ec5\u5c06\u81ea\u5df1\u7684\u601d\u7ef4\u4ece iOS \u7684 CGRect \u5750\u6807\u7cfb\u8f6c\u5316\u81f3 NSRect \u5750\u6807\u7cfb\u8fd8\u7b97\u7b80\u5355\u3002\u4f46\u5728\u540e\u7ee7\u7684\u903b\u8f91\u4e2d\uff0c\u56e0\u4e3a OCR \u7ed3\u679c\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRectangleObservation.boundingBox")," \u53c8\u4f1a\u56de\u5230 CGRect \u5750\u6807\u7cfb\uff0c\u5750\u6807\u7cfb\u7684\u9891\u7e41\u8f6c\u5316\u786e\u5b9e\u4f1a\u7ed9\u4eba\u5e26\u6765\u4e00\u5b9a\u7684\u56f0\u6270"),(0,l.kt)("h3",{id:"\u5c4f\u5e55\u6355\u6349"},"\u5c4f\u5e55\u6355\u6349"),(0,l.kt)("p",null,"\u60f3\u6355\u6349 macOS \u7684\u5c4f\u5e55\uff0c\u4f60\u53ef\u4ee5\u8c03\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u51fd\u6570"),(0,l.kt)("p",null,"\u4f46\u503c\u5f97\u6ce8\u610f\u7684\u662f\uff0c\u60f3\u8981\u622a\u53d6\u5c4f\u5e55\u4e0a\u5176\u4ed6\u8fdb\u7a0b\u7684\u5185\u5bb9(\u6bd4\u5982 IDE \u6216\u8005\u6d4f\u89c8\u5668)\uff0c\u9700\u8981\u9884\u5148\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGRequestScreenCaptureAccess"),"\uff0c\u7533\u8bf7\u5230\u622a\u53d6\u5176\u4ed6\u8fdb\u7a0b UI \u7684\u6743\u9650\uff0c\u5426\u5219 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u53ea\u80fd\u62ff\u5230 macOS \u7684\u684c\u9762\uff08\u4ee5\u53ca\u7a0b\u5e8f\u672c\u8eab\uff09"),(0,l.kt)("h3",{id:"ocr"},"OCR"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 Apple \u4e3a\u5f00\u53d1\u8005\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Vision - Recognizing Text")," \u8fdb\u884c\u5c4f\u5e55\u6587\u672c\u7684\u63d0\u53d6"),(0,l.kt)("p",null,"OCR \u8bf7\u6c42\u7684\u8c03\u7528\u7531 flutter side \u5524\u8d77\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'void dispatchOCRRequestToNative() async {\n final x = 100;\n final y = 100;\n final width = 300;\n final height = 80;\n final dimensions = [x, y, width, height];\n final result = await methodChannel.invokeMethod("captureAndOCR", dimensions);\n\n // parse the result from native\n // ...\n}\n')),(0,l.kt)("p",null,"Native side \u5728\u63a5\u6536\u5230\u8bf7\u6c42\u540e\uff0c\u4f1a\u6267\u884c\u622a\u5c4f\u548c\u8bc6\u522b\u64cd\u4f5c\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"import Vision\n// ...\nlet result : FlutterResult = ...\nlet image = captureScreen()\nlet handler = VNImageRequestHandler(cgImage: image)\nlet request = VNRecognizeTextRequest { request, error in\n // ocr finished\n let parsedResult = parse(request.results)\n result(parsedResult)\n}\nhandler.perform([request])\n")),(0,l.kt)("h4",{id:"\u6027\u80fd\u8868\u73b0"},"\u6027\u80fd\u8868\u73b0"),(0,l.kt)("p",null,"\u5728 macbook 2021 \u7684 m1 pro \u4e0a\uff0c\u4ee5 320\u271575 \u7684\u8bbe\u8ba1\u5206\u8fa8\u7387\u622a\u56fe(\u5b9e\u9645\u5206\u8fa8\u7387\u4e3a 640\u2715150)\uff0c\u6bcf\u79d2 11 \u5e27\u7684\u60c5\u51b5\u4e0b\u8fdb\u884c\u957f\u65f6\u95f4\u7684\u622a\u56fe\u548c OCR \u64cd\u4f5c\uff0c\u6574\u4e2a\u6d41\u7a0b\u7684\u5ef6\u8fdf\u5e73\u5747\u7ea6\u4e3a 65ms\uff0c\u6211\u8ba4\u4e3a\u8fd8\u7b97\u662f\u4e00\u4e2a\u53ef\u63a5\u53d7\u7684\u72b6\u6001"),(0,l.kt)("h4",{id:"\u8282\u80fd\u4f18\u5316"},"\u8282\u80fd\u4f18\u5316"),(0,l.kt)("p",null,"\u5373\u4fbf\u662f\u6027\u80fd\u5141\u8bb8\uff0c\u4f18\u5316\u4e5f\u662f\u5e94\u8be5\u505a\u7684\uff0c\u7528\u6237\u4e00\u65e6\u4f7f\u7528\u4f60\u7684 App\uff0c\u5c31\u611f\u89c9 macbook \u7684 C \u9762\u53d1\u70ed\uff0c\u8fd9\u662f\u65e0\u6cd5\u5bb9\u5fcd\u7684"),(0,l.kt)("p",null,"\u5f53\u524d\u5728 OCR \u6d41\u7a0b\u4e2d\u4e3b\u8981\u7684\u6027\u80fd\u4f18\u5316\u6b65\u9aa4\u662f\u5728 native \u7ef4\u62a4\u4e00\u4e2a\u5148\u8fdb\u5148\u51fa\uff0c\u6700\u5927\u5bb9\u91cf\u4e3a 40 \u5e27\u7684\u5b57\u5178\uff0c\u4ee5\u56fe\u7247\u6570\u636e\u4e3a key \u7f13\u5b58 OCR \u7684\u7ed3\u679c\uff0c\u4e0b\u9762\u662f\u7b80\u5316\u540e\u7684\u4ee3\u7801\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"let cacheManager = CacheManager()\n// ...\nlet image = captureScreen()\nlet key = image.dataProvider.data\nif let cachedResult = cacheManager[key] {\n delegate.onOCRResult(cachedResult)\n return\n}\n// ...\ndispatchOCRRequestToDeviceGPU(image) { result\n cacheManager[key] = result\n // ... other logic\n}\n// ...\n")),(0,l.kt)("p",null,"\u5f53\u9f20\u6807\u6307\u9488\u4f4d\u7f6e\u4e0d\u53d8\uff0c\u622a\u56fe\u83b7\u53d6\u7684\u56fe\u7247\u4e0d\u53d8\u65f6\uff0cnative \u5728\u5904\u7406 OCR \u8bf7\u6c42\u65f6\u4f1a\u5148\u547d\u4e2d\u7f13\u5b58\uff0c\u5e76\u76f4\u63a5\u8fd4\u56de\u7ed3\u679c\uff0c\u4ee5\u51cf\u5c11\u975e\u5fc5\u9700\u7684 GPU \u8c03\u7528"),(0,l.kt)("p",null,"\u5728\u7cfb\u7edf\u81ea\u5e26\u7684\u6d3b\u52a8\u76d1\u89c6\u5668\u4e2d\u67e5\u770b\u8fdb\u7a0b\u3002\u53d1\u73b0\uff0c\u5728\u5e94\u7528\u7684\u7f13\u5b58\u7b56\u7565\u540e\uff0c\u5728\u672c\u7a0b\u5e8f\u6d3b\u52a8\u65f6\uff0c\u5176 CPU/GPU \u5360\u7528\u7387\u786e\u5b9e\u964d\u4f4e\u4e86\u5f88\u591a\uff0c\u540c\u65f6\uff0cOCR \u7684\u7f13\u5b58\u7ed3\u679c\u88ab\u964d\u4e3a\u4e86 5ms"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u611f\u89c9\uff0c\u4ee5 ",(0,l.kt)("inlineCode",{parentName:"p"},"CFData")," \u4f5c\u4e3a key \u67e5\u8be2\u5b57\u5178\u8fd8\u4e0d\u662f\u6548\u7387\u6700\u9ad8\u7684\u7b97\u6cd5\uff0c\u5e94\u8be5\u53ef\u4ee5\u7ee7\u7eed\u63a2\u7d22\u4e00\u4e0b")),(0,l.kt)("h4",{id:"\u4e0d\u622a\u53d6\u81ea\u5df1"},"\u4e0d\u622a\u53d6\u81ea\u5df1"),(0,l.kt)("p",null,"\u5728\u6211\u8fdb\u884c\u5f00\u53d1\u65f6\uff0c\u53d1\u73b0\u5f53\u672c\u7a0b\u5e8f\u5728\u5c55\u793a\u5355\u8bcd\u91ca\u4e49 UI (HUD)\u65f6\uff0c\u56e0\u4e3a HUD \u672c\u8eab\u4e5f\u4f1a\u5728\u4e00\u5b9a\u8303\u56f4\u5185\u88ab\u622a\u5c4f\u51fd\u6570\u6355\u83b7\uff0c\u5bfc\u81f4 HUD \u4f1a\u5f71\u54cd OCR \u7684\u7ed3\u679c\uff0c\u800c\u5728 OCR \u7ed3\u679c\u53d8\u52a8\u540e\uff0cHUD \u53c8\u4f1a\u968f\u4e4b\u53d1\u751f\u53d8\u52a8\uff0c\u518d\u6b21\u5f71\u54cd OCR \u7ed3\u679c\uff0c\u5c31\u8fd9\u6837\u5faa\u73af\u5f80\u590d\uff0c\u8fde\u7f13\u5b58\u4e5f\u90fd\u5931\u6548\u4e86\u3002\u5728\u67e5\u9605\u548c\u5c1d\u8bd5\u5927\u91cf\u7684 API \u540e\uff0c\u6211\u7ec8\u4e8e\u5728 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u4e2d\u627e\u5230\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},"sharingType")," \u8fd9\u4e2a\u5c5e\u6027\uff0c\u5c4f\u5e55\u6355\u6349\u65b9\u6cd5\u6355\u83b7 HUD \u81ea\u8eab\uff0c\u603b\u7b97\u662f\u6253\u7834\u4e86\u8fd9\u4e2a\u94fe\u6761"),(0,l.kt)("h3",{id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d"},"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d"),(0,l.kt)("p",null,"\u5bf9 Vision - VNRecognizeTextRequest \u7684\u7ed3\u679c\u89e3\u6790\u662f\u4e2a\u590d\u6742\u7684\u4efb\u52a1"),(0,l.kt)("p",null,"\u6267\u884c\u8fd9\u4e2a\u590d\u6742\u4efb\u52a1\u7684\u539f\u56e0\u6709\u4e09\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u8bbe\u8ba1\u7684\u4ea4\u4e92\u662f\u201c\u7528\u6237\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u611f\u5174\u8da3\u7684\u5355\u8bcd\u4e0a\u65f6\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u201d\uff0c\u8fd9\u5c31\u8981\u6c42\u6211\u4eec\u77e5\u9053\u5c4f\u5e55\u4e0a\u6bcf\u4e2a\u5355\u8bcd\u7684\u5177\u4f53\u4f4d\u7f6e\u548c\u5185\u5bb9"),(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u9488\u5bf9\u6e90\u4ee3\u7801\u7279\u5316\uff0c\u5728\u9762\u5bf9",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E9%A9%BC%E5%B3%B0%E5%BC%8F%E5%A4%A7%E5%B0%8F%E5%86%99"},"\u9a7c\u5cf0\u547d\u540d\u6cd5"),"\u65f6\uff0c\u9700\u8981\u77e5\u9053\u6784\u6210\u4e00\u4e2a symbol \u7684\u6bcf\u4e2a\u5355\u8bcd\u7684\u610f\u601d\uff1a",(0,l.kt)("inlineCode",{parentName:"li"},"ThisIsAVeryVeryLongClassName")," -> ",(0,l.kt)("inlineCode",{parentName:"li"},"This"),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"Is")),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"A")),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Long"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Class"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Name")," (\u56e0\u8fc7\u4e8e\u7b80\u5355\uff0c\u4e22\u5f03\u957f\u5ea6\u5c0f\u4e8e 3 \u7684\u82f1\u8bed\u5355\u8bcd)"),(0,l.kt)("li",{parentName:"ol"},"\u6211\u4eec\u8981\u4ee5\u5355\u8bcd\u7684\u6587\u672c\u5185\u5bb9\u4e3a key\uff0c\u67e5\u8be2\u6570\u636e\u5e93\u3002\u5728\u6587\u672c\u5e8f\u5217\u5305\u542b\u7279\u6b8a\u5b57\u7b26\u65f6\uff0c\u6211\u4eec\u662f\u65e0\u6cd5\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u5230\u5355\u8bcd\u91ca\u4e49\u7684\u3002\u6240\u4ee5\u8981\u79fb\u9664\u975e\u5b57\u6bcd\u5b57\u7b26\u3002\u540c\u65f6\uff0c\u8fd9\u4e00\u64cd\u4f5c\u4e5f\u53ef\u4ee5\u81ea\u7136\u800c\u7136\u5730\u9002\u914d\u7f16\u7a0b\u8bed\u8a00\u4e2d\u7684\u5176\u4ed6\u547d\u540d\u65b9\u5f0f\uff0c\u6bd4\u5982\u79fb\u9664\u4e0b\u5212\u7ebf\u5c31\u53ef\u4ee5\u9002\u5e94",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E8%9B%87%E5%BD%A2%E5%91%BD%E5%90%8D%E6%B3%95"},"\u86c7\u5f62\u547d\u540d\u6cd5"))),(0,l.kt)("p",null,"\u5047\u8bbe\u6211\u4eec\u5728\u5bf9\u4e0b\u9762\u7684\u56fe\u7247\u6267\u884c OCR \u8bf7\u6c42"),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"ocr_text_source",src:n(3591).Z,width:"1038",height:"198"})),(0,l.kt)("p",null,"\u5728 Native \u7aef\u53d6 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizeTextRequest.results.topCandidates(1)")," \u540e\uff0c\u7ed3\u679c\u5982\u4e0b"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision provides its text-recognition capabilities through VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request, an image-based request type that finds and extracts text in images. The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following example shows how to use VNImageRequestHandler to perform a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest for recognizing text in the specified CGImage.",\n]\n')),(0,l.kt)("p",null,"\u5728 Native \u7aef\u4ee5",(0,l.kt)("strong",{parentName:"p"},"\u7a7a\u683c"),"\u5b57\u7b26\u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizedText.boundingBox(for:)")," \u5bf9 OCR \u7ed3\u679c\u8fdb\u884c\u7b2c\u4e00\u6b21\u6574\u7406\uff0c\u4e0a\u8ff0\u7684\u7ed3\u679c\u4f1a\u53d8\u4e3a\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text-recognition", "capabilities", "through", "VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request,", "an", "image-based", "request", "type", "that", "finds", "and", "extracts", "text", "in", "images.", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "to", "use", "VNImageRequestHandler", "to", "perform", "a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest", "for", "recognizing", "text", "in", "the", "specified", "CGImage.",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\u5c06 boundingBox \u7684\u8fd4\u56de\u503c\u548c\u622a\u5c4f\u5c3a\u5bf8\u8fdb\u884c\u4e58\u7b97\uff0c\u8fd4\u56de\u7ed9 flutter \u7aef\u3002Flutter \u7aef\u5728\u62ff\u5230\u7ed3\u679c\u540e\u4e3b\u8981\u8fdb\u884c\u4e24\u4e2a\u6b65\u9aa4\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u79fb\u9664\u975e\u82f1\u6587\u5b57\u7b26\uff0c\u8ba1\u7b97\u65b0\u7684 rect"),(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u5bf9\u5305\u542b\u5927\u5199\u5b57\u6bcd\u7684\u5b57\u7b26\u4e32\u8fdb\u884c\u5206\u5272\uff0c\u8ba1\u7b97\u65b0\u7684 rect")),(0,l.kt)("p",null,"\u8fd9\u6837\uff0c\u6587\u672c\u5757\u7684\u5185\u5bb9\u5728 flutter \u7aef\u5c31\u4f1a\u53d8\u6210\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text", "recognition", "capabilities", "through", "Recognize", "Text",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request","image", "based", "request", "type", "that", "finds", "and", "extracts", "text", "images", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "use", "Image", "Request", "Handler", "perform",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Recognize", "Text", "Request", "for", "recognizing", "text", "the", "specified", "Image",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\uff0cflutter \u4fa7\u4f1a\u4ee5\u4e0b\u9762\u7684\u4ee3\u7801\u5bf9\u8fd9\u4e9b OCR \u4fe1\u606f\u8fdb\u884c\u62bd\u8c61\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},"/// The representation of block\n///\n/// \u6587\u672c\u5757\u8868\u5f81\nclass Block {\n String text;\n double x;\n double y;\n double w;\n double h;\n}\n\n// ...\n\n/// All text representations in memory\n///\n/// \u6240\u6709\u6587\u672c\u5757\u8868\u5f81\nfinal blocks = StateProvider>((_) => []);\n")),(0,l.kt)("p",null,"\u4e0a\u9762\u7684\u6570\u636e\u5728\u5c4f\u5e55\u4e0a\u7684\u53ef\u89c6\u5316\u6548\u679c\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation_2",src:n(6607).Z,width:"1050",height:"184"})),(0,l.kt)("p",null,"\u5176\u4e2d\u5916\u56f4\u7eff\u6846\u4ee3\u8868\u622a\u5c4f\u8303\u56f4\uff0c\u5185\u90e8\u7684\u5c0f\u7eff\u6846\u4ee3\u8868 native \u7aef\u8fd4\u56de\u7684\u7ed3\u679c\uff0c\u5c0f\u7eff\u6846\u5185\u90e8\u7684\u84dd\u8272\u6846\u4ee3\u8868 flutter side \u7b2c\u4e8c\u6b21\u89e3\u6790\u7ed3\u679c\u3002\u53ef\u4ee5\u770b\u5230\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},'"VNImageRequestHandler"')," \u8fd9\u4e2a\u5b57\u7b26\u4e32\u88ab\u5206\u5272\u4e3a\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},'"Image"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Request"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Handler"')," \u8fd9\u4e09\u4e2a\u6587\u672c\u5757\u3002\u8fd9\u6837\uff0c\u6211\u5c31\u53ef\u4ee5\u77e5\u9053\u7528\u6237\u5230\u5e95\u5bf9\u54ea\u4e00\u6bb5\u5b57\u6bcd\u5e8f\u5217\uff08\u5355\u8bcd\uff09\u611f\u5174\u8da3\u4e86\uff08\u5047\u8bbe\u6e90\u4ee3\u7801\u7b26\u53f7\u7684\u547d\u540d\u662f\u89c4\u8303\u7684 \ud83d\ude07\uff09\u3002"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u4f60\u4e5f\u53ef\u4ee5\u6253\u5f00\u672c\u7a0b\u5e8f\u7684\u7528\u6237\u8bbe\u7f6e\u9762\u677f\u6765\u76f4\u63a5\u89c2\u5bdf\u5176\u8fd0\u884c\u65f6\u8868\u73b0\uff1a\n",(0,l.kt)("img",{alt:"dashboard_inspect",src:n(6521).Z,width:"970",height:"192"}))),(0,l.kt)("h3",{id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"},"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,'\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u4f1a\u521b\u5efa\u4e24\u4e2a flutter engine\uff0c\u5206\u522b\u7528\u4e8e\u6e32\u67d3"\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)"\u548c"\u6b22\u8fce\u4e0e\u8bbe\u7f6e\u9875\u9762(Dashboard)"'),(0,l.kt)("p",null,"\u8fd9\u4e24\u4e2a\u5f15\u64ce\u5747\u901a\u8fc7\u57fa\u672c\u65b9\u5f0f\u521b\u5efa\uff0c\u5747\u91c7\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")," \u4e0e native \u8fdb\u884c\u901a\u8baf"),(0,l.kt)("p",null,"\u53cc\u5f15\u64ce\u5171\u4eab\u540c\u4e00\u4efd\u4ee3\u7801\uff0c\u7ecf\u7531\u4e0d\u540c\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/dart-lang/sdk/blob/master/runtime/docs/compiler/aot/entry_point_pragma.md"},"dart \u5165\u53e3\u51fd\u6570"),"\u542f\u52a8\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'// \u8bbe\u7f6e\u9875\u9762 (Dashboard)\ndashboardEngine = FlutterEngine(name: "dashboard", project: nil)\ndashboardChannel = FlutterMethodChannel(name: "dashboard", binaryMessenger: dashboardEngine.binaryMessenger)\ndashboardEngine.run(withEntrypoint: "_dashboard")\n//...\n\n// \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\nhudEngine = FlutterEngine(name: "hud", project: nil)\nhudChannel = FlutterMethodChannel(name: "hud", binaryMessenger: hudEngine.binaryMessenger)\nhudChannel.run(withEntrypoint: "_hud")\n')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u4e2a\u4eba\u6ca1\u6709\u4f7f\u7528\u5b98\u65b9\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://docs.flutter.dev/add-to-app/multiple-flutters"},"FlutterEngineGroup \u65b9\u6848"),"\uff0c\u56e0\u4e3a\u5728\u5f53\u521d\u8fdb\u884c\u5c1d\u8bd5\u65f6\u53d1\u73b0\u4e86\u5f15\u64ce\u65e0\u54cd\u5e94\u7684\u95ee\u9898\uff0c\u81f3\u4eca\u4ecd\u7136\u662f P2 \u7ea7\u522b\u7684 open ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/flutter/flutter/issues/119403"},"issue"))),(0,l.kt)("h4",{id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"},"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u6765\u7ba1\u7406\u7edd\u5927\u90e8\u5206\u72b6\u6001\uff0c\u6240\u4ee5\u5728\u540c\u6b65\u72b6\u6001\u65f6\uff0c\u6211\u4e5f\u671f\u671b\u81ea\u5df1\u7684\u81ea\u5df1\u7684\u5fc3\u667a\u6a21\u578b\u53ef\u4ee5\u66f4\u8d34\u8fd1 riverpod"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0cDashboard engine \u9700\u8981\u5c06\u7528\u6237\u8bbe\u7f6e\u540c\u6b65\u81f3 HUD engine (\u6bd4\u5982\u66f4\u6539\u67e5\u8bcd\u5feb\u6377\u952e)\uff0c\u6211\u5728\u8fd9\u91cc\u76d1\u542c\u4e86 Dashboard engine \u5bf9\u5e94\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," \u7684\u53d8\u66f4\uff0c\u5e76\u901a\u8fc7 method channel\uff0c\u7ecf\u7531 native \u8f6c\u53d1\u81f3 HUD engine\uff0cHUD engine \u5728\u6536\u5230\u4e86\u901a\u77e5\u540e\uff0c\u518d\u66f4\u65b0\u81ea\u5df1\u7ef4\u62a4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider"),"\u3002\u4ece\u800c\u5b9e\u73b0\u4e86\u8de8\u5f15\u64ce\u7684\u72b6\u6001\u540c\u6b65"),(0,l.kt)("h3",{id:"\u91ca\u4e49\u67e5\u8be2"},"\u91ca\u4e49\u67e5\u8be2"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u5728\u67e5\u8be2\u5355\u8bcd\u91ca\u4e49\u65f6\u4f7f\u7528\u672c\u5730 sqlite \u6570\u636e\u5e93\uff0c\u6570\u636e\u5e93\u6e90\u4e8e ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/skywind3000/ECDICT-ultimate"},"ecdict-ultimate")),(0,l.kt)("p",null,"\u9274\u4e8e\u5f00\u6e90\u6570\u636e\u5e93\u8fc7\u4e8e\u5de8\u5927\uff0c\u4e14\u5b58\u5728\u4e00\u4e9b\u672c\u7a0b\u5e8f\u65e0\u9700\u4f7f\u7528\u7684 columns\uff0c\u6211\u53c8\u4f7f\u7528 dart \u548c ",(0,l.kt)("a",{parentName:"p",href:"https://pub.dev/packages/drift"},"drift")," \u5bf9\u5176\u8fdb\u884c\u4e86\u6e05\u6d17\u548c\u526a\u88c1"),(0,l.kt)("p",null,"\u5728\u67e5\u8be2\u65f6\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5c06\u67e5\u8be2\u7684\u672c\u6587\u4e32\u201c\u6253\u6563\u201d\uff0c\u5e76\u4e00\u9f50\u8bf7\u6c42\u6570\u636e\u5e93\uff0c\u4ee5\u671f\u80fd\u547d\u4e2d\u4e0d\u65ad\u5343\u53d8\u4e07\u5316\u7684\u82f1\u8bed\u5355\u8bcd\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'final sequence = "qwertyuiopasdfghjkl";\nfinal sequences = ["qwe","wer","ert",...,"qwer","wert",...,"qwert",...,"qwertyuiopasdfghjkl"];\nfinal List results = await queryDB(keys:sequences); // average latency: ~6ms\n')),(0,l.kt)("p",null,"\u5728\u8bf7\u6c42\u5b8c\u6210\u540e\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5bf9\u8bf7\u6c42\u7ed3\u679c\u8fdb\u884c\u53bb\u91cd\uff0c\u903b\u8f91\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b owl (\u732b\u5934\u9e70)"),(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b knowledge(\u77e5\u8bc6)"),(0,l.kt)("li",{parentName:"ul"},'\u5e8f\u5217 "knowledge" \u5305\u542b "owl"\uff0c\u79fb\u9664\u67e5\u8be2\u7ed3\u679c\u4e2d\u7684 owl')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u76ee\u524d\uff0c\u8be5\u903b\u8f91\u8fd8\u6ca1\u6709\u5b9e\u73b0\u5bf9\u51b2\u7a81\u7684\u5224\u5b9a\uff1a\u5982\u5411\u6570\u636e\u5e93\u67e5\u8be2\u7684 key \u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'"gitignore"'),"\uff0c\u67e5\u8be2\u7ed3\u679c\u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'["git", "tig", "ignore"]'),"\uff0c\u76ee\u524d\u8fd9\u4e09\u4e2a\u7ed3\u679c\u90fd\u4f1a\u88ab\u6e32\u67d3\u5230 HUD \u4e0a\u3002\u4f46\u663e\u7136\uff0c\u9009\u53d6 git \u548c ignore \u8fd9\u4e24\u4e2a\u67e5\u8be2\u7ed3\u679c\u9020\u6210\u7684 \u201c\u51b2\u7a81\u201d \u662f\u6700\u5c0f\u7684")),(0,l.kt)("h3",{id:"\u53d1\u97f3"},"\u53d1\u97f3"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/avfoundation/speech_synthesis"},"AVFoundation - Speech synthesis")," API \u6765\u5b9e\u65f6\u5408\u6210\u8bed\u97f3"),(0,l.kt)("p",null,"\u8be5 API \u53ef\u4ee5\u540c\u65f6\u5728 iOS/macOS \u4e0a\u4f7f\u7528\uff0c\u4e14\u53d1\u97f3\u8f83\u4e3a\u51c6\u786e\uff0c\u4ee5\u6211\u81ea\u8eab\u7684\u80fd\u529b(\u5168\u56fd\u5927\u5b66\u82f1\u8bed\u56db\u7ea7)\u6765\u770b\uff0c\u6548\u679c\u8fd8\u7b97\u6ee1\u610f\uff0c\u786e\u5b9e\u6bd4\u6211\u7684\u53d1\u97f3\u51c6 \ud83e\udd23"),(0,l.kt)("p",null,"\u5f53\u7136\uff0c\u540e\u7ee7\u5982\u679c\u6709\u66f4\u9ad8\u7684\u9700\u6c42\uff0c\u4e5f\u6709\u5f88\u591a\u5176\u4ed6\u53ef\u9009\u65b9\u6848"))}c.isMDXComponent=!0},3591:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/ocr_text_source-9af654c22d465590f3079fd8d4f2f24f.png"},1589:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/appkit_screens_coordinate-94563877ba8462b8412c6e945e1f7b5e.png"},6521:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/dashboard_inspect-0a3c0a7bba8e62a06a697e1555bb3ea4.png"},2731:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/intro-663e889360c7a0e605df52af33af3b11.gif"},8555:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_dark-ede87b48b4273a15fc0974c77b5586cf.png"},1974:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_light-980bb1be0d553350c1dcf22e0b4325e2.png"},3642:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/project_layout-b9e52932db96541ab24f4ed06bc28bc0.png"},9462:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation-e7dc02491d5b436f2ef66bd928e65e17.png"},6607:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation_1-7c6db883d1b12b61355048e553101f21.png"}}]); \ No newline at end of file diff --git a/assets/js/f812cb1f.6b84295d.js b/assets/js/f812cb1f.6b84295d.js new file mode 100644 index 0000000..ab6ab73 --- /dev/null +++ b/assets/js/f812cb1f.6b84295d.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdf_doc_source=self.webpackChunkdf_doc_source||[]).push([[594],{3905:(e,t,n)=>{n.d(t,{Zo:()=>d,kt:()=>k});var a=n(7294);function l(e,t,n){return t in e?Object.defineProperty(e,t,{value:n,enumerable:!0,configurable:!0,writable:!0}):e[t]=n,e}function r(e,t){var n=Object.keys(e);if(Object.getOwnPropertySymbols){var a=Object.getOwnPropertySymbols(e);t&&(a=a.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),n.push.apply(n,a)}return n}function i(e){for(var t=1;t=0||(l[n]=e[n]);return l}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(a=0;a=0||Object.prototype.propertyIsEnumerable.call(e,n)&&(l[n]=e[n])}return l}var p=a.createContext({}),s=function(e){var t=a.useContext(p),n=t;return e&&(n="function"==typeof e?e(t):i(i({},t),e)),n},d=function(e){var t=s(e.components);return a.createElement(p.Provider,{value:t},e.children)},u="mdxType",c={inlineCode:"code",wrapper:function(e){var t=e.children;return a.createElement(a.Fragment,{},t)}},m=a.forwardRef((function(e,t){var n=e.components,l=e.mdxType,r=e.originalType,p=e.parentName,d=o(e,["components","mdxType","originalType","parentName"]),u=s(n),m=l,k=u["".concat(p,".").concat(m)]||u[m]||c[m]||r;return n?a.createElement(k,i(i({ref:t},d),{},{components:n})):a.createElement(k,i({ref:t},d))}));function k(e,t){var n=arguments,l=t&&t.mdxType;if("string"==typeof e||l){var r=n.length,i=new Array(r);i[0]=m;var o={};for(var p in t)hasOwnProperty.call(t,p)&&(o[p]=t[p]);o.originalType=e,o[u]="string"==typeof e?e:l,i[1]=o;for(var s=2;s{n.r(t),n.d(t,{assets:()=>p,contentTitle:()=>i,default:()=>c,frontMatter:()=>r,metadata:()=>o,toc:()=>s});var a=n(7462),l=(n(7294),n(3905));const r={slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},i=void 0,o={permalink:"/valo-reader-doc/blog/tech_detail_macos",editUrl:"https://github.com/facebook/docusaurus/tree/main/packages/create-docusaurus/templates/shared/blog/tech_detail_macos.md",source:"@site/blog/tech_detail_macos.md",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",description:"\u6458\u8981",date:"2023-10-19T05:59:23.000Z",formattedDate:"2023\u5e7410\u670819\u65e5",tags:[{label:"technical",permalink:"/valo-reader-doc/blog/tags/technical"}],readingTime:23.58,hasTruncateMarker:!1,authors:[{name:"Ce Wang",title:"Solo dev",url:"https://github.com/haloWang",imageURL:"https://github.com/haloWang.png",key:"halowang"}],frontMatter:{slug:"tech_detail_macos",title:"Valo Reader for macOS \u6280\u672f\u6982\u8981",authors:["halowang"],tags:["technical"]},nextItem:{title:"\u5f00\u53d1\u76ee\u7684",permalink:"/valo-reader-doc/blog/motivation"}},p={authorsImageUrls:[void 0]},s=[{value:"\u6458\u8981",id:"\u6458\u8981",level:2},{value:"\u4ec0\u4e48\u662f Valo Reader?",id:"\u4ec0\u4e48\u662f-valo-reader",level:2},{value:"\u9879\u76ee\u5e03\u5c40",id:"\u9879\u76ee\u5e03\u5c40",level:2},{value:"\u5de5\u7a0b\u67b6\u6784",id:"\u5de5\u7a0b\u67b6\u6784",level:2},{value:"flutter \u4fa7",id:"flutter-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f",id:"\u4e3b\u7a0b\u5e8f",level:4},{value:"flutter package: \u6587\u672c\u5757\u8868\u5f81",id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81",level:4},{value:"flutter package: \u57fa\u7840\u5e93",id:"flutter-package-\u57fa\u7840\u5e93",level:4},{value:"native \u4fa7",id:"native-\u4fa7",level:3},{value:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206",level:4},{value:"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93",id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93",level:4},{value:"CocoaPods dependency: OCR \u529f\u80fd\u5e93",id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93",level:4},{value:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757",level:2},{value:"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)",id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud",level:3},{value:"\u5728 HUD \u4e2d\u7684\u72b6\u6001",id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001",level:4},{value:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)",id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard",level:3},{value:"\u539f\u751f\u4fa7 (native)",id:"\u539f\u751f\u4fa7-native",level:3},{value:"\u6280\u672f\u7ec6\u8282",id:"\u6280\u672f\u7ec6\u8282",level:2},{value:"\u591a\u5c4f\u5e55",id:"\u591a\u5c4f\u5e55",level:3},{value:"\u5c4f\u5e55\u6355\u6349",id:"\u5c4f\u5e55\u6355\u6349",level:3},{value:"OCR",id:"ocr",level:3},{value:"\u6027\u80fd\u8868\u73b0",id:"\u6027\u80fd\u8868\u73b0",level:4},{value:"\u8282\u80fd\u4f18\u5316",id:"\u8282\u80fd\u4f18\u5316",level:4},{value:"\u4e0d\u622a\u53d6\u81ea\u5df1",id:"\u4e0d\u622a\u53d6\u81ea\u5df1",level:4},{value:"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d",id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d",level:3},{value:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65",level:3},{value:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65",level:4},{value:"\u91ca\u4e49\u67e5\u8be2",id:"\u91ca\u4e49\u67e5\u8be2",level:3},{value:"\u53d1\u97f3",id:"\u53d1\u97f3",level:3}],d={toc:s},u="wrapper";function c(e){let{components:t,...r}=e;return(0,l.kt)(u,(0,a.Z)({},d,r,{components:t,mdxType:"MDXLayout"}),(0,l.kt)("h2",{id:"\u6458\u8981"},"\u6458\u8981"),(0,l.kt)("p",null,"\u672c\u6587\u4e3b\u8981\u4ecb\u7ecd\u4e86 Valo Reader for macOS\uff08\u4e0b\u6587\u7b80\u79f0\u4e3a\u201c\u672c\u7a0b\u5e8f\u201d\uff09\u7684",(0,l.kt)("a",{parentName:"p",href:"#%E9%A1%B9%E7%9B%AE%E5%B8%83%E5%B1%80"},"\u9879\u76ee\u5e03\u5c40"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E5%B7%A5%E7%A8%8B%E6%9E%B6%E6%9E%84"},"\u5de5\u7a0b\u67b6\u6784"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"#%E8%BF%90%E8%A1%8C%E6%97%B6%E5%8A%9F%E8%83%BD%E6%A8%A1%E5%9D%97"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),"\u4e0e",(0,l.kt)("a",{parentName:"p",href:"#%E6%8A%80%E6%9C%AF%E7%BB%86%E8%8A%82"},"\u6280\u672f\u7ec6\u8282"),"\u3002\u4f60\u8fd8\u53ef\u4ee5\u5728",(0,l.kt)("a",{parentName:"p",href:"/docs/intro"},"\u672c\u6587\u6863"),"\u7684\u5176\u4ed6\u9875\u9762\u4e86\u89e3\u672c\u7a0b\u5e8f\u7684",(0,l.kt)("a",{parentName:"p",href:"/valo-reader-doc/blog/motivation"},"\u5f00\u53d1\u52a8\u673a"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/usage"},"\u4f7f\u7528\u65b9\u5f0f"),"\uff0c",(0,l.kt)("a",{parentName:"p",href:"/docs/privacy_security"},"\u9690\u79c1\u4e0e\u5b89\u5168"),"\u7b49\u5185\u5bb9"),(0,l.kt)("h2",{id:"\u4ec0\u4e48\u662f-valo-reader"},"\u4ec0\u4e48\u662f Valo Reader?"),(0,l.kt)("p",null,"Valo Reader \u8fd9\u4e2a\u9879\u76ee\u662f\u6211\u4e2a\u4eba\u5728\u65e5\u5e38\u751f\u6d3b\u4e2d\u9010\u6e10\u840c\u751f\uff0c\u6f14\u5316\u548c\u5b9e\u65bd\u7684\u60f3\u6cd5\u3002\u662f\u6211\u4e3a\u4e86\u5728\u81ea\u5df1\u7535\u5b50\u8bbe\u5907\u8bbe\u5907\u4e0a\u80fd\u8f7b\u677e\u9605\u8bfb\u82f1\u6587\u5185\u5bb9\uff0c\u5e76\u6e10\u8fd1\u5f0f\u5730\u63d0\u9ad8\u81ea\u5df1\u7684\u82f1\u6587\u6c34\u5e73\uff0c\u800c\u81ea\u884c\u8bbe\u8ba1\u548c\u7814\u53d1\u7684\u4e00\u6b3e\u4ea7\u54c1\uff0c\u5176\u6838\u5fc3\u529f\u80fd\u5c55\u793a\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"intro",src:n(2731).Z,width:"960",height:"268"})),(0,l.kt)("p",null,"\u5f53\u4f60\u5728\u4f7f\u7528\u672c\u7a0b\u5e8f\u65f6\uff0c\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u4f60\u611f\u5230\u751f\u758f\u7684\u82f1\u8bed\u5355\u8bcd\u4e0a\u3002\u6309\u4e0b\u6307\u5b9a\u7684\u6309\u952e\uff08\u9ed8\u8ba4\u4e3a Fn\uff09\uff0c\u672c\u7a0b\u5e8f\u5373\u4f1a\u5728\u54cd\u5e94\u7684\u4f4d\u7f6e\u5c55\u793a\u5bf9\u5e94\u7684\u91ca\u4e49\uff0c\u540c\u65f6\u4f7f\u7528\u8bed\u97f3\u5408\u6210\u6765\u53d1\u97f3\u3002"),(0,l.kt)("p",null,"\u76ee\u524d\uff0c\u4f60\u53ef\u4ee5\u901a\u8fc7 ",(0,l.kt)("a",{parentName:"p",href:"https://apps.apple.com/cn/app/valo-reader/id6448040931"},"Mac App Store")," \u6216",(0,l.kt)("a",{parentName:"p",href:"/docs/installation"},"\u5176\u4ed6\u65b9\u5f0f"),"\u6765\u83b7\u53d6\u672c\u7a0b\u5e8f"),(0,l.kt)("h2",{id:"\u9879\u76ee\u5e03\u5c40"},"\u9879\u76ee\u5e03\u5c40"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u9879\u76ee\u7684\u4e3b\u4f53\u7ed3\u6784\uff1a\n",(0,l.kt)("img",{alt:"project_layout",src:n(3642).Z,width:"1996",height:"960"})),(0,l.kt)("p",null,"\u672c\u6587\u7684\u4f4d\u4e8e\u4e0a\u56fe\u4e2d\u7684\u6d45\u9ec4\u8272\u90e8\u5206\uff0c\u7531 ",(0,l.kt)("a",{parentName:"p",href:"https://docusaurus.io/"},"docusaurus")," \u751f\u6210"),(0,l.kt)("p",null,"\u672c\u9879\u76ee\u7684\u60f3\u6cd5\u6700\u65e9\u5b9e\u73b0\u4e8e\u5728\u6d4f\u89c8\u5668\u4e2d\u8fd0\u884c\u7684 js \u811a\u672c\uff0c\u4f60\u53ef\u4ee5\u5728 github \u4e0a\u770b\u5230\u672c\u9879\u76ee\u5728\u6d4f\u89c8\u5668\u4e0a\u7684\u65e9\u671f\u5b9e\u73b0\u53ca\u5bf9\u5e94\u7684",(0,l.kt)("a",{parentName:"p",href:"https://github.com/HaloWang/english_flow#%E5%8A%9F%E8%83%BD%E5%B1%95%E7%A4%BA"},"\u529f\u80fd\u5c55\u793a"),"\uff0c\u5373\u56fe\u4e2d\u6d45\u7eff\u8272\u90e8\u5206"),(0,l.kt)("p",null,"\u800c\u672c\u6587\u4e3b\u8981\u8bf4\u660e\u4e86\u4e0a\u56fe\u4e2d macOS\uff08\u6d45\u84dd\u8272\u90e8\u5206\uff09\u7684\u8fd0\u884c\u65b9\u5f0f\u4e0e\u6280\u672f\u7ec6\u8282"),(0,l.kt)("h2",{id:"\u5de5\u7a0b\u67b6\u6784"},"\u5de5\u7a0b\u67b6\u6784"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create XXX --platform macos")," \u547d\u4ee4\u521b\u5efa\uff0c\u5176\u6e90\u4ee3\u7801\u4e3b\u8981\u5206\u4e3a\u4e24\u90e8\u5206"),(0,l.kt)("h3",{id:"flutter-\u4fa7"},"flutter \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 dart \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f"},"\u4e3b\u7a0b\u5e8f"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u7684\u4e3b\u8981\u529f\u80fd\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u901a\u8fc7 method channel \u5b9e\u73b0 flutter \u4e0e native \u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u7528\u6237\u8bbe\u7f6e\u9762\u677f\uff08Dashboard\uff09"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09\u4e0e\u8bca\u65ad\u68c0\u67e5\u4fe1\u606f"),(0,l.kt)("li",{parentName:"ol"},"\u4f7f\u7528\u7b2c\u4e09\u65b9\u4f9d\u8d56\u63d0\u4f9b\u7684\u529f\u80fd")),(0,l.kt)("h4",{id:"flutter-package-\u6587\u672c\u5757\u8868\u5f81"},"flutter package: \u6587\u672c\u5757\u8868\u5f81"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u662f\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u4ee3\u7801\uff0c\u5728 macOS / iOS / Android \u4e09\u7aef\u5171\u4eab\u90e8\u5206\u903b\u8f91\u4e0e UI\uff0c\u5176\u4e3b\u8981\u529f\u80fd\u7531\u6709\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u89e3\u6790\u7528\u6237\u8bbe\u5907\u5c4f\u5e55\u4e0a\u8bc6\u522b\u5230\u7684\u6587\u672c\u5757\uff0c\u7ed3\u6784\u5316\u6587\u672c\u5757\uff0c\u5e76\u5c06\u5176\u7ef4\u62a4\u5230\u672c\u7a0b\u5e8f\u7684\u5185\u5b58\u4e2d"),(0,l.kt)("li",{parentName:"ol"},"\u58f0\u660e\u5e76\u5b9e\u73b0\u7528\u6237\u89c6\u89c9\u7126\u70b9\u4e0e\u5df2\u77e5\u7684\u6587\u672c\u4fe1\u606f\u7684\u4ea4\u4e92"),(0,l.kt)("li",{parentName:"ol"},"\u6e32\u67d3\u91ca\u4e49\u5c55\u793a\u9762\u677f\uff08HUD\uff09")),(0,l.kt)("h4",{id:"flutter-package-\u57fa\u7840\u5e93"},"flutter package: \u57fa\u7840\u5e93"),(0,l.kt)("p",null,"\u5728\u6211\u7684\u6240\u6709 flutter \u9879\u76ee\u4e2d\u5171\u4eab\u7684\u57fa\u7840\u4f9d\u8d56"),(0,l.kt)("h3",{id:"native-\u4fa7"},"native \u4fa7"),(0,l.kt)("p",null,"\u8fd9\u90e8\u5206\u7531 swift \u7f16\u5199\uff0c\u53ef\u5206\u4e3a\u4e09\u90e8\u5206"),(0,l.kt)("h4",{id:"\u4e3b\u7a0b\u5e8f\u90e8\u5206"},"\u4e3b\u7a0b\u5e8f\u90e8\u5206"),(0,l.kt)("p",null,"\u6301\u6709 flutter engine\uff0c\u901a\u8fc7 method channel \u6253\u901a flutter \u4e0e ",(0,l.kt)("inlineCode",{parentName:"p"},"Base"),"\u3001",(0,l.kt)("inlineCode",{parentName:"p"},"OCR")," \u4ee5\u53ca\u5e94\u7528\u7a0b\u5e8f\u672c\u8eab\u7684\u53cc\u5411\u8c03\u7528"),(0,l.kt)("p",null,"\u521b\u5efa\u9879\u76ee\u65f6\u9ed8\u8ba4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController")," \u5df2\u88ab\u5220\u9664"),(0,l.kt)("h4",{id:"cocoapods-dependency-cocoa-\u529f\u80fd\u5e93"},"CocoaPods dependency: Cocoa \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e Cocoa framework \u4ea4\u4e92\uff0c\u5982\uff1a\u6743\u9650\u7533\u8bf7\uff0c\u952e\u76d8\u76d1\u542c\uff0c\u9f20\u6807\u76d1\u542c\uff0c\u622a\u53d6\u5c4f\u5e55\uff0c\u79fb\u52a8\u627f\u8f7d flutter \u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow"),"\u4ee5\u53ca\u63d0\u4f9b\u57fa\u672c\u82e5\u5e72\u539f\u751f\u80fd\u529b"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u6211\u4e5f\u4f1a\u5728\u8be5\u4f9d\u8d56\u4e2d\u5b66\u4e60\uff0c\u5c1d\u8bd5\u548c\u5b9e\u73b0 Cocoa/AppKit \u72ec\u6709\u7684 API \u4e0e\u529f\u80fd"),(0,l.kt)("h4",{id:"cocoapods-dependency-ocr-\u529f\u80fd\u5e93"},"CocoaPods dependency: OCR \u529f\u80fd\u5e93"),(0,l.kt)("p",null,"\u4e0e ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Apple Vision framework - Text Recognizing")," \u4ea4\u4e92\uff0c\u5e76\u5b9e\u73b0\u90e8\u5206\u903b\u8f91\uff0c\u5982\uff1a\u53d1\u8d77\u6587\u672c\u8bc6\u522b\u8bf7\u6c42\uff0c\u7ef4\u62a4\u6587\u672c\u8bc6\u522b\u54cd\u5e94\u7f13\u5b58"),(0,l.kt)("p",null,"\u8be5\u90e8\u5206\u4f7f\u7528 CocoaPods \u521b\u5efa\uff0c\u5728 macOS \u548c iOS \u9879\u76ee\u540c\u65f6\u4f9d\u8d56\u5e76\u5171\u4eab\u4ee3\u7801"),(0,l.kt)("h2",{id:"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"},"\u8fd0\u884c\u65f6\u529f\u80fd\u6a21\u5757"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0c\u672c\u7a0b\u5e8f\u4e3b\u8981\u53ef\u5206\u4e3a\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)\u3001\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD) \u548c\u539f\u751f\u4fa7 (native) \u4e09\u4e2a\u6a21\u5757"),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u7684\u4e3b\u8981\u6a21\u5757\u53ca\u5176\u901a\u8baf\uff1a\n",(0,l.kt)("img",{alt:"modules_light",src:n(1974).Z+"#gh-light-mode-only",width:"1926",height:"894"}),"\n",(0,l.kt)("img",{alt:"modules_dark",src:n(8555).Z+"#gh-dark-mode-only",width:"1926",height:"894"})),(0,l.kt)("h3",{id:"\u91ca\u4e49\u5c55\u793a\u9762\u677f-hud"},"\u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)"),(0,l.kt)("p",null,"HUD \u662f\u4e00\u4e2a\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u81ea\u5df1\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterEngine")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")),(0,l.kt)("p",null,"HUD \u7684\u4e3b\u8981\u4efb\u52a1\u5305\u542b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u5411 native \u6d3e\u53d1 ocr \u8bf7\u6c42"),(0,l.kt)("li",{parentName:"ul"},"\u89e3\u6790 native \u5bf9\u5c4f\u5e55\u6307\u5b9a\u533a\u57df\u7684 ocr \u7ed3\u679c\uff0c\u5e76\u5728\u5185\u5b58\u4e2d\u7ef4\u62a4\u8be5\u7ed3\u679c\u4f9b\u540e\u7ee7\u7a0b\u5e8f\u903b\u8f91\u4f7f\u7528"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u4ee5\u5728\u9700\u8981\u65f6\u8ba1\u7b97\u91ca\u4e49\u5c55\u793a\u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u952e\u76d8\u6309\u952e\u72b6\u6001\uff0c\u5728\u53d8\u5316\u65f6\u6d3e\u53d1 ocr \u8bf7\u6c42\u5e76\u5c55\u793a\u5355\u8bcd\u91ca\u4e49"),(0,l.kt)("li",{parentName:"ul"},"\u5b9e\u65f6\u540c\u6b65 Dashboard engine \u4f20\u9012\u8fc7\u6765\u7684\u7528\u6237\u8bbe\u7f6e\uff0c\u5e76\u4fee\u6539\u5c55\u793a\u903b\u8f91")),(0,l.kt)("p",null,"\u4e0b\u56fe\u5c55\u793a\u4e86\u8be5\u6a21\u5757\u6240\u7ef4\u62a4\u7684\u6570\u636e\u6620\u5c04\u5230\u5c4f\u5e55\u4e0a\u65f6\u7684\u53ef\u89c6\u5316\u6548\u679c\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation",src:n(9462).Z,width:"1898",height:"222"})),(0,l.kt)("h4",{id:"\u5728-hud-\u4e2d\u7684\u72b6\u6001"},"\u5728 HUD \u4e2d\u7684\u72b6\u6001"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u7ba1\u7406\u7edd\u5927\u90e8\u5206\u7684\u72b6\u6001\u3002\u5728\u6784\u7b51\u672c\u7a0b\u5e8f\u65f6\uff0c\u6211\u7684\u601d\u8003\u8fc7\u7a0b\u4e3b\u8981\u57fa\u4e8e\u72b6\u6001\u4e0e\u72b6\u6001\u53d8\u5316\uff0c\u4ece\u5c55\u793a\u91ca\u4e49\u952e\u70b9\u51fb\u5230\u6e32\u67d3\u91ca\u4e49\u9762\u677f\u52a8\u753b"),(0,l.kt)("p",null,"\u5728\u5927\u91cf\u4f7f\u7528 riverpod \u4e2d\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"Provider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," / ",(0,l.kt)("inlineCode",{parentName:"p"},"ProviderContainer.listen")," \u540e\uff0c\u6211\u53ef\u4ee5\u6784\u5efa\u4e00\u4e2a\u9ad8\u5ea6\u5f02\u6b65\uff0c\u8c03\u7528\u987a\u5e8f\u4e0d\u654f\u611f\uff0c\u72b6\u6001\u53d8\u5316\u9a71\u52a8\u7684\u5e94\u7528\u7a0b\u5e8f\u3002\u5728\u7f16\u7801\u8fc7\u7a0b\u4e2d\uff0c\u8fd9\u79cd\u65b9\u5f0f\u8ba9\u7a0b\u5e8f\u5458\u53ef\u4ee5\u4ece\u4e00\u5927\u4e32\u7684\u547d\u4ee4\u5f0f(Imperative)\u65b9\u6cd5\u8c03\u7528\u4e2d\u89e3\u653e\u51fa\u6765\uff0c\u4ec5\u4ec5\u9700\u8981\u5173\u6ce8\u548c\u786e\u4fdd\u6bcf\u4e2a\u6700\u5c0f\u903b\u8f91\u5355\u5143\u2014\u2014",(0,l.kt)("inlineCode",{parentName:"p"},"Provider"),"\uff0c\u786e\u4fdd\u5176\u5b9e\u73b0\u662f\u6b63\u786e\u7684\u5373\u53ef\uff0c\u8fd9\u51cf\u5c11\u4e86\u6784\u5efa\u590d\u6742\u548c\u957f\u4e32\u903b\u8f91\u51fa\u9519\u7684\u53ef\u80fd"),(0,l.kt)("p",null,"\u4e0b\u9762\u7684\u4e24\u5f20\u56fe\u8868\u5c55\u793a\u4e86\u672c\u7a0b\u5e8f\u7684\u4e3b\u8981\u529f\u80fd\u6d41\u7a0b"),(0,l.kt)("p",null,"\u5207\u6362\u5c55\u793a\u91ca\u4e49\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:'graph LR\n A1[\u76d1\u542c\u952e\u76d8\u72b6\u6001];\n A[\u952e\u76d8\u72b6\u6001\u53d1\u751f\u6539\u53d8];\n query_button_down{\u5c55\u793a\u91ca\u4e49\u952e\u6309\u4e0b?};\n C[\u505c\u6b62\u5b9a\u65f6\u5668];\n D[\u542f\u52a8\u5b9a\u65f6\u5668\u5e76\u91cd\u7f6e\u6587\u672c\u5757\u72b6\u6001];\n mouse_state[\u9f20\u6807\u72b6\u6001];\n screen_state[\u5c4f\u5e55\u72b6\u6001];\n send_request_to_native["\u5411 native \u53d1\u9001\u622a\u56fe\u4e0e OCR \u8bf7\u6c42"];\n F[native OCR \u7ed3\u679c\u5df2\u83b7\u5f97];\n G[\u89e3\u6790 OCR \u7ed3\u679c];\n H1[\u66f4\u65b0\u6587\u672c\u5757\u72b6\u6001];\n\n mouse_state--\x3esend_request_to_native\n screen_state--\x3esend_request_to_native\n send_request_to_native-.->F;\n F--\x3eG;\n G--\x3eH1;\n A1-.->A;\n A--\x3equery_button_down;\n query_button_down--\x3e|false|C;\n query_button_down--\x3e|true|D;\n D-.->send_request_to_native;'}),(0,l.kt)("p",null,"\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u53d8\u66f4\u6d41\u7a0b\uff1a"),(0,l.kt)("mermaid",{value:"graph LR;\n H2[\u6587\u672c\u5757\u72b6\u6001\u53d8\u66f4];\n most_wanted_state_logic[\u7528\u6237\u5173\u6ce8\u6587\u672c\u5757\u76d1\u542c\u903b\u8f91];\n most_wanted_state_changed[\u72b6\u6001\u6539\u53d8];\n mouse_change0[\u76d1\u542c\u9f20\u6807\u72b6\u6001];\n mouse_change[\u9f20\u6807\u72b6\u6001\u6539\u53d8];\n sync_mouse_to_flutter[\u540c\u6b65\u9f20\u6807\u72b6\u6001\u81f3 flutter];\n sync_current_screen[\u540c\u6b65\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u6570\u636e];\n sync_current_screen_to_flutter[\u540c\u6b65\u5c4f\u5e55\u6570\u636e\u81f3 flutter];\n hasTextBlock{\u72b6\u6001\u7684\u503c\u662f\u5426\u4e3a\u7a7a\\n\u5373\u9f20\u6807\u5f53\u524d\u4f4d\u7f6e\u6ca1\u6709\u8bc6\u522b\u5230\u6587\u672c};\n queryDB[\u67e5\u8be2\u6570\u636e\u5e93];\n dbHasData{\u6570\u636e\u5e93\u5305\u542b\u91ca\u4e49};\n hideAllFlow[\u9690\u85cf\u6240\u6709\u91ca\u4e49\u9762\u677f];\n asParam[\u4f5c\u4e3aOCR\u8bf7\u6c42\u7684\u53c2\u6570];\n screen_changed{\u9f20\u6807\u6240\u5728\u5c4f\u5e55\u53d1\u751f\u53d8\u52a8?}\n\n \u5207\u6362\u5c55\u793a\u91ca\u4e49-.->H2\n mouse_change0-.->mouse_change--\x3esync_mouse_to_flutter;\n mouse_change --\x3e sync_current_screen --\x3e sync_current_screen_to_flutter;\n sync_current_screen_to_flutter --\x3e \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001\n sync_current_screen --\x3e screen_changed --\x3e |true| \u6539\u53d8\u627f\u8f7dHUD\u7684NSPanel\u4f4d\u7f6e;\n screen_changed --\x3e |false| return_0;\n H2 --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter --\x3e most_wanted_state_logic;\n sync_mouse_to_flutter -.-> asParam;\n \u66f4\u65b0\u5c4f\u5e55\u72b6\u6001 -.-> asParam;\n most_wanted_state_logic --\x3e most_wanted_state_changed --\x3e hasTextBlock;\n \u5c55\u793a\u91ca\u4e49\u952e\u72b6\u6001 --\x3e most_wanted_state_logic;\n most_wanted_state_changed --\x3e hideAllFlow;\n hasTextBlock --\x3e |false| return_1;\n hasTextBlock --\x3e |true| queryDB;\n queryDB --\x3e dbHasData;\n dbHasData --\x3e |true| \u6e32\u67d3\u5c55\u793a\u91ca\u4e49\u9762\u677f\u52a8\u753b\n dbHasData --\x3e |false| return_2;"}),(0,l.kt)("h3",{id:"\u7528\u6237\u8bbe\u7f6e\u9762\u677f-dashboard"},"\u7528\u6237\u8bbe\u7f6e\u9762\u677f (Dashboard)"),(0,l.kt)("p",null,"Dashboard \u662f\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPanel")," \u5b9e\u4f8b\uff0c\u5176\u5185\u90e8\u627f\u8f7d\u4e86\u4e00\u4e2a ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterViewController"),"\uff0c\u5e76\u7ed1\u5b9a\u4e86\u4e0d\u540c\u4e8e HUD \u7684\u72ec\u7acb engine \u4e0e method channel"),(0,l.kt)("p",null,"Dashboard \u7684\u4e3b\u8981\u4efb\u52a1\u662f\u4e3a\u7528\u6237\u63d0\u4f9b\u63a7\u5236\u672c\u7a0b\u5e8f\u7684 UI\uff0c\u5305\u62ec\u5feb\u6377\u952e\u8bbe\u7f6e\uff0c\u5f00\u673a\u81ea\u542f\uff0c\u5173\u4e8e\u672c\u7a0b\u5e8f\u7b49\u5e38\u89c1\u7528\u6237\u4ea4\u4e92"),(0,l.kt)("p",null,"\u540c\u65f6\uff0cDashboard \u4f1a\u7ef4\u62a4\u672c\u7a0b\u5e8f\u5728\u5185\u5b58\u548c\u786c\u76d8\u4e2d\u7684\u72b6\u6001\uff0c\u5e76\u5c06\u8fd9\u4e2a\u72b6\u6001\u901a\u8fc7 method channel \u540c\u6b65\u81f3\u4e0b\u9762\u5373\u5c06\u8981\u8bb2\u5230\u7684\u91ca\u4e49\u5c55\u793a\u9762\u677f"),(0,l.kt)("h3",{id:"\u539f\u751f\u4fa7-native"},"\u539f\u751f\u4fa7 (native)"),(0,l.kt)("p",null,"\u539f\u751f\u4fa7\u662f\u7531 ",(0,l.kt)("inlineCode",{parentName:"p"},"flutter create")," \u547d\u4ee4\u521b\u5efa\u51fa\u6765\u7684\u4f20\u7edf Xcode \u5de5\u7a0b"),(0,l.kt)("p",null,"\u5728\u539f\u751f\u4fa7\uff0c\u6211\u4e3b\u8981\u5173\u6ce8\u7684\u95ee\u9898\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u5904\u7406\u6765\u81ea ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterMethodChannel")," \u7684\u8c03\u7528\uff0c\u5e76\u5728\u9700\u8981\u65f6\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"li"},"FlutterResult")," \u54cd\u5e94\u5bf9\u5e94\u7684 flutter engine"),(0,l.kt)("li",{parentName:"ul"},"\u7ef4\u62a4\u627f\u8f7d HUD \u548c Dashboard \u7684\u4e24\u4e2a ",(0,l.kt)("inlineCode",{parentName:"li"},"NSWindow"),"\uff0c\u5c24\u5176\u662f\u5f53\u7528\u6237\u7684\u8bbe\u5907\u540c\u65f6\u94fe\u63a5\u591a\u4e2a\u5c4f\u5e55\u65f6\uff0c\u6b63\u786e\u5730\u8bbe\u7f6e HUD \u7684\u4f4d\u7f6e"),(0,l.kt)("li",{parentName:"ul"},"\u5c01\u88c5\u5e76\u5411 flutter \u63d0\u4f9b\u80fd\u529b\uff0c\u5982\uff1a",(0,l.kt)("ul",{parentName:"li"},(0,l.kt)("li",{parentName:"ul"},"\u76d1\u542c\u9f20\u6807\u4f4d\u7f6e\u53d8\u5316\u548c\u952e\u76d8\u72b6\u6001\u53d8\u5316\uff0c\u5e76\u5c06\u72b6\u6001\u540c\u6b65\u81f3 flutter"),(0,l.kt)("li",{parentName:"ul"},"\u4e3a flutter \u63d0\u4f9b OCR \u548c\u8bed\u97f3\u5408\u6210\u80fd\u529b"),(0,l.kt)("li",{parentName:"ul"},"\u5728\u6267\u884c OCR \u65f6\u4f7f\u7528 swift \u5b9e\u73b0\u5fc5\u8981\u7684\u6027\u80fd\u4f18\u5316")))),(0,l.kt)("h2",{id:"\u6280\u672f\u7ec6\u8282"},"\u6280\u672f\u7ec6\u8282"),(0,l.kt)("p",null,"\u5f97\u76ca\u4e8e Apple \u4e00\u8109\u76f8\u627f\u7684 API \u8bbe\u8ba1\u4ee5\u53ca\u540c\u6837\u7684\u7f16\u7a0b\u8bed\u8a00\uff0c\u719f\u7ec3\u4e8e iOS \u5e94\u7528\u5f00\u53d1(Cocoa Touch & UIKit)\u7684\u7a0b\u5e8f\u5458\u5728\u9762\u5bf9\u4e0e macOS \u4ea4\u4e92\u7684 Cocoa \u548c AppKit \u6846\u67b6\u65f6\uff0c\u53ef\u4ee5\u590d\u7528\u5f88\u591a\u601d\u60f3\u4e0e\u903b\u8f91"),(0,l.kt)("p",null,"\u4f46\u76f8\u6bd4\u4e8e iOS\uff0cmacOS \u7ed9\u4e86\u7528\u6237\u66f4\u5927\u7684\u821e\u53f0\uff0c\u4e5f\u8ba9\u5f00\u53d1\u8005\u9762\u5bf9\u4e86\u66f4\u591a\u7684\u6311\u6218: \u9f20\u6807\uff0c\u952e\u76d8\uff0c\u7a97\u53e3\u548c\u5c4f\u5e55"),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u672c\u7a0b\u5e8f\u7684\u8fc7\u7a0b\u4e2d\uff0c\u6211\u9047\u5230\u4e86\u8bf8\u591a\u7684\u56f0\u96be\uff0c\u4e5f\u5728\u8305\u585e\u987f\u5f00\u65f6\u6536\u83b7\u4e86\u5f88\u591a\u5feb\u4e50\uff0c\u6211\u5728\u8fd9\u4e00\u7ae0\u8282\u4f1a\u8bb0\u5f55\u4e00\u4e0b"),(0,l.kt)("h3",{id:"\u591a\u5c4f\u5e55"},"\u591a\u5c4f\u5e55"),(0,l.kt)("p",null,"macOS \u53ca\u5176\u8fd0\u884c\u7684\u786c\u4ef6\u8bbe\u5907\u5e38\u5e38\u8fde\u63a5\u7740\u591a\u5757\u5c4f\u5e55\uff0c\u672c\u7a0b\u5e8f\u7684\u6838\u5fc3\u529f\u80fd\u5c31\u662f\u5728\u4efb\u610f\u7684\u5c4f\u5e55\u4f4d\u7f6e\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\uff0c\u8fd9\u5c31\u8981\u786e\u4fdd\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)\u653e\u7f6e\u4e8e\u6b63\u786e\u7684\u4f4d\u7f6e\u4e0a"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u8bf7\u6c42 native \u6355\u6349\u5c4f\u5e55\u65f6\uff0c\u6355\u6349\u7684\u77e9\u5f62\u6846\u7684\u4f4d\u7f6e\u6b63\u786e"),(0,l.kt)("li",{parentName:"ol"},"flutter \u5728\u89e3\u6790 ocr response \u540e\uff0c\u7ed3\u679c\u53ef\u4ee5\u6b63\u786e\u5730\u548c\u5c4f\u5e55\u4e0a\u771f\u6b63\u7684\u5185\u5bb9\u5efa\u7acb\u6620\u5c04")),(0,l.kt)("p",null,"\u548c UIKit \u4e2d\u5e38\u7528\u7684 CGRect \u4e0d\u540c\uff0cAppKit \u5bf9\u8bbe\u5907\u5c4f\u5e55\u7684\u62bd\u8c61 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen"),"\uff0c\u5176\u5750\u6807\u4ee5 NSRect \u8ba1\u7b97\uff0c\u5750\u6807\u7cfb\u539f\u70b9\u4e3a macOS \u539f\u59cb\u5c4f\u5e55\u7684\u5de6\u4e0b\u89d2\u3002\u800c\u6b64\u65f6\uff0c\u5982\u679c\u4f60\u7ed9\u4f60\u7684\u8bbe\u5907\u8fde\u63a5\u4e0a\u4e86\u5176\u4ed6\u7684\u5c4f\u5e55\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},"NSScreen.screens")," \u6240\u5448\u73b0\u5c4f\u5e55\u5e03\u5c40\uff0c\u53ef\u80fd\u5c31\u4f1a\u53d8\u6210\u4e0b\u56fe\u6240\u793a\u7684\u6837\u5b50\uff1a\n",(0,l.kt)("img",{alt:"appkit_screens_coordinate",src:n(1589).Z,width:"1630",height:"734"})),(0,l.kt)("p",null,(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u7684\u5de6\u4e0b\u89d2\u5750\u6807\u503c\u5e76\u975e\u662f (0, 0)\uff0c\u800c\u662f ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 1")," \u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"screen 0")," \u7684 (0, 0) \u70b9\u7684\u76f8\u5bf9\u4f4d\u7f6e (w",(0,l.kt)("sub",null,"1"),", d",(0,l.kt)("sub",null,"2"),")\u3002\u901a\u8fc7\u76d1\u542c\u9f20\u6807\u79fb\u52a8\u4e8b\u4ef6\u83b7\u53d6\u7684\u9f20\u6807\u4f4d\u7f6e ",(0,l.kt)("inlineCode",{parentName:"p"},"event.locationInWindow"),"\uff0c\u4e5f\u662f\u76f8\u5bf9\u4e8e\u539f\u70b9\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSPoint")),(0,l.kt)("p",null,"\u5728\u5f00\u53d1\u65f6\uff0c\u4ec5\u4ec5\u5c06\u81ea\u5df1\u7684\u601d\u7ef4\u4ece iOS \u7684 CGRect \u5750\u6807\u7cfb\u8f6c\u5316\u81f3 NSRect \u5750\u6807\u7cfb\u8fd8\u7b97\u7b80\u5355\u3002\u4f46\u5728\u540e\u7ee7\u7684\u903b\u8f91\u4e2d\uff0c\u56e0\u4e3a OCR \u7ed3\u679c\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRectangleObservation.boundingBox")," \u53c8\u4f1a\u56de\u5230 CGRect \u5750\u6807\u7cfb\uff0c\u5750\u6807\u7cfb\u7684\u9891\u7e41\u8f6c\u5316\u786e\u5b9e\u4f1a\u7ed9\u4eba\u5e26\u6765\u4e00\u5b9a\u7684\u56f0\u6270"),(0,l.kt)("h3",{id:"\u5c4f\u5e55\u6355\u6349"},"\u5c4f\u5e55\u6355\u6349"),(0,l.kt)("p",null,"\u60f3\u6355\u6349 macOS \u7684\u5c4f\u5e55\uff0c\u4f60\u53ef\u4ee5\u8c03\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u51fd\u6570"),(0,l.kt)("p",null,"\u4f46\u503c\u5f97\u6ce8\u610f\u7684\u662f\uff0c\u60f3\u8981\u622a\u53d6\u5c4f\u5e55\u4e0a\u5176\u4ed6\u8fdb\u7a0b\u7684\u5185\u5bb9(\u6bd4\u5982 IDE \u6216\u8005\u6d4f\u89c8\u5668)\uff0c\u9700\u8981\u9884\u5148\u901a\u8fc7 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGRequestScreenCaptureAccess"),"\uff0c\u7533\u8bf7\u5230\u622a\u53d6\u5176\u4ed6\u8fdb\u7a0b UI \u7684\u6743\u9650\uff0c\u5426\u5219 ",(0,l.kt)("inlineCode",{parentName:"p"},"CGDisplayCreateImage")," \u53ea\u80fd\u62ff\u5230 macOS \u7684\u684c\u9762\uff08\u4ee5\u53ca\u7a0b\u5e8f\u672c\u8eab\uff09"),(0,l.kt)("h3",{id:"ocr"},"OCR"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 Apple \u4e3a\u5f00\u53d1\u8005\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/vision/recognizing_text_in_images"},"Vision - Recognizing Text")," \u8fdb\u884c\u5c4f\u5e55\u6587\u672c\u7684\u63d0\u53d6"),(0,l.kt)("p",null,"OCR \u8bf7\u6c42\u7684\u8c03\u7528\u7531 flutter side \u5524\u8d77\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'void dispatchOCRRequestToNative() async {\n final x = 100;\n final y = 100;\n final width = 300;\n final height = 80;\n final dimensions = [x, y, width, height];\n final result = await methodChannel.invokeMethod("captureAndOCR", dimensions);\n\n // parse the result from native\n // ...\n}\n')),(0,l.kt)("p",null,"Native side \u5728\u63a5\u6536\u5230\u8bf7\u6c42\u540e\uff0c\u4f1a\u6267\u884c\u622a\u5c4f\u548c\u8bc6\u522b\u64cd\u4f5c\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"import Vision\n// ...\nlet result : FlutterResult = ...\nlet image = captureScreen()\nlet handler = VNImageRequestHandler(cgImage: image)\nlet request = VNRecognizeTextRequest { request, error in\n // ocr finished\n let parsedResult = parse(request.results)\n result(parsedResult)\n}\nhandler.perform([request])\n")),(0,l.kt)("h4",{id:"\u6027\u80fd\u8868\u73b0"},"\u6027\u80fd\u8868\u73b0"),(0,l.kt)("p",null,"\u5728 macbook 2021 \u7684 m1 pro \u4e0a\uff0c\u4ee5 320\u271575 \u7684\u8bbe\u8ba1\u5206\u8fa8\u7387\u622a\u56fe(\u5b9e\u9645\u5206\u8fa8\u7387\u4e3a 640\u2715150)\uff0c\u6bcf\u79d2 11 \u5e27\u7684\u60c5\u51b5\u4e0b\u8fdb\u884c\u957f\u65f6\u95f4\u7684\u622a\u56fe\u548c OCR \u64cd\u4f5c\uff0c\u6574\u4e2a\u6d41\u7a0b\u7684\u5ef6\u8fdf\u5e73\u5747\u7ea6\u4e3a 65ms\uff0c\u6211\u8ba4\u4e3a\u8fd8\u7b97\u662f\u4e00\u4e2a\u53ef\u63a5\u53d7\u7684\u72b6\u6001"),(0,l.kt)("h4",{id:"\u8282\u80fd\u4f18\u5316"},"\u8282\u80fd\u4f18\u5316"),(0,l.kt)("p",null,"\u5373\u4fbf\u662f\u6027\u80fd\u5141\u8bb8\uff0c\u4f18\u5316\u4e5f\u662f\u5e94\u8be5\u505a\u7684\uff0c\u7528\u6237\u4e00\u65e6\u4f7f\u7528\u4f60\u7684 App\uff0c\u5c31\u611f\u89c9 macbook \u7684 C \u9762\u53d1\u70ed\uff0c\u8fd9\u662f\u65e0\u6cd5\u5bb9\u5fcd\u7684"),(0,l.kt)("p",null,"\u5f53\u524d\u5728 OCR \u6d41\u7a0b\u4e2d\u4e3b\u8981\u7684\u6027\u80fd\u4f18\u5316\u6b65\u9aa4\u662f\u5728 native \u7ef4\u62a4\u4e00\u4e2a\u5148\u8fdb\u5148\u51fa\uff0c\u6700\u5927\u5bb9\u91cf\u4e3a 40 \u5e27\u7684\u5b57\u5178\uff0c\u4ee5\u56fe\u7247\u6570\u636e\u4e3a key \u7f13\u5b58 OCR \u7684\u7ed3\u679c\uff0c\u4e0b\u9762\u662f\u7b80\u5316\u540e\u7684\u4ee3\u7801\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},"let cacheManager = CacheManager()\n// ...\nlet image = captureScreen()\nlet key = image.dataProvider.data\nif let cachedResult = cacheManager[key] {\n delegate.onOCRResult(cachedResult)\n return\n}\n// ...\ndispatchOCRRequestToDeviceGPU(image) { result\n cacheManager[key] = result\n // ... other logic\n}\n// ...\n")),(0,l.kt)("p",null,"\u5f53\u9f20\u6807\u6307\u9488\u4f4d\u7f6e\u4e0d\u53d8\uff0c\u622a\u56fe\u83b7\u53d6\u7684\u56fe\u7247\u4e0d\u53d8\u65f6\uff0cnative \u5728\u5904\u7406 OCR \u8bf7\u6c42\u65f6\u4f1a\u5148\u547d\u4e2d\u7f13\u5b58\uff0c\u5e76\u76f4\u63a5\u8fd4\u56de\u7ed3\u679c\uff0c\u4ee5\u51cf\u5c11\u975e\u5fc5\u9700\u7684 GPU \u8c03\u7528"),(0,l.kt)("p",null,"\u5728\u7cfb\u7edf\u81ea\u5e26\u7684\u6d3b\u52a8\u76d1\u89c6\u5668\u4e2d\u67e5\u770b\u8fdb\u7a0b\u3002\u53d1\u73b0\uff0c\u5728\u5e94\u7528\u7684\u7f13\u5b58\u7b56\u7565\u540e\uff0c\u5728\u672c\u7a0b\u5e8f\u6d3b\u52a8\u65f6\uff0c\u5176 CPU/GPU \u5360\u7528\u7387\u786e\u5b9e\u964d\u4f4e\u4e86\u5f88\u591a\uff0c\u540c\u65f6\uff0cOCR \u7684\u7f13\u5b58\u7ed3\u679c\u88ab\u964d\u4e3a\u4e86 5ms"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u611f\u89c9\uff0c\u4ee5 ",(0,l.kt)("inlineCode",{parentName:"p"},"CFData")," \u4f5c\u4e3a key \u67e5\u8be2\u5b57\u5178\u8fd8\u4e0d\u662f\u6548\u7387\u6700\u9ad8\u7684\u7b97\u6cd5\uff0c\u5e94\u8be5\u53ef\u4ee5\u7ee7\u7eed\u63a2\u7d22\u4e00\u4e0b")),(0,l.kt)("h4",{id:"\u4e0d\u622a\u53d6\u81ea\u5df1"},"\u4e0d\u622a\u53d6\u81ea\u5df1"),(0,l.kt)("p",null,"\u5728\u6211\u8fdb\u884c\u5f00\u53d1\u65f6\uff0c\u53d1\u73b0\u5f53\u672c\u7a0b\u5e8f\u5728\u5c55\u793a\u5355\u8bcd\u91ca\u4e49 UI (HUD)\u65f6\uff0c\u56e0\u4e3a HUD \u672c\u8eab\u4e5f\u4f1a\u5728\u4e00\u5b9a\u8303\u56f4\u5185\u88ab\u622a\u5c4f\u51fd\u6570\u6355\u83b7\uff0c\u5bfc\u81f4 HUD \u4f1a\u5f71\u54cd OCR \u7684\u7ed3\u679c\uff0c\u800c\u5728 OCR \u7ed3\u679c\u53d8\u52a8\u540e\uff0cHUD \u53c8\u4f1a\u968f\u4e4b\u53d1\u751f\u53d8\u52a8\uff0c\u518d\u6b21\u5f71\u54cd OCR \u7ed3\u679c\uff0c\u5c31\u8fd9\u6837\u5faa\u73af\u5f80\u590d\uff0c\u8fde\u7f13\u5b58\u4e5f\u90fd\u5931\u6548\u4e86\u3002\u5728\u67e5\u9605\u548c\u5c1d\u8bd5\u5927\u91cf\u7684 API \u540e\uff0c\u6211\u7ec8\u4e8e\u5728 ",(0,l.kt)("inlineCode",{parentName:"p"},"NSWindow")," \u4e2d\u627e\u5230\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},"sharingType")," \u8fd9\u4e2a\u5c5e\u6027\uff0c\u5c4f\u5e55\u6355\u6349\u65b9\u6cd5\u6355\u83b7 HUD \u81ea\u8eab\uff0c\u603b\u7b97\u662f\u6253\u7834\u4e86\u8fd9\u4e2a\u94fe\u6761"),(0,l.kt)("h3",{id:"\u7ed3\u679c\u89e3\u6790\u5206\u8bcd\u4e0e\u5b9a\u4f4d"},"\u7ed3\u679c\u89e3\u6790\uff0c\u5206\u8bcd\u4e0e\u5b9a\u4f4d"),(0,l.kt)("p",null,"\u5bf9 Vision - VNRecognizeTextRequest \u7684\u7ed3\u679c\u89e3\u6790\u662f\u4e2a\u590d\u6742\u7684\u4efb\u52a1"),(0,l.kt)("p",null,"\u6267\u884c\u8fd9\u4e2a\u590d\u6742\u4efb\u52a1\u7684\u539f\u56e0\u6709\u4e09\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u8bbe\u8ba1\u7684\u4ea4\u4e92\u662f\u201c\u7528\u6237\u5c06\u9f20\u6807\u79fb\u52a8\u81f3\u611f\u5174\u8da3\u7684\u5355\u8bcd\u4e0a\u65f6\u5c55\u793a\u5355\u8bcd\u91ca\u4e49\u201d\uff0c\u8fd9\u5c31\u8981\u6c42\u6211\u4eec\u77e5\u9053\u5c4f\u5e55\u4e0a\u6bcf\u4e2a\u5355\u8bcd\u7684\u5177\u4f53\u4f4d\u7f6e\u548c\u5185\u5bb9"),(0,l.kt)("li",{parentName:"ol"},"\u672c\u7a0b\u5e8f\u9488\u5bf9\u6e90\u4ee3\u7801\u7279\u5316\uff0c\u5728\u9762\u5bf9",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E9%A9%BC%E5%B3%B0%E5%BC%8F%E5%A4%A7%E5%B0%8F%E5%86%99"},"\u9a7c\u5cf0\u547d\u540d\u6cd5"),"\u65f6\uff0c\u9700\u8981\u77e5\u9053\u6784\u6210\u4e00\u4e2a symbol \u7684\u6bcf\u4e2a\u5355\u8bcd\u7684\u610f\u601d\uff1a",(0,l.kt)("inlineCode",{parentName:"li"},"ThisIsAVeryVeryLongClassName")," -> ",(0,l.kt)("inlineCode",{parentName:"li"},"This"),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"Is")),",",(0,l.kt)("del",{parentName:"li"},(0,l.kt)("inlineCode",{parentName:"del"},"A")),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Very"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Long"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Class"),",",(0,l.kt)("inlineCode",{parentName:"li"},"Name")," (\u56e0\u8fc7\u4e8e\u7b80\u5355\uff0c\u4e22\u5f03\u957f\u5ea6\u5c0f\u4e8e 3 \u7684\u82f1\u8bed\u5355\u8bcd)"),(0,l.kt)("li",{parentName:"ol"},"\u6211\u4eec\u8981\u4ee5\u5355\u8bcd\u7684\u6587\u672c\u5185\u5bb9\u4e3a key\uff0c\u67e5\u8be2\u6570\u636e\u5e93\u3002\u5728\u6587\u672c\u5e8f\u5217\u5305\u542b\u7279\u6b8a\u5b57\u7b26\u65f6\uff0c\u6211\u4eec\u662f\u65e0\u6cd5\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u5230\u5355\u8bcd\u91ca\u4e49\u7684\u3002\u6240\u4ee5\u8981\u79fb\u9664\u975e\u5b57\u6bcd\u5b57\u7b26\u3002\u540c\u65f6\uff0c\u8fd9\u4e00\u64cd\u4f5c\u4e5f\u53ef\u4ee5\u81ea\u7136\u800c\u7136\u5730\u9002\u914d\u7f16\u7a0b\u8bed\u8a00\u4e2d\u7684\u5176\u4ed6\u547d\u540d\u65b9\u5f0f\uff0c\u6bd4\u5982\u79fb\u9664\u4e0b\u5212\u7ebf\u5c31\u53ef\u4ee5\u9002\u5e94",(0,l.kt)("a",{parentName:"li",href:"https://zh.wikipedia.org/zh-cn/%E8%9B%87%E5%BD%A2%E5%91%BD%E5%90%8D%E6%B3%95"},"\u86c7\u5f62\u547d\u540d\u6cd5"))),(0,l.kt)("p",null,"\u5047\u8bbe\u6211\u4eec\u5728\u5bf9\u4e0b\u9762\u7684\u56fe\u7247\u6267\u884c OCR \u8bf7\u6c42"),(0,l.kt)("p",null,(0,l.kt)("img",{alt:"ocr_text_source",src:n(3591).Z,width:"1038",height:"198"})),(0,l.kt)("p",null,"\u5728 Native \u7aef\u53d6 ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizeTextRequest.results.topCandidates(1)")," \u540e\uff0c\u7ed3\u679c\u5982\u4e0b"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision provides its text-recognition capabilities through VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request, an image-based request type that finds and extracts text in images. The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following example shows how to use VNImageRequestHandler to perform a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest for recognizing text in the specified CGImage.",\n]\n')),(0,l.kt)("p",null,"\u5728 Native \u7aef\u4ee5",(0,l.kt)("strong",{parentName:"p"},"\u7a7a\u683c"),"\u5b57\u7b26\u548c ",(0,l.kt)("inlineCode",{parentName:"p"},"VNRecognizedText.boundingBox(for:)")," \u5bf9 OCR \u7ed3\u679c\u8fdb\u884c\u7b2c\u4e00\u6b21\u6574\u7406\uff0c\u4e0a\u8ff0\u7684\u7ed3\u679c\u4f1a\u53d8\u4e3a\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text-recognition", "capabilities", "through", "VNRecognizeText",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request,", "an", "image-based", "request", "type", "that", "finds", "and", "extracts", "text", "in", "images.", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "to", "use", "VNImageRequestHandler", "to", "perform", "a",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "VNRecognizeTextRequest", "for", "recognizing", "text", "in", "the", "specified", "CGImage.",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\u5c06 boundingBox \u7684\u8fd4\u56de\u503c\u548c\u622a\u5c4f\u5c3a\u5bf8\u8fdb\u884c\u4e58\u7b97\uff0c\u8fd4\u56de\u7ed9 flutter \u7aef\u3002Flutter \u7aef\u5728\u62ff\u5230\u7ed3\u679c\u540e\u4e3b\u8981\u8fdb\u884c\u4e24\u4e2a\u6b65\u9aa4\uff1a"),(0,l.kt)("ol",null,(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u79fb\u9664\u975e\u82f1\u6587\u5b57\u7b26\uff0c\u8ba1\u7b97\u65b0\u7684 rect"),(0,l.kt)("li",{parentName:"ol"},"\u5c06\u6587\u672c\u89c6\u4e3a\u7b49\u5bbd\u5b57\u4f53\uff0c\u5bf9\u5305\u542b\u5927\u5199\u5b57\u6bcd\u7684\u5b57\u7b26\u4e32\u8fdb\u884c\u5206\u5272\uff0c\u8ba1\u7b97\u65b0\u7684 rect")),(0,l.kt)("p",null,"\u8fd9\u6837\uff0c\u6587\u672c\u5757\u7684\u5185\u5bb9\u5728 flutter \u7aef\u5c31\u4f1a\u53d8\u6210\u8fd9\u6837\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'[\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Vision", "provides", "its", "text", "recognition", "capabilities", "through", "Recognize", "Text",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Request","image", "based", "request", "type", "that", "finds", "and", "extracts", "text", "images", "The",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "following", "example", "shows", "how", "use", "Image", "Request", "Handler", "perform",\n // \u540c\u65f6\u5305\u542b rect \u4fe1\u606f\n "Recognize", "Text", "Request", "for", "recognizing", "text", "the", "specified", "Image",\n]\n')),(0,l.kt)("p",null,"\u6700\u540e\uff0cflutter \u4fa7\u4f1a\u4ee5\u4e0b\u9762\u7684\u4ee3\u7801\u5bf9\u8fd9\u4e9b OCR \u4fe1\u606f\u8fdb\u884c\u62bd\u8c61\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},"/// The representation of block\n///\n/// \u6587\u672c\u5757\u8868\u5f81\nclass Block {\n String text;\n double x;\n double y;\n double w;\n double h;\n}\n\n// ...\n\n/// All text representations in memory\n///\n/// \u6240\u6709\u6587\u672c\u5757\u8868\u5f81\nfinal blocks = StateProvider>((_) => []);\n")),(0,l.kt)("p",null,"\u4e0a\u9762\u7684\u6570\u636e\u5728\u5c4f\u5e55\u4e0a\u7684\u53ef\u89c6\u5316\u6548\u679c\u5982\u4e0b\uff1a\n",(0,l.kt)("img",{alt:"text_block_representation_2",src:n(6607).Z,width:"1050",height:"184"})),(0,l.kt)("p",null,"\u5176\u4e2d\u5916\u56f4\u7eff\u6846\u4ee3\u8868\u622a\u5c4f\u8303\u56f4\uff0c\u5185\u90e8\u7684\u5c0f\u7eff\u6846\u4ee3\u8868 native \u7aef\u8fd4\u56de\u7684\u7ed3\u679c\uff0c\u5c0f\u7eff\u6846\u5185\u90e8\u7684\u84dd\u8272\u6846\u4ee3\u8868 flutter side \u7b2c\u4e8c\u6b21\u89e3\u6790\u7ed3\u679c\u3002\u53ef\u4ee5\u770b\u5230\uff0c",(0,l.kt)("inlineCode",{parentName:"p"},'"VNImageRequestHandler"')," \u8fd9\u4e2a\u5b57\u7b26\u4e32\u88ab\u5206\u5272\u4e3a\u4e86 ",(0,l.kt)("inlineCode",{parentName:"p"},'"Image"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Request"'),", ",(0,l.kt)("inlineCode",{parentName:"p"},'"Handler"')," \u8fd9\u4e09\u4e2a\u6587\u672c\u5757\u3002\u8fd9\u6837\uff0c\u6211\u5c31\u53ef\u4ee5\u77e5\u9053\u7528\u6237\u5230\u5e95\u5bf9\u54ea\u4e00\u6bb5\u5b57\u6bcd\u5e8f\u5217\uff08\u5355\u8bcd\uff09\u611f\u5174\u8da3\u4e86\uff08\u5047\u8bbe\u6e90\u4ee3\u7801\u7b26\u53f7\u7684\u547d\u540d\u662f\u89c4\u8303\u7684 \ud83d\ude07\uff09\u3002"),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u4f60\u4e5f\u53ef\u4ee5\u6253\u5f00\u672c\u7a0b\u5e8f\u7684\u7528\u6237\u8bbe\u7f6e\u9762\u677f\u6765\u76f4\u63a5\u89c2\u5bdf\u5176\u8fd0\u884c\u65f6\u8868\u73b0\uff1a\n",(0,l.kt)("img",{alt:"dashboard_inspect",src:n(6521).Z,width:"970",height:"192"}))),(0,l.kt)("h3",{id:"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"},"\u591a\u5f15\u64ce\u4e0e\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,'\u672c\u7a0b\u5e8f\u5728\u8fd0\u884c\u65f6\u4f1a\u521b\u5efa\u4e24\u4e2a flutter engine\uff0c\u5206\u522b\u7528\u4e8e\u6e32\u67d3"\u91ca\u4e49\u5c55\u793a\u9762\u677f(HUD)"\u548c"\u6b22\u8fce\u4e0e\u8bbe\u7f6e\u9875\u9762(Dashboard)"'),(0,l.kt)("p",null,"\u8fd9\u4e24\u4e2a\u5f15\u64ce\u5747\u901a\u8fc7\u57fa\u672c\u65b9\u5f0f\u521b\u5efa\uff0c\u5747\u91c7\u7528 ",(0,l.kt)("inlineCode",{parentName:"p"},"FlutterMethodChannel")," \u4e0e native \u8fdb\u884c\u901a\u8baf"),(0,l.kt)("p",null,"\u53cc\u5f15\u64ce\u5171\u4eab\u540c\u4e00\u4efd\u4ee3\u7801\uff0c\u7ecf\u7531\u4e0d\u540c\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/dart-lang/sdk/blob/master/runtime/docs/compiler/aot/entry_point_pragma.md"},"dart \u5165\u53e3\u51fd\u6570"),"\u542f\u52a8\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-swift"},'// \u8bbe\u7f6e\u9875\u9762 (Dashboard)\ndashboardEngine = FlutterEngine(name: "dashboard", project: nil)\ndashboardChannel = FlutterMethodChannel(name: "dashboard", binaryMessenger: dashboardEngine.binaryMessenger)\ndashboardEngine.run(withEntrypoint: "_dashboard")\n//...\n\n// \u91ca\u4e49\u5c55\u793a\u9762\u677f (HUD)\nhudEngine = FlutterEngine(name: "hud", project: nil)\nhudChannel = FlutterMethodChannel(name: "hud", binaryMessenger: hudEngine.binaryMessenger)\nhudChannel.run(withEntrypoint: "_hud")\n')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u6211\u4e2a\u4eba\u6ca1\u6709\u4f7f\u7528\u5b98\u65b9\u63d0\u4f9b\u7684 ",(0,l.kt)("a",{parentName:"p",href:"https://docs.flutter.dev/add-to-app/multiple-flutters"},"FlutterEngineGroup \u65b9\u6848"),"\uff0c\u56e0\u4e3a\u5728\u5f53\u521d\u8fdb\u884c\u5c1d\u8bd5\u65f6\u53d1\u73b0\u4e86\u5f15\u64ce\u65e0\u54cd\u5e94\u7684\u95ee\u9898\uff0c\u81f3\u4eca\u4ecd\u7136\u662f P2 \u7ea7\u522b\u7684 open ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/flutter/flutter/issues/119403"},"issue"))),(0,l.kt)("h4",{id:"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"},"\u8de8\u5f15\u64ce\u72b6\u6001\u540c\u6b65"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://riverpod.dev/"},"riverpod")," \u6765\u7ba1\u7406\u7edd\u5927\u90e8\u5206\u72b6\u6001\uff0c\u6240\u4ee5\u5728\u540c\u6b65\u72b6\u6001\u65f6\uff0c\u6211\u4e5f\u671f\u671b\u81ea\u5df1\u7684\u81ea\u5df1\u7684\u5fc3\u667a\u6a21\u578b\u53ef\u4ee5\u66f4\u8d34\u8fd1 riverpod"),(0,l.kt)("p",null,"\u5728\u8fd0\u884c\u65f6\uff0cDashboard engine \u9700\u8981\u5c06\u7528\u6237\u8bbe\u7f6e\u540c\u6b65\u81f3 HUD engine (\u6bd4\u5982\u66f4\u6539\u67e5\u8bcd\u5feb\u6377\u952e)\uff0c\u6211\u5728\u8fd9\u91cc\u76d1\u542c\u4e86 Dashboard engine \u5bf9\u5e94\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider")," \u7684\u53d8\u66f4\uff0c\u5e76\u901a\u8fc7 method channel\uff0c\u7ecf\u7531 native \u8f6c\u53d1\u81f3 HUD engine\uff0cHUD engine \u5728\u6536\u5230\u4e86\u901a\u77e5\u540e\uff0c\u518d\u66f4\u65b0\u81ea\u5df1\u7ef4\u62a4\u7684 ",(0,l.kt)("inlineCode",{parentName:"p"},"StateProvider"),"\u3002\u4ece\u800c\u5b9e\u73b0\u4e86\u8de8\u5f15\u64ce\u7684\u72b6\u6001\u540c\u6b65"),(0,l.kt)("h3",{id:"\u91ca\u4e49\u67e5\u8be2"},"\u91ca\u4e49\u67e5\u8be2"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u5728\u67e5\u8be2\u5355\u8bcd\u91ca\u4e49\u65f6\u4f7f\u7528\u672c\u5730 sqlite \u6570\u636e\u5e93\uff0c\u6570\u636e\u5e93\u6e90\u4e8e ",(0,l.kt)("a",{parentName:"p",href:"https://github.com/skywind3000/ECDICT-ultimate"},"ecdict-ultimate")),(0,l.kt)("p",null,"\u9274\u4e8e\u5f00\u6e90\u6570\u636e\u5e93\u8fc7\u4e8e\u5de8\u5927\uff0c\u4e14\u5b58\u5728\u4e00\u4e9b\u672c\u7a0b\u5e8f\u65e0\u9700\u4f7f\u7528\u7684 columns\uff0c\u6211\u53c8\u4f7f\u7528 dart \u548c ",(0,l.kt)("a",{parentName:"p",href:"https://pub.dev/packages/drift"},"drift")," \u5bf9\u5176\u8fdb\u884c\u4e86\u6e05\u6d17\u548c\u526a\u88c1"),(0,l.kt)("p",null,"\u5728\u67e5\u8be2\u65f6\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5c06\u67e5\u8be2\u7684\u672c\u6587\u4e32\u201c\u6253\u6563\u201d\uff0c\u5e76\u4e00\u9f50\u8bf7\u6c42\u6570\u636e\u5e93\uff0c\u4ee5\u671f\u80fd\u547d\u4e2d\u4e0d\u65ad\u5343\u53d8\u4e07\u5316\u7684\u82f1\u8bed\u5355\u8bcd\uff1a"),(0,l.kt)("pre",null,(0,l.kt)("code",{parentName:"pre",className:"language-dart"},'final sequence = "qwertyuiopasdfghjkl";\nfinal sequences = ["qwe","wer","ert",...,"qwer","wert",...,"qwert",...,"qwertyuiopasdfghjkl"];\nfinal List results = await queryDB(keys:sequences); // average latency: ~6ms\n')),(0,l.kt)("p",null,"\u5728\u8bf7\u6c42\u5b8c\u6210\u540e\uff0c\u672c\u7a0b\u5e8f\u4f1a\u5bf9\u8bf7\u6c42\u7ed3\u679c\u8fdb\u884c\u53bb\u91cd\uff0c\u903b\u8f91\u5982\u4e0b\uff1a"),(0,l.kt)("ul",null,(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b owl (\u732b\u5934\u9e70)"),(0,l.kt)("li",{parentName:"ul"},"\u7ed3\u679c\u4e2d\u5305\u542b knowledge(\u77e5\u8bc6)"),(0,l.kt)("li",{parentName:"ul"},'\u5e8f\u5217 "knowledge" \u5305\u542b "owl"\uff0c\u79fb\u9664\u67e5\u8be2\u7ed3\u679c\u4e2d\u7684 owl')),(0,l.kt)("admonition",{type:"note"},(0,l.kt)("p",{parentName:"admonition"},"\u76ee\u524d\uff0c\u8be5\u903b\u8f91\u8fd8\u6ca1\u6709\u5b9e\u73b0\u5bf9\u51b2\u7a81\u7684\u5224\u5b9a\uff1a\u5982\u5411\u6570\u636e\u5e93\u67e5\u8be2\u7684 key \u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'"gitignore"'),"\uff0c\u67e5\u8be2\u7ed3\u679c\u4e3a ",(0,l.kt)("inlineCode",{parentName:"p"},'["git", "tig", "ignore"]'),"\uff0c\u76ee\u524d\u8fd9\u4e09\u4e2a\u7ed3\u679c\u90fd\u4f1a\u88ab\u6e32\u67d3\u5230 HUD \u4e0a\u3002\u4f46\u663e\u7136\uff0c\u9009\u53d6 git \u548c ignore \u8fd9\u4e24\u4e2a\u67e5\u8be2\u7ed3\u679c\u9020\u6210\u7684 \u201c\u51b2\u7a81\u201d \u662f\u6700\u5c0f\u7684")),(0,l.kt)("h3",{id:"\u53d1\u97f3"},"\u53d1\u97f3"),(0,l.kt)("p",null,"\u672c\u7a0b\u5e8f\u4f7f\u7528 ",(0,l.kt)("a",{parentName:"p",href:"https://developer.apple.com/documentation/avfoundation/speech_synthesis"},"AVFoundation - Speech synthesis")," API \u6765\u5b9e\u65f6\u5408\u6210\u8bed\u97f3"),(0,l.kt)("p",null,"\u8be5 API \u53ef\u4ee5\u540c\u65f6\u5728 iOS/macOS \u4e0a\u4f7f\u7528\uff0c\u4e14\u53d1\u97f3\u8f83\u4e3a\u51c6\u786e\uff0c\u4ee5\u6211\u81ea\u8eab\u7684\u80fd\u529b(\u5168\u56fd\u5927\u5b66\u82f1\u8bed\u56db\u7ea7)\u6765\u770b\uff0c\u6548\u679c\u8fd8\u7b97\u6ee1\u610f\uff0c\u786e\u5b9e\u6bd4\u6211\u7684\u53d1\u97f3\u51c6 \ud83e\udd23"),(0,l.kt)("p",null,"\u5f53\u7136\uff0c\u540e\u7ee7\u5982\u679c\u6709\u66f4\u9ad8\u7684\u9700\u6c42\uff0c\u4e5f\u6709\u5f88\u591a\u5176\u4ed6\u53ef\u9009\u65b9\u6848"))}c.isMDXComponent=!0},3591:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/ocr_text_source-9af654c22d465590f3079fd8d4f2f24f.png"},1589:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/appkit_screens_coordinate-94563877ba8462b8412c6e945e1f7b5e.png"},6521:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/dashboard_inspect-0a3c0a7bba8e62a06a697e1555bb3ea4.png"},2731:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/intro-663e889360c7a0e605df52af33af3b11.gif"},8555:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_dark-ede87b48b4273a15fc0974c77b5586cf.png"},1974:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/modules_light-980bb1be0d553350c1dcf22e0b4325e2.png"},3642:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/project_layout-b9e52932db96541ab24f4ed06bc28bc0.png"},9462:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation-e7dc02491d5b436f2ef66bd928e65e17.png"},6607:(e,t,n)=>{n.d(t,{Z:()=>a});const a=n.p+"assets/images/text_block_representation_1-7c6db883d1b12b61355048e553101f21.png"}}]); \ No newline at end of file diff --git a/assets/js/runtime~main.519362e7.js b/assets/js/runtime~main.868bc311.js similarity index 54% rename from assets/js/runtime~main.519362e7.js rename to assets/js/runtime~main.868bc311.js index c4af836..37af0a4 100644 --- a/assets/js/runtime~main.519362e7.js +++ b/assets/js/runtime~main.868bc311.js @@ -1 +1 @@ -(()=>{"use strict";var e,r,t,c,a,f={},o={};function d(e){var r=o[e];if(void 0!==r)return r.exports;var t=o[e]={exports:{}};return f[e].call(t.exports,t,t.exports,d),t.exports}d.m=f,e=[],d.O=(r,t,c,a)=>{if(!t){var f=1/0;for(i=0;i=a)&&Object.keys(d.O).every((e=>d.O[e](t[n])))?t.splice(n--,1):(o=!1,a0&&e[i-1][2]>a;i--)e[i]=e[i-1];e[i]=[t,c,a]},d.n=e=>{var r=e&&e.__esModule?()=>e.default:()=>e;return d.d(r,{a:r}),r},t=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,d.t=function(e,c){if(1&c&&(e=this(e)),8&c)return e;if("object"==typeof e&&e){if(4&c&&e.__esModule)return e;if(16&c&&"function"==typeof e.then)return e}var a=Object.create(null);d.r(a);var f={};r=r||[null,t({}),t([]),t(t)];for(var o=2&c&&e;"object"==typeof o&&!~r.indexOf(o);o=t(o))Object.getOwnPropertyNames(o).forEach((r=>f[r]=()=>e[r]));return f.default=()=>e,d.d(a,f),a},d.d=(e,r)=>{for(var t in r)d.o(r,t)&&!d.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:r[t]})},d.f={},d.e=e=>Promise.all(Object.keys(d.f).reduce(((r,t)=>(d.f[t](e,r),r)),[])),d.u=e=>"assets/js/"+({13:"01a85c17",53:"935f2afb",70:"d1a7dadb",85:"1f391b9e",89:"a6aa9e1f",103:"ccc49370",122:"8ca7c345",217:"3b8c55ea",227:"0786957f",231:"880564fa",233:"b220c876",237:"1df93b7f",268:"970051c0",329:"2985228a",378:"b1446005",411:"2ca36fa6",414:"393be207",498:"a8742701",514:"1be78505",535:"814f3328",588:"dec09136",594:"f812cb1f",596:"fd9e7df5",608:"9e4087bc",610:"6875c492",649:"ebb011f4",671:"0e384e19",728:"23fe4fcc",742:"c3e6124d",827:"6476eba6",860:"66f08f32",894:"fc65744c",918:"17896441",932:"a9348d45"}[e]||e)+"."+{13:"27291cca",53:"a643a0e2",70:"2442bbdc",85:"af91296a",89:"894abcc0",98:"2883057a",103:"a54cf59a",122:"d22d37f3",217:"b061c624",227:"ae3e5a36",231:"12d08406",233:"2e5a1c24",237:"81fcd143",268:"14bdc0e0",316:"d5ceb690",329:"0587c634",378:"ce85c795",411:"53a9e222",414:"741a0402",487:"64d75fbe",498:"92cc4dc3",514:"3e559da3",529:"e38ca5d8",535:"52b5378a",588:"7948a095",594:"26bfec5d",596:"f3ff37a5",608:"75534213",610:"025c9d50",649:"c91967e2",671:"573956f4",724:"12c1bb45",728:"5a18a3a9",742:"1469d9ed",827:"206326c2",860:"905eeb5d",894:"30751d90",918:"b6157b0c",932:"b9190c2b",972:"4a3951f6"}[e]+".js",d.miniCssF=e=>{},d.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),d.o=(e,r)=>Object.prototype.hasOwnProperty.call(e,r),c={},a="df-doc-source:",d.l=(e,r,t,f)=>{if(c[e])c[e].push(r);else{var o,n;if(void 0!==t)for(var b=document.getElementsByTagName("script"),i=0;i{o.onerror=o.onload=null,clearTimeout(s);var a=c[e];if(delete c[e],o.parentNode&&o.parentNode.removeChild(o),a&&a.forEach((e=>e(t))),r)return r(t)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:o}),12e4);o.onerror=l.bind(null,o.onerror),o.onload=l.bind(null,o.onload),n&&document.head.appendChild(o)}},d.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},d.p="/valo-reader-doc/",d.gca=function(e){return e={17896441:"918","01a85c17":"13","935f2afb":"53",d1a7dadb:"70","1f391b9e":"85",a6aa9e1f:"89",ccc49370:"103","8ca7c345":"122","3b8c55ea":"217","0786957f":"227","880564fa":"231",b220c876:"233","1df93b7f":"237","970051c0":"268","2985228a":"329",b1446005:"378","2ca36fa6":"411","393be207":"414",a8742701:"498","1be78505":"514","814f3328":"535",dec09136:"588",f812cb1f:"594",fd9e7df5:"596","9e4087bc":"608","6875c492":"610",ebb011f4:"649","0e384e19":"671","23fe4fcc":"728",c3e6124d:"742","6476eba6":"827","66f08f32":"860",fc65744c:"894",a9348d45:"932"}[e]||e,d.p+d.u(e)},(()=>{var e={303:0,532:0};d.f.j=(r,t)=>{var c=d.o(e,r)?e[r]:void 0;if(0!==c)if(c)t.push(c[2]);else if(/^(303|532)$/.test(r))e[r]=0;else{var a=new Promise(((t,a)=>c=e[r]=[t,a]));t.push(c[2]=a);var f=d.p+d.u(r),o=new Error;d.l(f,(t=>{if(d.o(e,r)&&(0!==(c=e[r])&&(e[r]=void 0),c)){var a=t&&("load"===t.type?"missing":t.type),f=t&&t.target&&t.target.src;o.message="Loading chunk "+r+" failed.\n("+a+": "+f+")",o.name="ChunkLoadError",o.type=a,o.request=f,c[1](o)}}),"chunk-"+r,r)}},d.O.j=r=>0===e[r];var r=(r,t)=>{var c,a,f=t[0],o=t[1],n=t[2],b=0;if(f.some((r=>0!==e[r]))){for(c in o)d.o(o,c)&&(d.m[c]=o[c]);if(n)var i=n(d)}for(r&&r(t);b{"use strict";var e,r,t,c,a,f={},o={};function n(e){var r=o[e];if(void 0!==r)return r.exports;var t=o[e]={exports:{}};return f[e].call(t.exports,t,t.exports,n),t.exports}n.m=f,e=[],n.O=(r,t,c,a)=>{if(!t){var f=1/0;for(i=0;i=a)&&Object.keys(n.O).every((e=>n.O[e](t[d])))?t.splice(d--,1):(o=!1,a0&&e[i-1][2]>a;i--)e[i]=e[i-1];e[i]=[t,c,a]},n.n=e=>{var r=e&&e.__esModule?()=>e.default:()=>e;return n.d(r,{a:r}),r},t=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,n.t=function(e,c){if(1&c&&(e=this(e)),8&c)return e;if("object"==typeof e&&e){if(4&c&&e.__esModule)return e;if(16&c&&"function"==typeof e.then)return e}var a=Object.create(null);n.r(a);var f={};r=r||[null,t({}),t([]),t(t)];for(var o=2&c&&e;"object"==typeof o&&!~r.indexOf(o);o=t(o))Object.getOwnPropertyNames(o).forEach((r=>f[r]=()=>e[r]));return f.default=()=>e,n.d(a,f),a},n.d=(e,r)=>{for(var t in r)n.o(r,t)&&!n.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:r[t]})},n.f={},n.e=e=>Promise.all(Object.keys(n.f).reduce(((r,t)=>(n.f[t](e,r),r)),[])),n.u=e=>"assets/js/"+({13:"01a85c17",53:"935f2afb",70:"d1a7dadb",85:"1f391b9e",89:"a6aa9e1f",103:"ccc49370",122:"8ca7c345",217:"3b8c55ea",227:"0786957f",231:"880564fa",233:"b220c876",237:"1df93b7f",268:"970051c0",329:"2985228a",378:"b1446005",411:"2ca36fa6",414:"393be207",498:"a8742701",514:"1be78505",535:"814f3328",588:"dec09136",594:"f812cb1f",596:"fd9e7df5",608:"9e4087bc",610:"6875c492",649:"ebb011f4",671:"0e384e19",728:"23fe4fcc",742:"c3e6124d",827:"6476eba6",860:"66f08f32",894:"fc65744c",918:"17896441",932:"a9348d45"}[e]||e)+"."+{13:"27291cca",53:"a643a0e2",70:"2442bbdc",85:"af91296a",89:"894abcc0",98:"2883057a",103:"a54cf59a",122:"d22d37f3",217:"b061c624",227:"ae3e5a36",231:"12d08406",233:"2e5a1c24",237:"81fcd143",268:"14bdc0e0",316:"d5ceb690",329:"0587c634",378:"ce85c795",411:"53a9e222",414:"741a0402",487:"64d75fbe",498:"92cc4dc3",514:"3e559da3",529:"e38ca5d8",535:"52b5378a",588:"7948a095",594:"6b84295d",596:"f3ff37a5",608:"75534213",610:"025c9d50",649:"6cf2d7e7",671:"573956f4",724:"12c1bb45",728:"907478e0",742:"1469d9ed",827:"206326c2",860:"905eeb5d",894:"30751d90",918:"b6157b0c",932:"b9190c2b",972:"4a3951f6"}[e]+".js",n.miniCssF=e=>{},n.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),n.o=(e,r)=>Object.prototype.hasOwnProperty.call(e,r),c={},a="df-doc-source:",n.l=(e,r,t,f)=>{if(c[e])c[e].push(r);else{var o,d;if(void 0!==t)for(var b=document.getElementsByTagName("script"),i=0;i{o.onerror=o.onload=null,clearTimeout(s);var a=c[e];if(delete c[e],o.parentNode&&o.parentNode.removeChild(o),a&&a.forEach((e=>e(t))),r)return r(t)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:o}),12e4);o.onerror=l.bind(null,o.onerror),o.onload=l.bind(null,o.onload),d&&document.head.appendChild(o)}},n.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},n.p="/valo-reader-doc/",n.gca=function(e){return e={17896441:"918","01a85c17":"13","935f2afb":"53",d1a7dadb:"70","1f391b9e":"85",a6aa9e1f:"89",ccc49370:"103","8ca7c345":"122","3b8c55ea":"217","0786957f":"227","880564fa":"231",b220c876:"233","1df93b7f":"237","970051c0":"268","2985228a":"329",b1446005:"378","2ca36fa6":"411","393be207":"414",a8742701:"498","1be78505":"514","814f3328":"535",dec09136:"588",f812cb1f:"594",fd9e7df5:"596","9e4087bc":"608","6875c492":"610",ebb011f4:"649","0e384e19":"671","23fe4fcc":"728",c3e6124d:"742","6476eba6":"827","66f08f32":"860",fc65744c:"894",a9348d45:"932"}[e]||e,n.p+n.u(e)},(()=>{var e={303:0,532:0};n.f.j=(r,t)=>{var c=n.o(e,r)?e[r]:void 0;if(0!==c)if(c)t.push(c[2]);else if(/^(303|532)$/.test(r))e[r]=0;else{var a=new Promise(((t,a)=>c=e[r]=[t,a]));t.push(c[2]=a);var f=n.p+n.u(r),o=new Error;n.l(f,(t=>{if(n.o(e,r)&&(0!==(c=e[r])&&(e[r]=void 0),c)){var a=t&&("load"===t.type?"missing":t.type),f=t&&t.target&&t.target.src;o.message="Loading chunk "+r+" failed.\n("+a+": "+f+")",o.name="ChunkLoadError",o.type=a,o.request=f,c[1](o)}}),"chunk-"+r,r)}},n.O.j=r=>0===e[r];var r=(r,t)=>{var c,a,f=t[0],o=t[1],d=t[2],b=0;if(f.some((r=>0!==e[r]))){for(c in o)n.o(o,c)&&(n.m[c]=o[c]);if(d)var i=d(n)}for(r&&r(t);b 历史博文 | Valo Reader - + - + \ No newline at end of file diff --git a/blog/atom.xml b/blog/atom.xml index c38fbd2..d6f8ac5 100644 --- a/blog/atom.xml +++ b/blog/atom.xml @@ -15,7 +15,7 @@ 摘要

本文主要介绍了 Valo Reader for macOS(下文简称为“本程序”)的项目布局工程架构运行时功能模块技术细节。你还可以在本文档的其他页面了解本程序的开发动机使用方式隐私与安全等内容

什么是 Valo Reader?

Valo Reader 这个项目是我个人在日常生活中逐渐萌生,演化和实施的想法。是我为了在自己电子设备设备上能轻松阅读英文内容,并渐近式地提高自己的英文水平,而自行设计和研发的一款产品,其核心功能展示如下: intro

当你在使用本程序时,将鼠标移动至你感到生疏的英语单词上。按下指定的按键(默认为 Fn),本程序即会在响应的位置展示对应的释义,同时使用语音合成来发音。

目前,你可以通过 Mac App Store其他方式来获取本程序

项目布局

下图展示了本项目的主体结构: -project_layout

本文的位置则处于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: +project_layout

本文的位于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: modules_light modules_dark

释义展示面板 (HUD)

HUD 是一个的 NSWindow 实例,其内部承载了一个 FlutterViewController,并绑定了自己的 FlutterEngineFlutterMethodChannel

HUD 的主要任务包含:

  • 通过 FlutterMethodChannel 向 native 派发 ocr 请求
  • 解析 native 对屏幕指定区域的 ocr 结果,并在内存中维护该结果供后继程序逻辑使用
  • 监听鼠标位置以在需要时计算释义展示的位置
  • 监听键盘按键状态,在变化时派发 ocr 请求并展示单词释义
  • 实时同步 Dashboard engine 传递过来的用户设置,并修改展示逻辑

下图展示了该模块所维护的数据映射到屏幕上时的可视化效果: text_block_representation

在 HUD 中的状态

本程序使用 riverpod 管理绝大部分的状态。在构筑本程序时,我的思考过程主要基于状态与状态变化,从展示释义键点击到渲染释义面板动画

在大量使用 riverpod 中的 Provider / StateProvider / ProviderContainer.listen 后,我可以构建一个高度异步,调用顺序不敏感,状态变化驱动的应用程序。在编码过程中,这种方式让程序员可以从一大串的命令式(Imperative)方法调用中解放出来,仅仅需要关注和确保每个最小逻辑单元——Provider,确保其实现是正确的即可,这减少了构建复杂和长串逻辑出错的可能

下面的两张图表展示了本程序的主要功能流程

切换展示释义流程:

用户关注文本块变更流程:

用户设置面板 (Dashboard)

Dashboard 是一个 NSPanel 实例,其内部承载了一个 FlutterViewController,并绑定了不同于 HUD 的独立 engine 与 method channel

Dashboard 的主要任务是为用户提供控制本程序的 UI,包括快捷键设置,开机自启,关于本程序等常见用户交互

同时,Dashboard 会维护本程序在内存和硬盘中的状态,并将这个状态通过 method channel 同步至下面即将要讲到的释义展示面板

原生侧 (native)

原生侧是由 flutter create 命令创建出来的传统 Xcode 工程

在原生侧,我主要关注的问题如下:

  • 处理来自 FlutterMethodChannel 的调用,并在需要时通过 FlutterResult 响应对应的 flutter engine
  • 维护承载 HUD 和 Dashboard 的两个 NSWindow,尤其是当用户的设备同时链接多个屏幕时,正确地设置 HUD 的位置
  • 封装并向 flutter 提供能力,如:
    • 监听鼠标位置变化和键盘状态变化,并将状态同步至 flutter
    • 为 flutter 提供 OCR 和语音合成能力
    • 在执行 OCR 时使用 swift 实现必要的性能优化

技术细节

得益于 Apple 一脉相承的 API 设计以及同样的编程语言,熟练于 iOS 应用开发(Cocoa Touch & UIKit)的程序员在面对与 macOS 交互的 Cocoa 和 AppKit 框架时,可以复用很多思想与逻辑

但相比于 iOS,macOS 给了用户更大的舞台,也让开发者面对了更多的挑战: 鼠标,键盘,窗口和屏幕

在开发本程序的过程中,我遇到了诸多的困难,也在茅塞顿开时收获了很多快乐,我在这一章节会记录一下

多屏幕

macOS 及其运行的硬件设备常常连接着多块屏幕,本程序的核心功能就是在任意的屏幕位置展示单词释义,这就要确保:

  1. 将释义展示面板(HUD)放置于正确的位置上
  2. flutter 在请求 native 捕捉屏幕时,捕捉的矩形框的位置正确
  3. flutter 在解析 ocr response 后,结果可以正确地和屏幕上真正的内容建立映射

和 UIKit 中常用的 CGRect 不同,AppKit 对设备屏幕的抽象 NSScreen,其坐标以 NSRect 计算,坐标系原点为 macOS 原始屏幕的左下角。而此时,如果你给你的设备连接上了其他的屏幕,NSScreen.screens 所呈现屏幕布局,可能就会变成下图所示的样子: diff --git a/blog/index.html b/blog/index.html index d0fb0b8..e2c636c 100644 --- a/blog/index.html +++ b/blog/index.html @@ -5,21 +5,21 @@ Blog | Valo Reader - +

· 阅读需 24 分钟
Ce Wang

摘要

本文主要介绍了 Valo Reader for macOS(下文简称为“本程序”)的项目布局工程架构运行时功能模块技术细节。你还可以在本文档的其他页面了解本程序的开发动机使用方式隐私与安全等内容

什么是 Valo Reader?

Valo Reader 这个项目是我个人在日常生活中逐渐萌生,演化和实施的想法。是我为了在自己电子设备设备上能轻松阅读英文内容,并渐近式地提高自己的英文水平,而自行设计和研发的一款产品,其核心功能展示如下: intro

当你在使用本程序时,将鼠标移动至你感到生疏的英语单词上。按下指定的按键(默认为 Fn),本程序即会在响应的位置展示对应的释义,同时使用语音合成来发音。

目前,你可以通过 Mac App Store其他方式来获取本程序

项目布局

下图展示了本项目的主体结构: -project_layout

本文的位置则处于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: +project_layout

本文的位于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: modules_light modules_dark

释义展示面板 (HUD)

HUD 是一个的 NSWindow 实例,其内部承载了一个 FlutterViewController,并绑定了自己的 FlutterEngineFlutterMethodChannel

HUD 的主要任务包含:

  • 通过 FlutterMethodChannel 向 native 派发 ocr 请求
  • 解析 native 对屏幕指定区域的 ocr 结果,并在内存中维护该结果供后继程序逻辑使用
  • 监听鼠标位置以在需要时计算释义展示的位置
  • 监听键盘按键状态,在变化时派发 ocr 请求并展示单词释义
  • 实时同步 Dashboard engine 传递过来的用户设置,并修改展示逻辑

下图展示了该模块所维护的数据映射到屏幕上时的可视化效果: text_block_representation

在 HUD 中的状态

本程序使用 riverpod 管理绝大部分的状态。在构筑本程序时,我的思考过程主要基于状态与状态变化,从展示释义键点击到渲染释义面板动画

在大量使用 riverpod 中的 Provider / StateProvider / ProviderContainer.listen 后,我可以构建一个高度异步,调用顺序不敏感,状态变化驱动的应用程序。在编码过程中,这种方式让程序员可以从一大串的命令式(Imperative)方法调用中解放出来,仅仅需要关注和确保每个最小逻辑单元——Provider,确保其实现是正确的即可,这减少了构建复杂和长串逻辑出错的可能

下面的两张图表展示了本程序的主要功能流程

切换展示释义流程:

用户关注文本块变更流程:

用户设置面板 (Dashboard)

Dashboard 是一个 NSPanel 实例,其内部承载了一个 FlutterViewController,并绑定了不同于 HUD 的独立 engine 与 method channel

Dashboard 的主要任务是为用户提供控制本程序的 UI,包括快捷键设置,开机自启,关于本程序等常见用户交互

同时,Dashboard 会维护本程序在内存和硬盘中的状态,并将这个状态通过 method channel 同步至下面即将要讲到的释义展示面板

原生侧 (native)

原生侧是由 flutter create 命令创建出来的传统 Xcode 工程

在原生侧,我主要关注的问题如下:

  • 处理来自 FlutterMethodChannel 的调用,并在需要时通过 FlutterResult 响应对应的 flutter engine
  • 维护承载 HUD 和 Dashboard 的两个 NSWindow,尤其是当用户的设备同时链接多个屏幕时,正确地设置 HUD 的位置
  • 封装并向 flutter 提供能力,如:
    • 监听鼠标位置变化和键盘状态变化,并将状态同步至 flutter
    • 为 flutter 提供 OCR 和语音合成能力
    • 在执行 OCR 时使用 swift 实现必要的性能优化

技术细节

得益于 Apple 一脉相承的 API 设计以及同样的编程语言,熟练于 iOS 应用开发(Cocoa Touch & UIKit)的程序员在面对与 macOS 交互的 Cocoa 和 AppKit 框架时,可以复用很多思想与逻辑

但相比于 iOS,macOS 给了用户更大的舞台,也让开发者面对了更多的挑战: 鼠标,键盘,窗口和屏幕

在开发本程序的过程中,我遇到了诸多的困难,也在茅塞顿开时收获了很多快乐,我在这一章节会记录一下

多屏幕

macOS 及其运行的硬件设备常常连接着多块屏幕,本程序的核心功能就是在任意的屏幕位置展示单词释义,这就要确保:

  1. 将释义展示面板(HUD)放置于正确的位置上
  2. flutter 在请求 native 捕捉屏幕时,捕捉的矩形框的位置正确
  3. flutter 在解析 ocr response 后,结果可以正确地和屏幕上真正的内容建立映射

和 UIKit 中常用的 CGRect 不同,AppKit 对设备屏幕的抽象 NSScreen,其坐标以 NSRect 计算,坐标系原点为 macOS 原始屏幕的左下角。而此时,如果你给你的设备连接上了其他的屏幕,NSScreen.screens 所呈现屏幕布局,可能就会变成下图所示的样子: appkit_screens_coordinate

screen 1 的左下角坐标值并非是 (0, 0),而是 screen 1screen 0 的 (0, 0) 点的相对位置 (w1, d2)。通过监听鼠标移动事件获取的鼠标位置 event.locationInWindow,也是相对于原点的 NSPoint

在开发时,仅仅将自己的思维从 iOS 的 CGRect 坐标系转化至 NSRect 坐标系还算简单。但在后继的逻辑中,因为 OCR 结果的 VNRectangleObservation.boundingBox 又会回到 CGRect 坐标系,坐标系的频繁转化确实会给人带来一定的困扰

屏幕捕捉

想捕捉 macOS 的屏幕,你可以调用 CGDisplayCreateImage 函数

但值得注意的是,想要截取屏幕上其他进程的内容(比如 IDE 或者浏览器),需要预先通过 CGRequestScreenCaptureAccess,申请到截取其他进程 UI 的权限,否则 CGDisplayCreateImage 只能拿到 macOS 的桌面(以及程序本身)

OCR

本程序使用 Apple 为开发者提供的 Vision - Recognizing Text 进行屏幕文本的提取

OCR 请求的调用由 flutter side 唤起:

void dispatchOCRRequestToNative() async {
final x = 100;
final y = 100;
final width = 300;
final height = 80;
final dimensions = [x, y, width, height];
final result = await methodChannel.invokeMethod("captureAndOCR", dimensions);

// parse the result from native
// ...
}

Native side 在接收到请求后,会执行截屏和识别操作:

import Vision
// ...
let result : FlutterResult = ...
let image = captureScreen()
let handler = VNImageRequestHandler(cgImage: image)
let request = VNRecognizeTextRequest { request, error in
// ocr finished
let parsedResult = parse(request.results)
result(parsedResult)
}
handler.perform([request])

性能表现

在 macbook 2021 的 m1 pro 上,以 320✕75 的设计分辨率截图(实际分辨率为 640✕150),每秒 11 帧的情况下进行长时间的截图和 OCR 操作,整个流程的延迟平均约为 65ms,我认为还算是一个可接受的状态

节能优化

即便是性能允许,优化也是应该做的,用户一旦使用你的 App,就感觉 macbook 的 C 面发热,这是无法容忍的

当前在 OCR 流程中主要的性能优化步骤是在 native 维护一个先进先出,最大容量为 40 帧的字典,以图片数据为 key 缓存 OCR 的结果,下面是简化后的代码:

let cacheManager = CacheManager<CFData, Result>()
// ...
let image = captureScreen()
let key = image.dataProvider.data
if let cachedResult = cacheManager[key] {
delegate.onOCRResult(cachedResult)
return
}
// ...
dispatchOCRRequestToDeviceGPU(image) { result
cacheManager[key] = result
// ... other logic
}
// ...

当鼠标指针位置不变,截图获取的图片不变时,native 在处理 OCR 请求时会先命中缓存,并直接返回结果,以减少非必需的 GPU 调用

在系统自带的活动监视器中查看进程。发现,在应用的缓存策略后,在本程序活动时,其 CPU/GPU 占用率确实降低了很多,同时,OCR 的缓存结果被降为了 5ms

备注

我感觉,以 CFData 作为 key 查询字典还不是效率最高的算法,应该可以继续探索一下

不截取自己

在我进行开发时,发现当本程序在展示单词释义 UI (HUD)时,因为 HUD 本身也会在一定范围内被截屏函数捕获,导致 HUD 会影响 OCR 的结果,而在 OCR 结果变动后,HUD 又会随之发生变动,再次影响 OCR 结果,就这样循环往复,连缓存也都失效了。在查阅和尝试大量的 API 后,我终于在 NSWindow 中找到了 sharingType 这个属性,屏幕捕捉方法捕获 HUD 自身,总算是打破了这个链条

结果解析,分词与定位

对 Vision - VNRecognizeTextRequest 的结果解析是个复杂的任务

执行这个复杂任务的原因有三:

  1. 本程序设计的交互是“用户将鼠标移动至感兴趣的单词上时展示单词释义”,这就要求我们知道屏幕上每个单词的具体位置和内容
  2. 本程序针对源代码特化,在面对驼峰命名法时,需要知道构成一个 symbol 的每个单词的意思:ThisIsAVeryVeryLongClassName -> This,Is,A,Very,Very,Long,Class,Name (因过于简单,丢弃长度小于 3 的英语单词)
  3. 我们要以单词的文本内容为 key,查询数据库。在文本序列包含特殊字符时,我们是无法从数据库中查到单词释义的。所以要移除非字母字符。同时,这一操作也可以自然而然地适配编程语言中的其他命名方式,比如移除下划线就可以适应蛇形命名法

假设我们在对下面的图片执行 OCR 请求

ocr_text_source

在 Native 端取 VNRecognizeTextRequest.results.topCandidates(1) 后,结果如下

[
// 同时包含 rect 信息
"Vision provides its text-recognition capabilities through VNRecognizeText",
// 同时包含 rect 信息
"Request, an image-based request type that finds and extracts text in images. The",
// 同时包含 rect 信息
"following example shows how to use VNImageRequestHandler to perform a",
// 同时包含 rect 信息
"VNRecognizeTextRequest for recognizing text in the specified CGImage.",
]

在 Native 端以空格字符和 VNRecognizedText.boundingBox(for:) 对 OCR 结果进行第一次整理,上述的结果会变为这样:

[
// 同时包含 rect 信息
"Vision", "provides", "its", "text-recognition", "capabilities", "through", "VNRecognizeText",
// 同时包含 rect 信息
"Request,", "an", "image-based", "request", "type", "that", "finds", "and", "extracts", "text", "in", "images.", "The",
// 同时包含 rect 信息
"following", "example", "shows", "how", "to", "use", "VNImageRequestHandler", "to", "perform", "a",
// 同时包含 rect 信息
"VNRecognizeTextRequest", "for", "recognizing", "text", "in", "the", "specified", "CGImage.",
]

最后将 boundingBox 的返回值和截屏尺寸进行乘算,返回给 flutter 端。Flutter 端在拿到结果后主要进行两个步骤:

  1. 将文本视为等宽字体,移除非英文字符,计算新的 rect
  2. 将文本视为等宽字体,对包含大写字母的字符串进行分割,计算新的 rect

这样,文本块的内容在 flutter 端就会变成这样:

[
// 同时包含 rect 信息
"Vision", "provides", "its", "text", "recognition", "capabilities", "through", "Recognize", "Text",
// 同时包含 rect 信息
"Request","image", "based", "request", "type", "that", "finds", "and", "extracts", "text", "images", "The",
// 同时包含 rect 信息
"following", "example", "shows", "how", "use", "Image", "Request", "Handler", "perform",
// 同时包含 rect 信息
"Recognize", "Text", "Request", "for", "recognizing", "text", "the", "specified", "Image",
]

最后,flutter 侧会以下面的代码对这些 OCR 信息进行抽象:

/// The representation of block
///
/// 文本块表征
class Block {
String text;
double x;
double y;
double w;
double h;
}

// ...

/// All text representations in memory
///
/// 所有文本块表征
final blocks = StateProvider<List<Block>>((_) => []);

上面的数据在屏幕上的可视化效果如下: text_block_representation_2

其中外围绿框代表截屏范围,内部的小绿框代表 native 端返回的结果,小绿框内部的蓝色框代表 flutter side 第二次解析结果。可以看到,"VNImageRequestHandler" 这个字符串被分割为了 "Image", "Request", "Handler" 这三个文本块。这样,我就可以知道用户到底对哪一段字母序列(单词)感兴趣了(假设源代码符号的命名是规范的 😇)。

备注

你也可以打开本程序的用户设置面板来直接观察其运行时表现: dashboard_inspect

多引擎与状态同步

本程序在运行时会创建两个 flutter engine,分别用于渲染"释义展示面板(HUD)"和"欢迎与设置页面(Dashboard)"

这两个引擎均通过基本方式创建,均采用 FlutterMethodChannel 与 native 进行通讯

双引擎共享同一份代码,经由不同的 dart 入口函数启动:

// 设置页面 (Dashboard)
dashboardEngine = FlutterEngine(name: "dashboard", project: nil)
dashboardChannel = FlutterMethodChannel(name: "dashboard", binaryMessenger: dashboardEngine.binaryMessenger)
dashboardEngine.run(withEntrypoint: "_dashboard")
//...

// 释义展示面板 (HUD)
hudEngine = FlutterEngine(name: "hud", project: nil)
hudChannel = FlutterMethodChannel(name: "hud", binaryMessenger: hudEngine.binaryMessenger)
hudChannel.run(withEntrypoint: "_hud")
备注

我个人没有使用官方提供的 FlutterEngineGroup 方案,因为在当初进行尝试时发现了引擎无响应的问题,至今仍然是 P2 级别的 open issue

跨引擎状态同步

本程序使用 riverpod 来管理绝大部分状态,所以在同步状态时,我也期望自己的自己的心智模型可以更贴近 riverpod

在运行时,Dashboard engine 需要将用户设置同步至 HUD engine (比如更改查词快捷键),我在这里监听了 Dashboard engine 对应的 StateProvider 的变更,并通过 method channel,经由 native 转发至 HUD engine,HUD engine 在收到了通知后,再更新自己维护的 StateProvider。从而实现了跨引擎的状态同步

释义查询

本程序在查询单词释义时使用本地 sqlite 数据库,数据库源于 ecdict-ultimate

鉴于开源数据库过于巨大,且存在一些本程序无需使用的 columns,我又使用 dart 和 drift 对其进行了清洗和剪裁

在查询时,本程序会将查询的本文串“打散”,并一齐请求数据库,以期能命中不断千变万化的英语单词:

final sequence = "qwertyuiopasdfghjkl";
final sequences = ["qwe","wer","ert",...,"qwer","wert",...,"qwert",...,"qwertyuiopasdfghjkl"];
final List<QueryResult> results = await queryDB(keys:sequences); // average latency: ~6ms

在请求完成后,本程序会对请求结果进行去重,逻辑如下:

  • 结果中包含 owl (猫头鹰)
  • 结果中包含 knowledge(知识)
  • 序列 "knowledge" 包含 "owl",移除查询结果中的 owl
备注

目前,该逻辑还没有实现对冲突的判定:如向数据库查询的 key 为 "gitignore",查询结果为 ["git", "tig", "ignore"],目前这三个结果都会被渲染到 HUD 上。但显然,选取 git 和 ignore 这两个查询结果造成的 “冲突” 是最小的

发音

本程序使用 AVFoundation - Speech synthesis API 来实时合成语音

该 API 可以同时在 iOS/macOS 上使用,且发音较为准确,以我自身的能力(全国大学英语四级)来看,效果还算满意,确实比我的发音准 🤣

当然,后继如果有更高的需求,也有很多其他可选方案

· 阅读需 6 分钟
Ce Wang

我的需求

以下是我的两个常用需求:

  • 阅读和理解浏览器中的英文内容,如 github,stackoverflow,reddit
  • 在桌面平台上阅读和理解一些从未被翻译成汉语的英文书籍,这些书籍的大部分为 PDF 文件

我遇到的困难

但在进行上述的阅读理解活动时,我遇到了一些困难:

  • 我经常忘记英语单词的意思,或不敢确认一个单词的意思。甚至,我经常会忘记 30 秒之前刚刚查过的单词
  • 当前各种查单词的应用程序和网页对于我来来说都有些繁琐,我需要等待词典软件的网络请求,用人脑忽略词典软件中的广告。我还要保证自己在繁琐的操作流程中不会分心
  • 使用全文翻译软件,会导致相当一部分的信息丢失或偏离原本含义,我需要对全文翻译软件纠错,或者直接查看原文才能了解到真正的释义

举个例子,我不知道“babel”这个单词的汉语意思。但是,当这个词在英语语料中出现时,它将你的思绪从“思考作者表达的含义”,转变为“思考 babel 是什么意思”。当我,查询到了“babel”的意思后,在返回来将自己的注意力投入之前的上下文。这个发生在人脑中的上下文切换,过程耗时,容易出错,正反馈几乎没有。

我认为,这些困难增加了我从事脑力活动时的心智负担,降低了我的阅读效率。甚至,直接导致我分心,精神不集中,最终放弃阅读,放弃获取知识。

面对上述的困难,我们通常的解决办法都是“背单词”。可是,背单词对于我来说是个痛苦的过程:

  • 我需要背诵大量的,使用频率极低,并且因使用频率低而非常容易变得生疏的单词
  • 考纲内的单词从来都是为大家准备的,背诵考纲内的单词不能覆盖到专业领域的词汇
  • 在汉语环境中,保证大脑对大量英语单词的熟悉和快速响应,需要花费很多额外的时间与精力

对我来说,背单词的效率太低了,奖励来的太晚,太不确定,需要强大的意志力才能持续下去。投入产出比太高。

矛盾

上述这些动作都会对于我来说是较大的心智负担。不断地干扰着我的阅读 - 理解过程。阻碍我通过英语获取信息与知识,而我当前所期望获取的信息与知识,大都由英语书写而成。

解决办法

面对这些困难。我经过种种思考,探索,实践。开发了这款名为 Valo Reader 的应用程序。这款应用程序通过 OCR 技术,实时地识别用户当前正在阅读的英文文本,经由程序内置词典和用户设置,展示单词释义。在这款应用程中,我期望接近可能地降低用户在阅读英文语料时,因为掌握的英语词汇量较少,而产生的额外心智负担。

🎉 我希望,我们在使用这款软件阅读英文,不需要有单词基础了,这可以大幅减少生疏单词带来的心智负担。助力我们的阅读 - 理解过程。

AppIcon

备注

想要在不同平台上获取 Valo Reader,请参阅安装

为什么不是 macOS 自带的三指轻点

当用户在使用 macOS 时可以通过触摸板快速唤起系统自带的字典查询单词, 这个功能被 macOS 称为查询与数据检测器(Look up & data detectors)

相比于系统自带功能, Valo Reader 当前主要有以下优势:

  • 所见即可查: 可以及时查询图片, 视频乃至电子游戏中的英语单词, 只要这个英语单词显示在了屏幕上, 这是 macOS 系统所做不到的
  • 编程友好: 可以查询源代码中的每个单词: PascalCase 将被 Valo Reader 识别为 "pascal""case" 并被分别查询

为什么不是其他词典软件

也有很多软件具有实时的功能, 但是据观察, 他们的在屏幕上取词的功能似乎都没有直接调用 Vision framework. 导致查询缓慢, 取词不精确等问题

- + \ No newline at end of file diff --git a/blog/motivation/index.html b/blog/motivation/index.html index f9a16fb..d854109 100644 --- a/blog/motivation/index.html +++ b/blog/motivation/index.html @@ -5,13 +5,13 @@ 开发目的 | Valo Reader - +

开发目的

· 阅读需 6 分钟
Ce Wang

我的需求

以下是我的两个常用需求:

  • 阅读和理解浏览器中的英文内容,如 github,stackoverflow,reddit
  • 在桌面平台上阅读和理解一些从未被翻译成汉语的英文书籍,这些书籍的大部分为 PDF 文件

我遇到的困难

但在进行上述的阅读理解活动时,我遇到了一些困难:

  • 我经常忘记英语单词的意思,或不敢确认一个单词的意思。甚至,我经常会忘记 30 秒之前刚刚查过的单词
  • 当前各种查单词的应用程序和网页对于我来来说都有些繁琐,我需要等待词典软件的网络请求,用人脑忽略词典软件中的广告。我还要保证自己在繁琐的操作流程中不会分心
  • 使用全文翻译软件,会导致相当一部分的信息丢失或偏离原本含义,我需要对全文翻译软件纠错,或者直接查看原文才能了解到真正的释义

举个例子,我不知道“babel”这个单词的汉语意思。但是,当这个词在英语语料中出现时,它将你的思绪从“思考作者表达的含义”,转变为“思考 babel 是什么意思”。当我,查询到了“babel”的意思后,在返回来将自己的注意力投入之前的上下文。这个发生在人脑中的上下文切换,过程耗时,容易出错,正反馈几乎没有。

我认为,这些困难增加了我从事脑力活动时的心智负担,降低了我的阅读效率。甚至,直接导致我分心,精神不集中,最终放弃阅读,放弃获取知识。

面对上述的困难,我们通常的解决办法都是“背单词”。可是,背单词对于我来说是个痛苦的过程:

  • 我需要背诵大量的,使用频率极低,并且因使用频率低而非常容易变得生疏的单词
  • 考纲内的单词从来都是为大家准备的,背诵考纲内的单词不能覆盖到专业领域的词汇
  • 在汉语环境中,保证大脑对大量英语单词的熟悉和快速响应,需要花费很多额外的时间与精力

对我来说,背单词的效率太低了,奖励来的太晚,太不确定,需要强大的意志力才能持续下去。投入产出比太高。

矛盾

上述这些动作都会对于我来说是较大的心智负担。不断地干扰着我的阅读 - 理解过程。阻碍我通过英语获取信息与知识,而我当前所期望获取的信息与知识,大都由英语书写而成。

解决办法

面对这些困难。我经过种种思考,探索,实践。开发了这款名为 Valo Reader 的应用程序。这款应用程序通过 OCR 技术,实时地识别用户当前正在阅读的英文文本,经由程序内置词典和用户设置,展示单词释义。在这款应用程中,我期望接近可能地降低用户在阅读英文语料时,因为掌握的英语词汇量较少,而产生的额外心智负担。

🎉 我希望,我们在使用这款软件阅读英文,不需要有单词基础了,这可以大幅减少生疏单词带来的心智负担。助力我们的阅读 - 理解过程。

AppIcon

备注

想要在不同平台上获取 Valo Reader,请参阅安装

为什么不是 macOS 自带的三指轻点

当用户在使用 macOS 时可以通过触摸板快速唤起系统自带的字典查询单词, 这个功能被 macOS 称为查询与数据检测器(Look up & data detectors)

相比于系统自带功能, Valo Reader 当前主要有以下优势:

  • 所见即可查: 可以及时查询图片, 视频乃至电子游戏中的英语单词, 只要这个英语单词显示在了屏幕上, 这是 macOS 系统所做不到的
  • 编程友好: 可以查询源代码中的每个单词: PascalCase 将被 Valo Reader 识别为 "pascal""case" 并被分别查询

为什么不是其他词典软件

也有很多软件具有实时的功能, 但是据观察, 他们的在屏幕上取词的功能似乎都没有直接调用 Vision framework. 导致查询缓慢, 取词不精确等问题

- + \ No newline at end of file diff --git a/blog/rss.xml b/blog/rss.xml index 24fa432..df25ea8 100644 --- a/blog/rss.xml +++ b/blog/rss.xml @@ -16,7 +16,7 @@ 摘要

本文主要介绍了 Valo Reader for macOS(下文简称为“本程序”)的项目布局工程架构运行时功能模块技术细节。你还可以在本文档的其他页面了解本程序的开发动机使用方式隐私与安全等内容

什么是 Valo Reader?

Valo Reader 这个项目是我个人在日常生活中逐渐萌生,演化和实施的想法。是我为了在自己电子设备设备上能轻松阅读英文内容,并渐近式地提高自己的英文水平,而自行设计和研发的一款产品,其核心功能展示如下: intro

当你在使用本程序时,将鼠标移动至你感到生疏的英语单词上。按下指定的按键(默认为 Fn),本程序即会在响应的位置展示对应的释义,同时使用语音合成来发音。

目前,你可以通过 Mac App Store其他方式来获取本程序

项目布局

下图展示了本项目的主体结构: -project_layout

本文的位置则处于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: +project_layout

本文的位于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: modules_light modules_dark

释义展示面板 (HUD)

HUD 是一个的 NSWindow 实例,其内部承载了一个 FlutterViewController,并绑定了自己的 FlutterEngineFlutterMethodChannel

HUD 的主要任务包含:

  • 通过 FlutterMethodChannel 向 native 派发 ocr 请求
  • 解析 native 对屏幕指定区域的 ocr 结果,并在内存中维护该结果供后继程序逻辑使用
  • 监听鼠标位置以在需要时计算释义展示的位置
  • 监听键盘按键状态,在变化时派发 ocr 请求并展示单词释义
  • 实时同步 Dashboard engine 传递过来的用户设置,并修改展示逻辑

下图展示了该模块所维护的数据映射到屏幕上时的可视化效果: text_block_representation

在 HUD 中的状态

本程序使用 riverpod 管理绝大部分的状态。在构筑本程序时,我的思考过程主要基于状态与状态变化,从展示释义键点击到渲染释义面板动画

在大量使用 riverpod 中的 Provider / StateProvider / ProviderContainer.listen 后,我可以构建一个高度异步,调用顺序不敏感,状态变化驱动的应用程序。在编码过程中,这种方式让程序员可以从一大串的命令式(Imperative)方法调用中解放出来,仅仅需要关注和确保每个最小逻辑单元——Provider,确保其实现是正确的即可,这减少了构建复杂和长串逻辑出错的可能

下面的两张图表展示了本程序的主要功能流程

切换展示释义流程:

用户关注文本块变更流程:

用户设置面板 (Dashboard)

Dashboard 是一个 NSPanel 实例,其内部承载了一个 FlutterViewController,并绑定了不同于 HUD 的独立 engine 与 method channel

Dashboard 的主要任务是为用户提供控制本程序的 UI,包括快捷键设置,开机自启,关于本程序等常见用户交互

同时,Dashboard 会维护本程序在内存和硬盘中的状态,并将这个状态通过 method channel 同步至下面即将要讲到的释义展示面板

原生侧 (native)

原生侧是由 flutter create 命令创建出来的传统 Xcode 工程

在原生侧,我主要关注的问题如下:

  • 处理来自 FlutterMethodChannel 的调用,并在需要时通过 FlutterResult 响应对应的 flutter engine
  • 维护承载 HUD 和 Dashboard 的两个 NSWindow,尤其是当用户的设备同时链接多个屏幕时,正确地设置 HUD 的位置
  • 封装并向 flutter 提供能力,如:
    • 监听鼠标位置变化和键盘状态变化,并将状态同步至 flutter
    • 为 flutter 提供 OCR 和语音合成能力
    • 在执行 OCR 时使用 swift 实现必要的性能优化

技术细节

得益于 Apple 一脉相承的 API 设计以及同样的编程语言,熟练于 iOS 应用开发(Cocoa Touch & UIKit)的程序员在面对与 macOS 交互的 Cocoa 和 AppKit 框架时,可以复用很多思想与逻辑

但相比于 iOS,macOS 给了用户更大的舞台,也让开发者面对了更多的挑战: 鼠标,键盘,窗口和屏幕

在开发本程序的过程中,我遇到了诸多的困难,也在茅塞顿开时收获了很多快乐,我在这一章节会记录一下

多屏幕

macOS 及其运行的硬件设备常常连接着多块屏幕,本程序的核心功能就是在任意的屏幕位置展示单词释义,这就要确保:

  1. 将释义展示面板(HUD)放置于正确的位置上
  2. flutter 在请求 native 捕捉屏幕时,捕捉的矩形框的位置正确
  3. flutter 在解析 ocr response 后,结果可以正确地和屏幕上真正的内容建立映射

和 UIKit 中常用的 CGRect 不同,AppKit 对设备屏幕的抽象 NSScreen,其坐标以 NSRect 计算,坐标系原点为 macOS 原始屏幕的左下角。而此时,如果你给你的设备连接上了其他的屏幕,NSScreen.screens 所呈现屏幕布局,可能就会变成下图所示的样子: diff --git a/blog/tags/index.html b/blog/tags/index.html index d05b54c..ddcd7c2 100644 --- a/blog/tags/index.html +++ b/blog/tags/index.html @@ -5,13 +5,13 @@ 标签 | Valo Reader - +

- + \ No newline at end of file diff --git a/blog/tags/motivation/index.html b/blog/tags/motivation/index.html index d1594c3..90072c6 100644 --- a/blog/tags/motivation/index.html +++ b/blog/tags/motivation/index.html @@ -5,13 +5,13 @@ 1 篇博文 含有标签「motivation」 | Valo Reader - +

1 篇博文 含有标签「motivation」

查看所有标签

· 阅读需 6 分钟
Ce Wang

我的需求

以下是我的两个常用需求:

  • 阅读和理解浏览器中的英文内容,如 github,stackoverflow,reddit
  • 在桌面平台上阅读和理解一些从未被翻译成汉语的英文书籍,这些书籍的大部分为 PDF 文件

我遇到的困难

但在进行上述的阅读理解活动时,我遇到了一些困难:

  • 我经常忘记英语单词的意思,或不敢确认一个单词的意思。甚至,我经常会忘记 30 秒之前刚刚查过的单词
  • 当前各种查单词的应用程序和网页对于我来来说都有些繁琐,我需要等待词典软件的网络请求,用人脑忽略词典软件中的广告。我还要保证自己在繁琐的操作流程中不会分心
  • 使用全文翻译软件,会导致相当一部分的信息丢失或偏离原本含义,我需要对全文翻译软件纠错,或者直接查看原文才能了解到真正的释义

举个例子,我不知道“babel”这个单词的汉语意思。但是,当这个词在英语语料中出现时,它将你的思绪从“思考作者表达的含义”,转变为“思考 babel 是什么意思”。当我,查询到了“babel”的意思后,在返回来将自己的注意力投入之前的上下文。这个发生在人脑中的上下文切换,过程耗时,容易出错,正反馈几乎没有。

我认为,这些困难增加了我从事脑力活动时的心智负担,降低了我的阅读效率。甚至,直接导致我分心,精神不集中,最终放弃阅读,放弃获取知识。

面对上述的困难,我们通常的解决办法都是“背单词”。可是,背单词对于我来说是个痛苦的过程:

  • 我需要背诵大量的,使用频率极低,并且因使用频率低而非常容易变得生疏的单词
  • 考纲内的单词从来都是为大家准备的,背诵考纲内的单词不能覆盖到专业领域的词汇
  • 在汉语环境中,保证大脑对大量英语单词的熟悉和快速响应,需要花费很多额外的时间与精力

对我来说,背单词的效率太低了,奖励来的太晚,太不确定,需要强大的意志力才能持续下去。投入产出比太高。

矛盾

上述这些动作都会对于我来说是较大的心智负担。不断地干扰着我的阅读 - 理解过程。阻碍我通过英语获取信息与知识,而我当前所期望获取的信息与知识,大都由英语书写而成。

解决办法

面对这些困难。我经过种种思考,探索,实践。开发了这款名为 Valo Reader 的应用程序。这款应用程序通过 OCR 技术,实时地识别用户当前正在阅读的英文文本,经由程序内置词典和用户设置,展示单词释义。在这款应用程中,我期望接近可能地降低用户在阅读英文语料时,因为掌握的英语词汇量较少,而产生的额外心智负担。

🎉 我希望,我们在使用这款软件阅读英文,不需要有单词基础了,这可以大幅减少生疏单词带来的心智负担。助力我们的阅读 - 理解过程。

AppIcon

备注

想要在不同平台上获取 Valo Reader,请参阅安装

为什么不是 macOS 自带的三指轻点

当用户在使用 macOS 时可以通过触摸板快速唤起系统自带的字典查询单词, 这个功能被 macOS 称为查询与数据检测器(Look up & data detectors)

相比于系统自带功能, Valo Reader 当前主要有以下优势:

  • 所见即可查: 可以及时查询图片, 视频乃至电子游戏中的英语单词, 只要这个英语单词显示在了屏幕上, 这是 macOS 系统所做不到的
  • 编程友好: 可以查询源代码中的每个单词: PascalCase 将被 Valo Reader 识别为 "pascal""case" 并被分别查询

为什么不是其他词典软件

也有很多软件具有实时的功能, 但是据观察, 他们的在屏幕上取词的功能似乎都没有直接调用 Vision framework. 导致查询缓慢, 取词不精确等问题

- + \ No newline at end of file diff --git a/blog/tags/technical/index.html b/blog/tags/technical/index.html index f883e98..b1a846b 100644 --- a/blog/tags/technical/index.html +++ b/blog/tags/technical/index.html @@ -5,21 +5,21 @@ 1 篇博文 含有标签「technical」 | Valo Reader - +

1 篇博文 含有标签「technical」

查看所有标签

· 阅读需 24 分钟
Ce Wang

摘要

本文主要介绍了 Valo Reader for macOS(下文简称为“本程序”)的项目布局工程架构运行时功能模块技术细节。你还可以在本文档的其他页面了解本程序的开发动机使用方式隐私与安全等内容

什么是 Valo Reader?

Valo Reader 这个项目是我个人在日常生活中逐渐萌生,演化和实施的想法。是我为了在自己电子设备设备上能轻松阅读英文内容,并渐近式地提高自己的英文水平,而自行设计和研发的一款产品,其核心功能展示如下: intro

当你在使用本程序时,将鼠标移动至你感到生疏的英语单词上。按下指定的按键(默认为 Fn),本程序即会在响应的位置展示对应的释义,同时使用语音合成来发音。

目前,你可以通过 Mac App Store其他方式来获取本程序

项目布局

下图展示了本项目的主体结构: -project_layout

本文的位置则处于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: +project_layout

本文的位于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: modules_light modules_dark

释义展示面板 (HUD)

HUD 是一个的 NSWindow 实例,其内部承载了一个 FlutterViewController,并绑定了自己的 FlutterEngineFlutterMethodChannel

HUD 的主要任务包含:

  • 通过 FlutterMethodChannel 向 native 派发 ocr 请求
  • 解析 native 对屏幕指定区域的 ocr 结果,并在内存中维护该结果供后继程序逻辑使用
  • 监听鼠标位置以在需要时计算释义展示的位置
  • 监听键盘按键状态,在变化时派发 ocr 请求并展示单词释义
  • 实时同步 Dashboard engine 传递过来的用户设置,并修改展示逻辑

下图展示了该模块所维护的数据映射到屏幕上时的可视化效果: text_block_representation

在 HUD 中的状态

本程序使用 riverpod 管理绝大部分的状态。在构筑本程序时,我的思考过程主要基于状态与状态变化,从展示释义键点击到渲染释义面板动画

在大量使用 riverpod 中的 Provider / StateProvider / ProviderContainer.listen 后,我可以构建一个高度异步,调用顺序不敏感,状态变化驱动的应用程序。在编码过程中,这种方式让程序员可以从一大串的命令式(Imperative)方法调用中解放出来,仅仅需要关注和确保每个最小逻辑单元——Provider,确保其实现是正确的即可,这减少了构建复杂和长串逻辑出错的可能

下面的两张图表展示了本程序的主要功能流程

切换展示释义流程:

用户关注文本块变更流程:

用户设置面板 (Dashboard)

Dashboard 是一个 NSPanel 实例,其内部承载了一个 FlutterViewController,并绑定了不同于 HUD 的独立 engine 与 method channel

Dashboard 的主要任务是为用户提供控制本程序的 UI,包括快捷键设置,开机自启,关于本程序等常见用户交互

同时,Dashboard 会维护本程序在内存和硬盘中的状态,并将这个状态通过 method channel 同步至下面即将要讲到的释义展示面板

原生侧 (native)

原生侧是由 flutter create 命令创建出来的传统 Xcode 工程

在原生侧,我主要关注的问题如下:

  • 处理来自 FlutterMethodChannel 的调用,并在需要时通过 FlutterResult 响应对应的 flutter engine
  • 维护承载 HUD 和 Dashboard 的两个 NSWindow,尤其是当用户的设备同时链接多个屏幕时,正确地设置 HUD 的位置
  • 封装并向 flutter 提供能力,如:
    • 监听鼠标位置变化和键盘状态变化,并将状态同步至 flutter
    • 为 flutter 提供 OCR 和语音合成能力
    • 在执行 OCR 时使用 swift 实现必要的性能优化

技术细节

得益于 Apple 一脉相承的 API 设计以及同样的编程语言,熟练于 iOS 应用开发(Cocoa Touch & UIKit)的程序员在面对与 macOS 交互的 Cocoa 和 AppKit 框架时,可以复用很多思想与逻辑

但相比于 iOS,macOS 给了用户更大的舞台,也让开发者面对了更多的挑战: 鼠标,键盘,窗口和屏幕

在开发本程序的过程中,我遇到了诸多的困难,也在茅塞顿开时收获了很多快乐,我在这一章节会记录一下

多屏幕

macOS 及其运行的硬件设备常常连接着多块屏幕,本程序的核心功能就是在任意的屏幕位置展示单词释义,这就要确保:

  1. 将释义展示面板(HUD)放置于正确的位置上
  2. flutter 在请求 native 捕捉屏幕时,捕捉的矩形框的位置正确
  3. flutter 在解析 ocr response 后,结果可以正确地和屏幕上真正的内容建立映射

和 UIKit 中常用的 CGRect 不同,AppKit 对设备屏幕的抽象 NSScreen,其坐标以 NSRect 计算,坐标系原点为 macOS 原始屏幕的左下角。而此时,如果你给你的设备连接上了其他的屏幕,NSScreen.screens 所呈现屏幕布局,可能就会变成下图所示的样子: appkit_screens_coordinate

screen 1 的左下角坐标值并非是 (0, 0),而是 screen 1screen 0 的 (0, 0) 点的相对位置 (w1, d2)。通过监听鼠标移动事件获取的鼠标位置 event.locationInWindow,也是相对于原点的 NSPoint

在开发时,仅仅将自己的思维从 iOS 的 CGRect 坐标系转化至 NSRect 坐标系还算简单。但在后继的逻辑中,因为 OCR 结果的 VNRectangleObservation.boundingBox 又会回到 CGRect 坐标系,坐标系的频繁转化确实会给人带来一定的困扰

屏幕捕捉

想捕捉 macOS 的屏幕,你可以调用 CGDisplayCreateImage 函数

但值得注意的是,想要截取屏幕上其他进程的内容(比如 IDE 或者浏览器),需要预先通过 CGRequestScreenCaptureAccess,申请到截取其他进程 UI 的权限,否则 CGDisplayCreateImage 只能拿到 macOS 的桌面(以及程序本身)

OCR

本程序使用 Apple 为开发者提供的 Vision - Recognizing Text 进行屏幕文本的提取

OCR 请求的调用由 flutter side 唤起:

void dispatchOCRRequestToNative() async {
final x = 100;
final y = 100;
final width = 300;
final height = 80;
final dimensions = [x, y, width, height];
final result = await methodChannel.invokeMethod("captureAndOCR", dimensions);

// parse the result from native
// ...
}

Native side 在接收到请求后,会执行截屏和识别操作:

import Vision
// ...
let result : FlutterResult = ...
let image = captureScreen()
let handler = VNImageRequestHandler(cgImage: image)
let request = VNRecognizeTextRequest { request, error in
// ocr finished
let parsedResult = parse(request.results)
result(parsedResult)
}
handler.perform([request])

性能表现

在 macbook 2021 的 m1 pro 上,以 320✕75 的设计分辨率截图(实际分辨率为 640✕150),每秒 11 帧的情况下进行长时间的截图和 OCR 操作,整个流程的延迟平均约为 65ms,我认为还算是一个可接受的状态

节能优化

即便是性能允许,优化也是应该做的,用户一旦使用你的 App,就感觉 macbook 的 C 面发热,这是无法容忍的

当前在 OCR 流程中主要的性能优化步骤是在 native 维护一个先进先出,最大容量为 40 帧的字典,以图片数据为 key 缓存 OCR 的结果,下面是简化后的代码:

let cacheManager = CacheManager<CFData, Result>()
// ...
let image = captureScreen()
let key = image.dataProvider.data
if let cachedResult = cacheManager[key] {
delegate.onOCRResult(cachedResult)
return
}
// ...
dispatchOCRRequestToDeviceGPU(image) { result
cacheManager[key] = result
// ... other logic
}
// ...

当鼠标指针位置不变,截图获取的图片不变时,native 在处理 OCR 请求时会先命中缓存,并直接返回结果,以减少非必需的 GPU 调用

在系统自带的活动监视器中查看进程。发现,在应用的缓存策略后,在本程序活动时,其 CPU/GPU 占用率确实降低了很多,同时,OCR 的缓存结果被降为了 5ms

备注

我感觉,以 CFData 作为 key 查询字典还不是效率最高的算法,应该可以继续探索一下

不截取自己

在我进行开发时,发现当本程序在展示单词释义 UI (HUD)时,因为 HUD 本身也会在一定范围内被截屏函数捕获,导致 HUD 会影响 OCR 的结果,而在 OCR 结果变动后,HUD 又会随之发生变动,再次影响 OCR 结果,就这样循环往复,连缓存也都失效了。在查阅和尝试大量的 API 后,我终于在 NSWindow 中找到了 sharingType 这个属性,屏幕捕捉方法捕获 HUD 自身,总算是打破了这个链条

结果解析,分词与定位

对 Vision - VNRecognizeTextRequest 的结果解析是个复杂的任务

执行这个复杂任务的原因有三:

  1. 本程序设计的交互是“用户将鼠标移动至感兴趣的单词上时展示单词释义”,这就要求我们知道屏幕上每个单词的具体位置和内容
  2. 本程序针对源代码特化,在面对驼峰命名法时,需要知道构成一个 symbol 的每个单词的意思:ThisIsAVeryVeryLongClassName -> This,Is,A,Very,Very,Long,Class,Name (因过于简单,丢弃长度小于 3 的英语单词)
  3. 我们要以单词的文本内容为 key,查询数据库。在文本序列包含特殊字符时,我们是无法从数据库中查到单词释义的。所以要移除非字母字符。同时,这一操作也可以自然而然地适配编程语言中的其他命名方式,比如移除下划线就可以适应蛇形命名法

假设我们在对下面的图片执行 OCR 请求

ocr_text_source

在 Native 端取 VNRecognizeTextRequest.results.topCandidates(1) 后,结果如下

[
// 同时包含 rect 信息
"Vision provides its text-recognition capabilities through VNRecognizeText",
// 同时包含 rect 信息
"Request, an image-based request type that finds and extracts text in images. The",
// 同时包含 rect 信息
"following example shows how to use VNImageRequestHandler to perform a",
// 同时包含 rect 信息
"VNRecognizeTextRequest for recognizing text in the specified CGImage.",
]

在 Native 端以空格字符和 VNRecognizedText.boundingBox(for:) 对 OCR 结果进行第一次整理,上述的结果会变为这样:

[
// 同时包含 rect 信息
"Vision", "provides", "its", "text-recognition", "capabilities", "through", "VNRecognizeText",
// 同时包含 rect 信息
"Request,", "an", "image-based", "request", "type", "that", "finds", "and", "extracts", "text", "in", "images.", "The",
// 同时包含 rect 信息
"following", "example", "shows", "how", "to", "use", "VNImageRequestHandler", "to", "perform", "a",
// 同时包含 rect 信息
"VNRecognizeTextRequest", "for", "recognizing", "text", "in", "the", "specified", "CGImage.",
]

最后将 boundingBox 的返回值和截屏尺寸进行乘算,返回给 flutter 端。Flutter 端在拿到结果后主要进行两个步骤:

  1. 将文本视为等宽字体,移除非英文字符,计算新的 rect
  2. 将文本视为等宽字体,对包含大写字母的字符串进行分割,计算新的 rect

这样,文本块的内容在 flutter 端就会变成这样:

[
// 同时包含 rect 信息
"Vision", "provides", "its", "text", "recognition", "capabilities", "through", "Recognize", "Text",
// 同时包含 rect 信息
"Request","image", "based", "request", "type", "that", "finds", "and", "extracts", "text", "images", "The",
// 同时包含 rect 信息
"following", "example", "shows", "how", "use", "Image", "Request", "Handler", "perform",
// 同时包含 rect 信息
"Recognize", "Text", "Request", "for", "recognizing", "text", "the", "specified", "Image",
]

最后,flutter 侧会以下面的代码对这些 OCR 信息进行抽象:

/// The representation of block
///
/// 文本块表征
class Block {
String text;
double x;
double y;
double w;
double h;
}

// ...

/// All text representations in memory
///
/// 所有文本块表征
final blocks = StateProvider<List<Block>>((_) => []);

上面的数据在屏幕上的可视化效果如下: text_block_representation_2

其中外围绿框代表截屏范围,内部的小绿框代表 native 端返回的结果,小绿框内部的蓝色框代表 flutter side 第二次解析结果。可以看到,"VNImageRequestHandler" 这个字符串被分割为了 "Image", "Request", "Handler" 这三个文本块。这样,我就可以知道用户到底对哪一段字母序列(单词)感兴趣了(假设源代码符号的命名是规范的 😇)。

备注

你也可以打开本程序的用户设置面板来直接观察其运行时表现: dashboard_inspect

多引擎与状态同步

本程序在运行时会创建两个 flutter engine,分别用于渲染"释义展示面板(HUD)"和"欢迎与设置页面(Dashboard)"

这两个引擎均通过基本方式创建,均采用 FlutterMethodChannel 与 native 进行通讯

双引擎共享同一份代码,经由不同的 dart 入口函数启动:

// 设置页面 (Dashboard)
dashboardEngine = FlutterEngine(name: "dashboard", project: nil)
dashboardChannel = FlutterMethodChannel(name: "dashboard", binaryMessenger: dashboardEngine.binaryMessenger)
dashboardEngine.run(withEntrypoint: "_dashboard")
//...

// 释义展示面板 (HUD)
hudEngine = FlutterEngine(name: "hud", project: nil)
hudChannel = FlutterMethodChannel(name: "hud", binaryMessenger: hudEngine.binaryMessenger)
hudChannel.run(withEntrypoint: "_hud")
备注

我个人没有使用官方提供的 FlutterEngineGroup 方案,因为在当初进行尝试时发现了引擎无响应的问题,至今仍然是 P2 级别的 open issue

跨引擎状态同步

本程序使用 riverpod 来管理绝大部分状态,所以在同步状态时,我也期望自己的自己的心智模型可以更贴近 riverpod

在运行时,Dashboard engine 需要将用户设置同步至 HUD engine (比如更改查词快捷键),我在这里监听了 Dashboard engine 对应的 StateProvider 的变更,并通过 method channel,经由 native 转发至 HUD engine,HUD engine 在收到了通知后,再更新自己维护的 StateProvider。从而实现了跨引擎的状态同步

释义查询

本程序在查询单词释义时使用本地 sqlite 数据库,数据库源于 ecdict-ultimate

鉴于开源数据库过于巨大,且存在一些本程序无需使用的 columns,我又使用 dart 和 drift 对其进行了清洗和剪裁

在查询时,本程序会将查询的本文串“打散”,并一齐请求数据库,以期能命中不断千变万化的英语单词:

final sequence = "qwertyuiopasdfghjkl";
final sequences = ["qwe","wer","ert",...,"qwer","wert",...,"qwert",...,"qwertyuiopasdfghjkl"];
final List<QueryResult> results = await queryDB(keys:sequences); // average latency: ~6ms

在请求完成后,本程序会对请求结果进行去重,逻辑如下:

  • 结果中包含 owl (猫头鹰)
  • 结果中包含 knowledge(知识)
  • 序列 "knowledge" 包含 "owl",移除查询结果中的 owl
备注

目前,该逻辑还没有实现对冲突的判定:如向数据库查询的 key 为 "gitignore",查询结果为 ["git", "tig", "ignore"],目前这三个结果都会被渲染到 HUD 上。但显然,选取 git 和 ignore 这两个查询结果造成的 “冲突” 是最小的

发音

本程序使用 AVFoundation - Speech synthesis API 来实时合成语音

该 API 可以同时在 iOS/macOS 上使用,且发音较为准确,以我自身的能力(全国大学英语四级)来看,效果还算满意,确实比我的发音准 🤣

当然,后继如果有更高的需求,也有很多其他可选方案

- + \ No newline at end of file diff --git a/blog/tech_detail_macos/index.html b/blog/tech_detail_macos/index.html index 832a749..d0ce727 100644 --- a/blog/tech_detail_macos/index.html +++ b/blog/tech_detail_macos/index.html @@ -5,21 +5,21 @@ Valo Reader for macOS 技术概要 | Valo Reader - +

Valo Reader for macOS 技术概要

· 阅读需 24 分钟
Ce Wang

摘要

本文主要介绍了 Valo Reader for macOS(下文简称为“本程序”)的项目布局工程架构运行时功能模块技术细节。你还可以在本文档的其他页面了解本程序的开发动机使用方式隐私与安全等内容

什么是 Valo Reader?

Valo Reader 这个项目是我个人在日常生活中逐渐萌生,演化和实施的想法。是我为了在自己电子设备设备上能轻松阅读英文内容,并渐近式地提高自己的英文水平,而自行设计和研发的一款产品,其核心功能展示如下: intro

当你在使用本程序时,将鼠标移动至你感到生疏的英语单词上。按下指定的按键(默认为 Fn),本程序即会在响应的位置展示对应的释义,同时使用语音合成来发音。

目前,你可以通过 Mac App Store其他方式来获取本程序

项目布局

下图展示了本项目的主体结构: -project_layout

本文的位置则处于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: +project_layout

本文的位于上图中的浅黄色部分,由 docusaurus 生成

本项目的想法最早实现于在浏览器中运行的 js 脚本,你可以在 github 上看到本项目在浏览器上的早期实现及对应的功能展示,即图中浅绿色部分

而本文主要说明了上图中 macOS(浅蓝色部分)的运行方式与技术细节

工程架构

本程序由 flutter create XXX --platform macos 命令创建,其源代码主要分为两部分

flutter 侧

这部分由 dart 编写,可分为三部分

主程序

该部分的主要功能有:

  1. 通过 method channel 实现 flutter 与 native 的交互
  2. 渲染用户设置面板(Dashboard)
  3. 渲染释义展示面板(HUD)与诊断检查信息
  4. 使用第三方依赖提供的功能

flutter package: 文本块表征

这部分是本程序的核心代码,在 macOS / iOS / Android 三端共享部分逻辑与 UI,其主要功能由有:

  1. 解析用户设备屏幕上识别到的文本块,结构化文本块,并将其维护到本程序的内存中
  2. 声明并实现用户视觉焦点与已知的文本信息的交互
  3. 渲染释义展示面板(HUD)

flutter package: 基础库

在我的所有 flutter 项目中共享的基础依赖

native 侧

这部分由 swift 编写,可分为三部分

主程序部分

持有 flutter engine,通过 method channel 打通 flutter 与 BaseOCR 以及应用程序本身的双向调用

创建项目时默认的 FlutterViewController 已被删除

CocoaPods dependency: Cocoa 功能库

与 Cocoa framework 交互,如:权限申请,键盘监听,鼠标监听,截取屏幕,移动承载 flutter 的 NSWindow以及提供基本若干原生能力

该部分使用 CocoaPods 创建,我也会在该依赖中学习,尝试和实现 Cocoa/AppKit 独有的 API 与功能

CocoaPods dependency: OCR 功能库

Apple Vision framework - Text Recognizing 交互,并实现部分逻辑,如:发起文本识别请求,维护文本识别响应缓存

该部分使用 CocoaPods 创建,在 macOS 和 iOS 项目同时依赖并共享代码

运行时功能模块

在运行时,本程序主要可分为用户设置面板 (Dashboard)、释义展示面板 (HUD) 和原生侧 (native) 三个模块

下图展示了本程序在运行时的主要模块及其通讯: modules_light modules_dark

释义展示面板 (HUD)

HUD 是一个的 NSWindow 实例,其内部承载了一个 FlutterViewController,并绑定了自己的 FlutterEngineFlutterMethodChannel

HUD 的主要任务包含:

  • 通过 FlutterMethodChannel 向 native 派发 ocr 请求
  • 解析 native 对屏幕指定区域的 ocr 结果,并在内存中维护该结果供后继程序逻辑使用
  • 监听鼠标位置以在需要时计算释义展示的位置
  • 监听键盘按键状态,在变化时派发 ocr 请求并展示单词释义
  • 实时同步 Dashboard engine 传递过来的用户设置,并修改展示逻辑

下图展示了该模块所维护的数据映射到屏幕上时的可视化效果: text_block_representation

在 HUD 中的状态

本程序使用 riverpod 管理绝大部分的状态。在构筑本程序时,我的思考过程主要基于状态与状态变化,从展示释义键点击到渲染释义面板动画

在大量使用 riverpod 中的 Provider / StateProvider / ProviderContainer.listen 后,我可以构建一个高度异步,调用顺序不敏感,状态变化驱动的应用程序。在编码过程中,这种方式让程序员可以从一大串的命令式(Imperative)方法调用中解放出来,仅仅需要关注和确保每个最小逻辑单元——Provider,确保其实现是正确的即可,这减少了构建复杂和长串逻辑出错的可能

下面的两张图表展示了本程序的主要功能流程

切换展示释义流程:

用户关注文本块变更流程:

用户设置面板 (Dashboard)

Dashboard 是一个 NSPanel 实例,其内部承载了一个 FlutterViewController,并绑定了不同于 HUD 的独立 engine 与 method channel

Dashboard 的主要任务是为用户提供控制本程序的 UI,包括快捷键设置,开机自启,关于本程序等常见用户交互

同时,Dashboard 会维护本程序在内存和硬盘中的状态,并将这个状态通过 method channel 同步至下面即将要讲到的释义展示面板

原生侧 (native)

原生侧是由 flutter create 命令创建出来的传统 Xcode 工程

在原生侧,我主要关注的问题如下:

  • 处理来自 FlutterMethodChannel 的调用,并在需要时通过 FlutterResult 响应对应的 flutter engine
  • 维护承载 HUD 和 Dashboard 的两个 NSWindow,尤其是当用户的设备同时链接多个屏幕时,正确地设置 HUD 的位置
  • 封装并向 flutter 提供能力,如:
    • 监听鼠标位置变化和键盘状态变化,并将状态同步至 flutter
    • 为 flutter 提供 OCR 和语音合成能力
    • 在执行 OCR 时使用 swift 实现必要的性能优化

技术细节

得益于 Apple 一脉相承的 API 设计以及同样的编程语言,熟练于 iOS 应用开发(Cocoa Touch & UIKit)的程序员在面对与 macOS 交互的 Cocoa 和 AppKit 框架时,可以复用很多思想与逻辑

但相比于 iOS,macOS 给了用户更大的舞台,也让开发者面对了更多的挑战: 鼠标,键盘,窗口和屏幕

在开发本程序的过程中,我遇到了诸多的困难,也在茅塞顿开时收获了很多快乐,我在这一章节会记录一下

多屏幕

macOS 及其运行的硬件设备常常连接着多块屏幕,本程序的核心功能就是在任意的屏幕位置展示单词释义,这就要确保:

  1. 将释义展示面板(HUD)放置于正确的位置上
  2. flutter 在请求 native 捕捉屏幕时,捕捉的矩形框的位置正确
  3. flutter 在解析 ocr response 后,结果可以正确地和屏幕上真正的内容建立映射

和 UIKit 中常用的 CGRect 不同,AppKit 对设备屏幕的抽象 NSScreen,其坐标以 NSRect 计算,坐标系原点为 macOS 原始屏幕的左下角。而此时,如果你给你的设备连接上了其他的屏幕,NSScreen.screens 所呈现屏幕布局,可能就会变成下图所示的样子: appkit_screens_coordinate

screen 1 的左下角坐标值并非是 (0, 0),而是 screen 1screen 0 的 (0, 0) 点的相对位置 (w1, d2)。通过监听鼠标移动事件获取的鼠标位置 event.locationInWindow,也是相对于原点的 NSPoint

在开发时,仅仅将自己的思维从 iOS 的 CGRect 坐标系转化至 NSRect 坐标系还算简单。但在后继的逻辑中,因为 OCR 结果的 VNRectangleObservation.boundingBox 又会回到 CGRect 坐标系,坐标系的频繁转化确实会给人带来一定的困扰

屏幕捕捉

想捕捉 macOS 的屏幕,你可以调用 CGDisplayCreateImage 函数

但值得注意的是,想要截取屏幕上其他进程的内容(比如 IDE 或者浏览器),需要预先通过 CGRequestScreenCaptureAccess,申请到截取其他进程 UI 的权限,否则 CGDisplayCreateImage 只能拿到 macOS 的桌面(以及程序本身)

OCR

本程序使用 Apple 为开发者提供的 Vision - Recognizing Text 进行屏幕文本的提取

OCR 请求的调用由 flutter side 唤起:

void dispatchOCRRequestToNative() async {
final x = 100;
final y = 100;
final width = 300;
final height = 80;
final dimensions = [x, y, width, height];
final result = await methodChannel.invokeMethod("captureAndOCR", dimensions);

// parse the result from native
// ...
}

Native side 在接收到请求后,会执行截屏和识别操作:

import Vision
// ...
let result : FlutterResult = ...
let image = captureScreen()
let handler = VNImageRequestHandler(cgImage: image)
let request = VNRecognizeTextRequest { request, error in
// ocr finished
let parsedResult = parse(request.results)
result(parsedResult)
}
handler.perform([request])

性能表现

在 macbook 2021 的 m1 pro 上,以 320✕75 的设计分辨率截图(实际分辨率为 640✕150),每秒 11 帧的情况下进行长时间的截图和 OCR 操作,整个流程的延迟平均约为 65ms,我认为还算是一个可接受的状态

节能优化

即便是性能允许,优化也是应该做的,用户一旦使用你的 App,就感觉 macbook 的 C 面发热,这是无法容忍的

当前在 OCR 流程中主要的性能优化步骤是在 native 维护一个先进先出,最大容量为 40 帧的字典,以图片数据为 key 缓存 OCR 的结果,下面是简化后的代码:

let cacheManager = CacheManager<CFData, Result>()
// ...
let image = captureScreen()
let key = image.dataProvider.data
if let cachedResult = cacheManager[key] {
delegate.onOCRResult(cachedResult)
return
}
// ...
dispatchOCRRequestToDeviceGPU(image) { result
cacheManager[key] = result
// ... other logic
}
// ...

当鼠标指针位置不变,截图获取的图片不变时,native 在处理 OCR 请求时会先命中缓存,并直接返回结果,以减少非必需的 GPU 调用

在系统自带的活动监视器中查看进程。发现,在应用的缓存策略后,在本程序活动时,其 CPU/GPU 占用率确实降低了很多,同时,OCR 的缓存结果被降为了 5ms

备注

我感觉,以 CFData 作为 key 查询字典还不是效率最高的算法,应该可以继续探索一下

不截取自己

在我进行开发时,发现当本程序在展示单词释义 UI (HUD)时,因为 HUD 本身也会在一定范围内被截屏函数捕获,导致 HUD 会影响 OCR 的结果,而在 OCR 结果变动后,HUD 又会随之发生变动,再次影响 OCR 结果,就这样循环往复,连缓存也都失效了。在查阅和尝试大量的 API 后,我终于在 NSWindow 中找到了 sharingType 这个属性,屏幕捕捉方法捕获 HUD 自身,总算是打破了这个链条

结果解析,分词与定位

对 Vision - VNRecognizeTextRequest 的结果解析是个复杂的任务

执行这个复杂任务的原因有三:

  1. 本程序设计的交互是“用户将鼠标移动至感兴趣的单词上时展示单词释义”,这就要求我们知道屏幕上每个单词的具体位置和内容
  2. 本程序针对源代码特化,在面对驼峰命名法时,需要知道构成一个 symbol 的每个单词的意思:ThisIsAVeryVeryLongClassName -> This,Is,A,Very,Very,Long,Class,Name (因过于简单,丢弃长度小于 3 的英语单词)
  3. 我们要以单词的文本内容为 key,查询数据库。在文本序列包含特殊字符时,我们是无法从数据库中查到单词释义的。所以要移除非字母字符。同时,这一操作也可以自然而然地适配编程语言中的其他命名方式,比如移除下划线就可以适应蛇形命名法

假设我们在对下面的图片执行 OCR 请求

ocr_text_source

在 Native 端取 VNRecognizeTextRequest.results.topCandidates(1) 后,结果如下

[
// 同时包含 rect 信息
"Vision provides its text-recognition capabilities through VNRecognizeText",
// 同时包含 rect 信息
"Request, an image-based request type that finds and extracts text in images. The",
// 同时包含 rect 信息
"following example shows how to use VNImageRequestHandler to perform a",
// 同时包含 rect 信息
"VNRecognizeTextRequest for recognizing text in the specified CGImage.",
]

在 Native 端以空格字符和 VNRecognizedText.boundingBox(for:) 对 OCR 结果进行第一次整理,上述的结果会变为这样:

[
// 同时包含 rect 信息
"Vision", "provides", "its", "text-recognition", "capabilities", "through", "VNRecognizeText",
// 同时包含 rect 信息
"Request,", "an", "image-based", "request", "type", "that", "finds", "and", "extracts", "text", "in", "images.", "The",
// 同时包含 rect 信息
"following", "example", "shows", "how", "to", "use", "VNImageRequestHandler", "to", "perform", "a",
// 同时包含 rect 信息
"VNRecognizeTextRequest", "for", "recognizing", "text", "in", "the", "specified", "CGImage.",
]

最后将 boundingBox 的返回值和截屏尺寸进行乘算,返回给 flutter 端。Flutter 端在拿到结果后主要进行两个步骤:

  1. 将文本视为等宽字体,移除非英文字符,计算新的 rect
  2. 将文本视为等宽字体,对包含大写字母的字符串进行分割,计算新的 rect

这样,文本块的内容在 flutter 端就会变成这样:

[
// 同时包含 rect 信息
"Vision", "provides", "its", "text", "recognition", "capabilities", "through", "Recognize", "Text",
// 同时包含 rect 信息
"Request","image", "based", "request", "type", "that", "finds", "and", "extracts", "text", "images", "The",
// 同时包含 rect 信息
"following", "example", "shows", "how", "use", "Image", "Request", "Handler", "perform",
// 同时包含 rect 信息
"Recognize", "Text", "Request", "for", "recognizing", "text", "the", "specified", "Image",
]

最后,flutter 侧会以下面的代码对这些 OCR 信息进行抽象:

/// The representation of block
///
/// 文本块表征
class Block {
String text;
double x;
double y;
double w;
double h;
}

// ...

/// All text representations in memory
///
/// 所有文本块表征
final blocks = StateProvider<List<Block>>((_) => []);

上面的数据在屏幕上的可视化效果如下: text_block_representation_2

其中外围绿框代表截屏范围,内部的小绿框代表 native 端返回的结果,小绿框内部的蓝色框代表 flutter side 第二次解析结果。可以看到,"VNImageRequestHandler" 这个字符串被分割为了 "Image", "Request", "Handler" 这三个文本块。这样,我就可以知道用户到底对哪一段字母序列(单词)感兴趣了(假设源代码符号的命名是规范的 😇)。

备注

你也可以打开本程序的用户设置面板来直接观察其运行时表现: dashboard_inspect

多引擎与状态同步

本程序在运行时会创建两个 flutter engine,分别用于渲染"释义展示面板(HUD)"和"欢迎与设置页面(Dashboard)"

这两个引擎均通过基本方式创建,均采用 FlutterMethodChannel 与 native 进行通讯

双引擎共享同一份代码,经由不同的 dart 入口函数启动:

// 设置页面 (Dashboard)
dashboardEngine = FlutterEngine(name: "dashboard", project: nil)
dashboardChannel = FlutterMethodChannel(name: "dashboard", binaryMessenger: dashboardEngine.binaryMessenger)
dashboardEngine.run(withEntrypoint: "_dashboard")
//...

// 释义展示面板 (HUD)
hudEngine = FlutterEngine(name: "hud", project: nil)
hudChannel = FlutterMethodChannel(name: "hud", binaryMessenger: hudEngine.binaryMessenger)
hudChannel.run(withEntrypoint: "_hud")
备注

我个人没有使用官方提供的 FlutterEngineGroup 方案,因为在当初进行尝试时发现了引擎无响应的问题,至今仍然是 P2 级别的 open issue

跨引擎状态同步

本程序使用 riverpod 来管理绝大部分状态,所以在同步状态时,我也期望自己的自己的心智模型可以更贴近 riverpod

在运行时,Dashboard engine 需要将用户设置同步至 HUD engine (比如更改查词快捷键),我在这里监听了 Dashboard engine 对应的 StateProvider 的变更,并通过 method channel,经由 native 转发至 HUD engine,HUD engine 在收到了通知后,再更新自己维护的 StateProvider。从而实现了跨引擎的状态同步

释义查询

本程序在查询单词释义时使用本地 sqlite 数据库,数据库源于 ecdict-ultimate

鉴于开源数据库过于巨大,且存在一些本程序无需使用的 columns,我又使用 dart 和 drift 对其进行了清洗和剪裁

在查询时,本程序会将查询的本文串“打散”,并一齐请求数据库,以期能命中不断千变万化的英语单词:

final sequence = "qwertyuiopasdfghjkl";
final sequences = ["qwe","wer","ert",...,"qwer","wert",...,"qwert",...,"qwertyuiopasdfghjkl"];
final List<QueryResult> results = await queryDB(keys:sequences); // average latency: ~6ms

在请求完成后,本程序会对请求结果进行去重,逻辑如下:

  • 结果中包含 owl (猫头鹰)
  • 结果中包含 knowledge(知识)
  • 序列 "knowledge" 包含 "owl",移除查询结果中的 owl
备注

目前,该逻辑还没有实现对冲突的判定:如向数据库查询的 key 为 "gitignore",查询结果为 ["git", "tig", "ignore"],目前这三个结果都会被渲染到 HUD 上。但显然,选取 git 和 ignore 这两个查询结果造成的 “冲突” 是最小的

发音

本程序使用 AVFoundation - Speech synthesis API 来实时合成语音

该 API 可以同时在 iOS/macOS 上使用,且发音较为准确,以我自身的能力(全国大学英语四级)来看,效果还算满意,确实比我的发音准 🤣

当然,后继如果有更高的需求,也有很多其他可选方案

- + \ No newline at end of file diff --git a/docs/delete/index.html b/docs/delete/index.html index 6b76355..353973b 100644 --- a/docs/delete/index.html +++ b/docs/delete/index.html @@ -5,13 +5,13 @@ 删除 | Valo Reader - +

删除

如果你在使用了 Valo Reader 后,认为这个程序不能满足你的需求。

我非常希望知道你宝贵的建议,请联系开发者

你可以直接将 Valo Reader.app 文件从你的访达(Finder)中删除

你还可以运行下面的指令以清除缓存和支持文件:

rm -rf /Users/YOUR_USERNAME_IN_OS/Library/Application\ Support/com.valo.babel;
rm -rf /Users/YOUR_USERNAME_IN_OS/Library/Caches/com.valo.babel;
rm -rf /Users/YOUR_USERNAME_IN_OS/Library/Preferences/com.valo.babel.plist;
rm -rf /Users/YOUR_USERNAME_IN_OS/Library/Saved\ Application\ State/com.valo.babel.savedState;
rm -rf /Users/YOUR_USERNAME_IN_OS/Library/LaunchAgents/Valo\ Reader.plist;
警告

请确保你知晓上述命令的目标文件

- + \ No newline at end of file diff --git a/docs/feedback/index.html b/docs/feedback/index.html index ef61388..f2c015f 100644 --- a/docs/feedback/index.html +++ b/docs/feedback/index.html @@ -5,13 +5,13 @@ 交流与反馈 | Valo Reader - +

交流与反馈

通过 github

你可以在本项目 github repo 上通过 issue 提交 bug 和改进建议等

通过邮件

你可以通过我的电子邮箱联系我 halowang1991@gmail.com

通过应用市场本身的评价

你可以在 App Store / Play Store 发表你的评价

- + \ No newline at end of file diff --git a/docs/installation/index.html b/docs/installation/index.html index 6b01f0f..00d33f9 100644 --- a/docs/installation/index.html +++ b/docs/installation/index.html @@ -5,13 +5,13 @@ 安装 | Valo Reader - +

安装

当前你可以在 iOS,Android 上安装 Valo Reader。你也可以在 macOS 上安装 Valo Reader Beta 版本。

在 macOS 上安装

你可以通过以下方式在你的 macOS 中安装 Valo Reader

要求系统版本高于 macOS Big Sur 11.1

在 iOS / iPadOS 安装

如果你有一台运行 iOS 13 / iPadOS 13 及以上的 iPhone / iPad,你可以前往 App Store 下载 Valo Reader

点击此处进入 App Store

在安卓平台上安装

  • 你可以前往 Play Store 上下载 Valo Reader
  • 你也可以前往 Github release 页面下载 Valo Reader (Android) 及其历史版本

点击此处进入 Play Store

- + \ No newline at end of file diff --git a/docs/intro/index.html b/docs/intro/index.html index d2a91e8..bcb50d4 100644 --- a/docs/intro/index.html +++ b/docs/intro/index.html @@ -5,13 +5,13 @@ 简介 | Valo Reader - +

简介

Valo Reader 是一款英语阅读辅助软件

intro

Valo Reader 通过 OCR 技术,实时识别你当前正在阅读的英文文本,经由程序内置词典和你的自定义设置,在收到你的指示后,立即显示英文单词的释义。

通过 Valo Reader,用户可以更加轻松地阅读屏幕上的英文内容,减轻心智负担,乃至渐进式地提高自己的英文水平

想要获取 Valo Reader?你可以在 macOS、iOS 以及 Android 上下载并安装这款软件。

想要进一步了解 Valo Reader 作用?请参阅开发目的

- + \ No newline at end of file diff --git a/docs/others/index.html b/docs/others/index.html index 2423887..02e8fdb 100644 --- a/docs/others/index.html +++ b/docs/others/index.html @@ -5,13 +5,13 @@ 其他 | Valo Reader - +

其他

TODO

bug 修复

这款产品当前肯定还是有一些 bug 的,我在后续的开发中会依照优先级对其进行修复

用户自定义词典

我期望让用户可以自定义词典,自己增删改查 Valo Reader 内置的词典。并且,在多个设备上(比如,在 iOS 和 macOS 之间)同步用户的词典

更多功能

我的脑中还有很多关于提高英语阅读体验的点子等待实现

商业化

我还是非常期待能获得物质上的奖励的。RMB 是对早期独立开发者最好的鼓励。也是对 app 最好的支持

Windows

受限于我的技术能力,我只完成了 iOS / Android / macOS 上的核心功能。目前还没研究如何在 Windows 上实现 Valo Reader。不过,我感觉这是一件非常有趣的事情:

想想看,你凭借兴趣游玩《群星》、《星空》这样的 PC 游戏作品。边玩边学英语。玩了一遍之后去 reddit 相关论坛随便逛,随便发言。

🥳 岂不美哉

使用 Valo Reader 额外的好处

备注

一下观点仅代表我个人的主观感觉,未经证实

  • Valo Reader 可以永久性地提升我的英语水平,即便我不再使用它了
  • 这款软件可以降低阅读英语的不适感,恐惧感和心理压力。是某种程度上的脱敏治疗
  • 英语水平的提升又可以帮助我提高编程水平:我对关键字变得更加敏感了,我可以直接阅读 api 文档了。我可以通过 suggestions 探索自己想要查询的 api 了
- + \ No newline at end of file diff --git a/docs/privacy_security/index.html b/docs/privacy_security/index.html index c1f253e..2756ddf 100644 --- a/docs/privacy_security/index.html +++ b/docs/privacy_security/index.html @@ -5,13 +5,13 @@ 隐私与安全 | Valo Reader - +

隐私与安全

应用程序本身

Valo Reader 不会将你阅读的任何内容,以任何形式存储至或发送出设备。你无需担心 Valo Reader 会泄露任何隐私或重要数据。

应用程序所依赖的 OCR SDK

OCR

iOS / macOS

在 iOS 和 macOS 平台上,Valo Reader 使用 Apple Vision Framework - Recognizing Text

备注

想要查看 Valo Reader for macOS 依赖的其他第三方组件, 请点击菜单栏, 点击“License”

Android

在 Android 平台上,Valo Reader 使用 Google ML Kit

备注

想要查看 Valo Reader 手机版依赖的其他第三方组件,请打开“设置”,滚动至底部,点击“License”

- + \ No newline at end of file diff --git a/docs/settings/index.html b/docs/settings/index.html index 61b0fb9..f6e1ed2 100644 --- a/docs/settings/index.html +++ b/docs/settings/index.html @@ -5,13 +5,13 @@ 设置 | Valo Reader - +

设置

macOS

当前 Valo Reader for macOS 提供了一定程度的自定义设置, 以满足不同使用习惯的用户

请点击菜单栏上的图标, 打开 Valo Reader for macOS 的控制面板

Android / iOS

当前 Valo Reader mobile 提供了一定程度的自定义设置,以满足不同英语水品和偏好的用户。

想要了解如何在 iOS 或 Android 上对 Valo Reader 进行自定义设置,请打开 Valo Reader,点击“设置”。

- + \ No newline at end of file diff --git a/docs/usage/index.html b/docs/usage/index.html index 2f911eb..63581b2 100644 --- a/docs/usage/index.html +++ b/docs/usage/index.html @@ -5,13 +5,13 @@ 使用 | Valo Reader - +

使用

Valo Reader 在第一次运行时会自动展示使用帮助

macOS

想要重新了解如何在 macOS 上使用 Valo Reader。请点击菜单栏上的 Valo Reader 托盘图标, 点击“使用帮助”

Android / iOS

想要重新了解如何在 iOS 或 Android 上使用 Valo Reader。请打开 Valo Reader,点击“如何使用”。

- + \ No newline at end of file diff --git a/index.html b/index.html index 890d316..5e3c26c 100644 --- a/index.html +++ b/index.html @@ -5,13 +5,13 @@ Hello from Valo Reader | Valo Reader - +
- + \ No newline at end of file diff --git a/markdown-page/index.html b/markdown-page/index.html index c58ea56..6c4c423 100644 --- a/markdown-page/index.html +++ b/markdown-page/index.html @@ -5,13 +5,13 @@ Markdown page example | Valo Reader - +

Markdown page example

You don't need React to write simple standalone pages.

- + \ No newline at end of file