{"id":428678,"date":"2024-08-08T21:00:37","date_gmt":"2024-08-08T21:00:37","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=428678"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=428678","title":{"rendered":"<span>\u041f\u0438\u0448\u0435\u043c \u0441\u0432\u043e\u0451 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0434\u043b\u044f \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 PIN \u043d\u0430 \u0434\u0440\u0443\u0433\u0438\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f<\/span>"},"content":{"rendered":"<div><!--[--><!--]--><\/div>\n<div id=\"post-content-body\">\n<div>\n<div class=\"article-formatted-body article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<h2>\u041f\u0440\u0435\u0434\u044b\u0441\u0442\u043e\u0440\u0438\u044f<\/h2>\n<p>\u0415\u0449\u0451 \u0441\u00a0\u0434\u0435\u0442\u0441\u0442\u0432\u0430 \u043c\u043e\u0439 \u043e\u0442\u0435\u0446 \u043f\u0440\u0438\u0443\u0447\u0438\u043b \u043c\u0435\u043d\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0430\u043d\u0442\u0438\u0432\u0438\u0440\u0443\u0441\u0430\u043c\u0438. \u0421\u043e\u0431\u043b\u044e\u0434\u0430\u044f \u0442\u0440\u0430\u0434\u0438\u0446\u0438\u0438, \u044f\u00a0\u043a\u0443\u043f\u0438\u043b \u0441\u0435\u0431\u0435 \u043f\u043e\u0434\u043f\u0438\u0441\u043a\u0443 \u043d\u0430\u00a0\u0430\u043d\u0442\u0438\u0432\u0438\u0440\u0443\u0441 \u0434\u043b\u044f \u0410\u043d\u0434\u0440\u043e\u0438\u0434\u0430. \u041e\u043a\u0430\u0437\u0430\u043b\u043e\u0441\u044c, \u0432\u00a0\u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0438 \u0435\u0441\u0442\u044c \u043a\u0440\u0430\u0439\u043d\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u0430\u044f \u0444\u0438\u0447\u0430\u00a0\u2014 \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u041f\u0418\u041d-\u043a\u043e\u0434\u0430 \u0434\u043b\u044f \u0434\u0440\u0443\u0433\u0438\u0445 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u043d\u0430\u00a0\u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0435. \u0418\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0439 \u043e\u043d\u0430 \u0431\u044b\u043b\u0430 \u0434\u043b\u044f \u043c\u0435\u043d\u044f \u0442\u0435\u043c, \u0447\u0442\u043e\u00a0\u044f, \u043a\u0430\u043a \u043c\u043e\u0431\u0438\u043b\u044c\u043d\u044b\u0439 \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a, \u043d\u0435\u00a0\u0438\u043c\u0435\u043b \u043d\u0438\u00a0\u043c\u0430\u043b\u0435\u0439\u0448\u0435\u0433\u043e \u043f\u043e\u043d\u044f\u0442\u0438\u044f, \u043a\u0430\u043a \u043f\u043e\u0434\u043e\u0431\u043d\u043e\u0435 \u043c\u043e\u0436\u043d\u043e \u0441\u0434\u0435\u043b\u0430\u0442\u044c. \u0418\u00a0\u0432\u043e\u0442 \u0442\u0435\u043f\u0435\u0440\u044c, \u043f\u043e\u0441\u043b\u0435 \u043d\u0435\u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0445 \u0440\u0430\u0441\u043a\u043e\u043f\u043e\u043a \u0438\u00a0\u043f\u0440\u043e\u0434\u0435\u043b\u0430\u043d\u043d\u043e\u0439 \u0440\u0430\u0431\u043e\u0442\u044b, \u044f\u00a0\u0434\u0435\u043b\u044e\u0441\u044c \u0441\u0432\u043e\u0438\u043c \u043e\u043f\u044b\u0442\u043e\u043c.<\/p>\n<h2>\u041f\u043b\u0430\u043d<\/h2>\n<p>\u041f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0434\u043e\u043b\u0436\u043d\u043e \u0443\u043c\u0435\u0442\u044c:<\/p>\n<ol>\n<li>\n<p>\u0420\u0430\u0441\u043f\u043e\u0437\u043d\u0430\u0432\u0430\u0442\u044c, \u043a\u043e\u0433\u0434\u0430 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u041f\u0418\u041d-\u043a\u043e\u0434\u043e\u043c;<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434 \u043f\u0440\u0438 \u043e\u0442\u043a\u0440\u044b\u0442\u0438\u0438 \u0441\u0430\u043c\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f (\u0432\u00a0\u0440\u0430\u043c\u043a\u0430\u0445 \u00ab\u0441\u0430\u043c\u043e\u0437\u0430\u0449\u0438\u0442\u044b\u00bb);<\/p>\n<\/li>\n<li>\n<p>\u0412\u044b\u0431\u0438\u0440\u0430\u0442\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u0434\u043b\u044f \u0431\u043b\u043e\u043a\u0430;<\/p>\n<\/li>\n<li>\n<p>\u0421\u043e\u0437\u0434\u0430\u0432\u0430\u0442\u044c\/\u043c\u0435\u043d\u044f\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434;<\/p>\n<\/li>\n<li>\n<p>\u0417\u0430\u0449\u0438\u0449\u0430\u0442\u044c \u0441\u0435\u0431\u044f \u043e\u0442\u00a0\u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f.<\/p>\n<\/li>\n<\/ol>\n<h2>\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u0438\u0435 \u0442\u0435\u043a\u0443\u0449\u0435\u0433\u043e \u0437\u0430\u043f\u0443\u0449\u0435\u043d\u043d\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f<\/h2>\n<p>\u041f\u0435\u0440\u0432\u044b\u043c \u0434\u0435\u043b\u043e\u043c \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0431\u044b\u043b\u043e \u0440\u0435\u0448\u0438\u0442\u044c \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0443\u00a0\u2014 \u00ab\u043a\u0430\u043a \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c, \u0447\u0442\u043e \u043e\u0442\u043a\u0440\u044b\u043b\u043e\u0441\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043d\u0443\u0436\u043d\u043e \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u0430\u0442\u044c?\u00bb.<\/p>\n<p>\u0421\u043d\u0430\u0447\u0430\u043b\u0430 \u043c\u043e\u0439 \u0432\u0437\u0433\u043b\u044f\u0434 \u043f\u0430\u043b \u043d\u0430\u00a0\u0432\u0430\u0440\u0438\u0430\u043d\u0442 \u0441\u00a0\u043e\u0442\u043b\u043e\u0432\u043e\u043c \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0445 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439 \u0447\u0435\u0440\u0435\u0437 <em>BroadcastReceiver<\/em>.<br \/> \u041a\u00a0\u0441\u043e\u0436\u0430\u043b\u0435\u043d\u0438\u044e, \u043e\u043a\u0430\u0437\u0430\u043b\u043e\u0441\u044c \u0447\u0442\u043e \u043d\u0435\u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e \u043e\u0442\u043b\u043e\u0432\u0438\u0442\u044c \u0438\u043d\u0442\u0435\u043d\u0442 \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f.<\/p>\n<p>\u0412\u0442\u043e\u0440\u044b\u043c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u044b\u043c \u0440\u0435\u0448\u0435\u043d\u0438\u0435\u043c \u0431\u044b\u043b\u043e \u0440\u0435\u0433\u0443\u043b\u044f\u0440\u043d\u043e (\u0440\u0430\u0437 \u0432\u00a0\u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0441\u0435\u043a\u0443\u043d\u0434) \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0442\u044c \u0437\u0430\u043f\u0443\u0449\u0435\u043d\u043d\u044b\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f, \u0447\u0435\u0440\u0435\u0437 <em>ActivityManager.runningAppProcesses()<\/em>.<br \/> \u041d\u043e, \u043d\u0430\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u044f\u00a0\u043f\u043e\u043d\u0438\u043c\u0430\u044e, \u0441\u0435\u0439\u0447\u0430\u0441 \u0434\u0430\u043d\u043d\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u044b \u0442\u0435\u043a\u0443\u0449\u0435\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f, \u0447\u0442\u043e \u043d\u0430\u043c \u043d\u0435\u00a0\u043f\u043e\u0434\u0445\u043e\u0434\u0438\u0442.<\/p>\n<p>\u0427\u0442\u043e\u00a0\u0436, \u043e\u0441\u0442\u0430\u0451\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e&#8230;\u0441\u0442\u0430\u0440\u044b\u0439 \u0434\u043e\u0431\u0440\u044b\u0439 <em>AccessibilityService<\/em><\/p>\n<p><em>AccessibilityService<\/em>\u00a0\u2014 \u043e\u0447\u0435\u043d\u044c \u043c\u043e\u0449\u043d\u044b\u0439 \u0438\u00a0\u043e\u0434\u043d\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u043e \u043e\u043f\u0430\u0441\u043d\u044b\u0439 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442. \u0415\u0433\u043e \u00ab\u043f\u043e\u043b\u0435\u0437\u043d\u044b\u0435 \u0444\u0443\u043d\u043a\u0446\u0438\u0438\u00bb (\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u0438\u0435 \u0438\u00a0\u043e\u0437\u0432\u0443\u0447\u0438\u0432\u0430\u043d\u0438\u0435 \u0442\u0435\u043a\u0441\u0442\u0430 \u043d\u0430\u00a0\u044d\u043a\u0440\u0430\u043d\u0435, \u043d\u0430\u0436\u0430\u0442\u0438\u0435 \u043d\u0430\u00a0\u043a\u043d\u043e\u043f\u043a\u0438, \u0441\u0432\u0430\u0439\u043f\u044b \u0438\u00a0\u0442.\u0434.) \u0434\u043b\u044f \u043b\u044e\u0434\u0435\u0439 \u0441\u00a0\u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u043d\u044b\u043c\u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044f\u043c\u0438 \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432\u00a0\u0441\u0430\u043c\u044b\u0445 \u0443\u0436\u0430\u0441\u043d\u044b\u0445 \u0446\u0435\u043b\u044f\u0445, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0432\u00a0\u043d\u0430\u0448\u0438\u0445. \u0422\u0430\u043a \u0438\u00a0\u043f\u043e\u0441\u0442\u0443\u043f\u0438\u043c.<\/p>\n<h2>\u041d\u0430\u0447\u0430\u043b\u044c\u043d\u043e\u0435 \u0440\u0435\u0448\u0435\u043d\u0438\u0435<\/h2>\n<p>\u0414\u043b\u044f \u043d\u0430\u0447\u0430\u043b\u0430 \u043c\u044b\u00a0\u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0432\u00a0\u043c\u0430\u043d\u0438\u0444\u0435\u0441\u0442 \u043d\u0430\u0448 \u0441\u0435\u0440\u0432\u0438\u0441 \u0438\u00a0\u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u044b\u0435 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u044f:<\/p>\n<p>AndroidManifest.xml<\/p>\n<pre><code class=\"xml\">   &lt;uses-permission        android:name=\"android.permission.BIND_ACCESSIBILITY_SERVICE\"        tools:ignore=\"ProtectedPermissions\" \/&gt;    &lt;uses-permission        android:name=\"android.permission.QUERY_ALL_PACKAGES\"        tools:ignore=\"QueryAllPackagesPermission\" \/&gt;    &lt;uses-permission android:name=\"android.permission.FOREGROUND_SERVICE\" \/&gt;    &lt;uses-permission android:name=\"android.permission.FOREGROUND_SERVICE_SPECIAL_USE\" \/&gt;    &lt;uses-permission android:name=\"android.permission.POST_NOTIFICATIONS\" \/&gt;   ***        &lt;service            android:name=\".service.PinAccessibilityService\"            android:exported=\"true\"            android:foregroundServiceType=\"specialUse\"            android:permission=\"android.permission.BIND_ACCESSIBILITY_SERVICE\"&gt;            &lt;property                android:name=\"android.app.PROPERTY_SPECIAL_USE_FGS_SUBTYPE\"                android:value=\"pin_code_for_another_apps\" \/&gt;             &lt;intent-filter&gt;                &lt;action android:name=\"android.accessibilityservice.AccessibilityService\" \/&gt;            &lt;\/intent-filter&gt;             &lt;meta-data                android:name=\"android.accessibilityservice\"                android:resource=\"@xml\/accessibilityservice\" \/&gt;        &lt;\/service&gt; <\/code><\/pre>\n<p>\u041c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0435 \u0434\u043b\u044f <em>PinAccessibilityService<\/em>.<br \/> accessibilityservice.xml<\/p>\n<pre><code class=\"xml\">&lt;?xml version=\"1.0\" encoding=\"utf-8\"?&gt; &lt;accessibility-service xmlns:android=\"http:\/\/schemas.android.com\/apk\/res\/android\"    android:accessibilityEventTypes=\"typeAllMask\"    android:canRequestEnhancedWebAccessibility=\"true\"    android:notificationTimeout=\"100\"    android:accessibilityFeedbackType=\"feedbackGeneric\" android:accessibilityFlags=\"flagRetrieveInteractiveWindows|flagIncludeNotImportantViews\"    android:canRetrieveWindowContent=\"true\"\/&gt; <\/code><\/pre>\n<p>\u041a\u0430\u043a\u0438\u0435 \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u044b\u0435 \u0437\u0430\u0434\u0430\u0447\u0438 \u0434\u043e\u043b\u0436\u0435\u043d \u0440\u0435\u0448\u0430\u0442\u044c \u043d\u0430\u0448 \u0441\u0435\u0440\u0432\u0438\u0441:<\/p>\n<ul>\n<li>\n<p>\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0442\u044c, \u0447\u0442\u043e \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u043c\u0438 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u0449\u0438\u0442\u0438\u0442\u044c;<\/p>\n<\/li>\n<li>\n<p>\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0442\u044c, \u0447\u0442\u043e \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u043c\u0438 \u043d\u0430\u0448\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u0447\u0442\u043e\u0431\u044b \u043e\u043d\u043e \u0437\u0430\u0449\u0438\u0449\u0430\u043b\u043e \u0441\u0430\u043c\u043e \u0441\u0435\u0431\u044f;<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u0432\u0432\u043e\u0434\u043e\u043c \u041f\u0418\u041d-\u043a\u043e\u0434\u0430.<\/p>\n<\/li>\n<\/ul>\n<p>\u041f\u0440\u0438\u0441\u0442\u0443\u043f\u0438\u043c \u043a\u00a0\u043d\u0430\u043f\u0438\u0441\u0430\u043d\u0438\u044e \u0441\u0430\u043c\u043e\u0433\u043e <em>AccessibilityService<\/em>.<\/p>\n<p>\u041f\u0440\u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0435 \u0441\u0435\u0440\u0432\u0438\u0441\u0430 \u043c\u044b\u00a0\u0437\u0430\u043a\u0440\u0435\u043f\u043b\u044f\u0435\u043c \u0443\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u0435, \u0447\u0442\u043e\u0431\u044b \u0441\u043b\u0435\u0434\u0438\u0442\u044c, \u0436\u0438\u0432\u00a0\u043b\u0438 \u043d\u0430\u0448 \u0441\u0435\u0440\u0432\u0438\u0441 (<em>AccessibilityService<\/em> \u043c\u043e\u0436\u0435\u0442 \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0432\u00a0\u0444\u043e\u043d\u0435 \u0431\u0435\u0437 \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u043a\u0430\u043a <em>ForegroundService<\/em>).<\/p>\n<p>PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">class PinAccessibilityService : AccessibilityService() {     override fun onCreate() {        super.onCreate()        startForeground()    }     private fun startForeground() {        val channelId = createNotificationChannel()         val notification = NotificationCompat.Builder(this, channelId)            .setContentTitle(\"Pin On App\")            .setContentText(\"Apps protected\")            .setOngoing(true)            .setSmallIcon(R.mipmap.ic_launcher)            .setPriority(PRIORITY_HIGH)            .setCategory(Notification.CATEGORY_SERVICE)            .build()        startForeground(101, notification)    }     private fun createNotificationChannel(): String {        val channelId = \"pin_on_app_service\"        val channelName = \"PinOnApp\"        val channel = NotificationChannel(            channelId,            channelName,            NotificationManager.IMPORTANCE_HIGH,        ).apply {            lightColor = Color.BLUE            lockscreenVisibility = Notification.VISIBILITY_PRIVATE        }         val service = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager        service.createNotificationChannel(channel)         return channelId    } } <\/code><\/pre>\n<p>\u0414\u0430\u043b\u0435\u0435, \u043c\u044b\u00a0\u043f\u044b\u0442\u0430\u0435\u043c\u0441\u044f \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c, \u0447\u0442\u043e \u0437\u0430\u00a0\u044d\u043a\u0440\u0430\u043d \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u043c\u0438\u00a0\u0438, \u0432\u00a0\u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0442\u00a0\u044d\u0442\u043e\u0433\u043e, \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u0435\u043c \u0438\u043b\u0438 \u043d\u0435\u00a0\u0431\u043b\u043e\u043a\u0438\u0440\u0443\u0435\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f.<\/p>\n<p>\u041b\u043e\u0433\u0438\u043a\u0430 \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0430\u044f:<\/p>\n<ol>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f <em>Launcher<\/em>\u00a0\u2014 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0431\u044b\u043b \u043d\u0430\u00a0\u043e\u0441\u043d\u043e\u0432\u043d\u043e\u043c \u044d\u043a\u0440\u0430\u043d\u0435, \u0430\u00a0\u0437\u043d\u0430\u0447\u0438\u0442 \u043f\u0440\u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0435 \u0437\u0430\u0449\u0438\u0449\u0435\u043d\u043d\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u0432\u0432\u043e\u0434\u043e\u043c \u041f\u0418\u041d-\u043a\u043e\u0434\u0430.<\/p>\n<\/li>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043d\u0443\u0436\u043d\u043e \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u0430\u0442\u044c, \u0438\u00a0\u041f\u0418\u041d \u043d\u0435\u00a0\u0431\u044b\u043b \u0432\u0432\u0435\u0434\u0435\u043d\u00a0\u2014 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u0432\u0432\u043e\u0434\u043e\u043c \u041f\u0418\u041d-\u043a\u043e\u0434\u0430. \u041d\u0443\u0436\u043d\u043e\u0435 \u043d\u0430\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u043c \u043f\u043e\u00a0\u0435\u0433\u043e <em>packageName<\/em>.<\/p>\n<\/li>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u043f\u043e\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0437\u0430\u0449\u0438\u0442\u0443 \u043d\u0430\u00a0\u043d\u0430\u0448\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u0442\u043e\u00a0\u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u043f\u043e\u043f\u0430\u0434\u0430\u0442\u044c \u0432\u00a0\u0446\u0438\u043a\u043b \u0438\u00a0\u0431\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u043e \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0432\u0432\u043e\u0434\u0430 \u041f\u0418\u041d\u0430, \u0442.\u043a.\u00a0\u043c\u044b\u00a0\u043e\u0440\u0438\u0435\u043d\u0442\u0438\u0440\u0443\u0435\u043c\u0441\u044f \u043d\u0430\u00a0<em>packageName<\/em> \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439, \u0430\u00a0<em>packageName<\/em> \u0443\u00a0\u044d\u043a\u0440\u0430\u043d\u0430 \u0431\u043b\u043e\u043a\u0430 \u0438\u00a0\u043d\u0430\u0448\u0435\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u043e\u0434\u0438\u043d\u0430\u043a\u043e\u0432\u044b\u0439. \u0414\u043b\u044f \u0440\u0435\u0448\u0435\u043d\u0438\u044f \u044d\u0442\u043e\u0439 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u043c\u044b\u00a0\u0437\u0430\u0432\u044f\u0436\u0435\u043c\u0441\u044f \u043d\u0430\u00a0\u0443\u043d\u0438\u043a\u0430\u043b\u044c\u043d\u044b\u0439 \u0438\u0434\u0435\u043d\u0442\u0438\u0444\u0438\u043a\u0430\u0442\u043e\u0440 \u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0433\u043e \u044d\u043a\u0440\u0430\u043d\u0430.<\/p>\n<\/li>\n<\/ol>\n<p><em>Launcher-\u044b<\/em> \u043d\u0430\u00a0\u0440\u0430\u0437\u043d\u044b\u0445\u00a0\u041e\u0421 \u0440\u0430\u0437\u043d\u044b\u0435, \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u043c\u044b\u00a0\u0437\u0430\u0440\u0430\u043d\u0435\u0435 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0441\u043f\u0438\u0441\u043e\u043a \u0432\u0441\u0435\u0445 <em>Launcher-\u043e\u0432<\/em> \u043d\u0430\u00a0\u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0435.<\/p>\n<p>PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">   private val scope = CoroutineScope(Dispatchers.Main + SupervisorJob())    private val event = MutableSharedFlow&lt;AccessibilityEvent&gt;(       extraBufferCapacity = 1,       onBufferOverflow = BufferOverflow.DROP_OLDEST,    )     private val spRepository by inject&lt;SharedPreferencesRepository&gt;()    private var packageForPinList = spRepository.packageIdList     private var launcherList = listOf&lt;String&gt;()     private val isPinCodeNotExist        get() = spRepository.pinCode.isNullOrEmpty()    private val isCorrectPin        get() = spRepository.isCorrectPin == true     init {        event            .filter { !isPinCodeNotExist &amp;&amp; !isCorrectPin }            .onEach { _ -&gt;                spRepository.isCorrectPin = false                 val startActivityIntent = Intent(applicationContext, ConfirmActivity::class.java)                    .apply {                        setFlags(                            Intent.FLAG_ACTIVITY_NEW_TASK                                    or Intent.FLAG_ACTIVITY_CLEAR_TASK                                    or Intent.FLAG_ACTIVITY_NO_ANIMATION                                    or Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS,                        )                    }                startActivity(startActivityIntent)            }.catch {                Log.e(\"ERROR\", it.message, it)            }.launchIn(scope)    }     override fun onCreate() {        super.onCreate()        startForeground()        setLauncherListOnDevice()    }     private fun setLauncherListOnDevice() {        val i = Intent(Intent.ACTION_MAIN).apply {            addCategory(Intent.CATEGORY_HOME)        }        launcherList = packageManager.queryIntentActivities(i, 0).map {            it.activityInfo.packageName        }    }     override fun onAccessibilityEvent(event: AccessibilityEvent?) {        if (event != null) {            if (launcherList.contains(event.packageName)) {                spRepository.isCorrectPin = false            } else if (packageForPinList.contains(event.packageName) || event.isMainActivityShowed()) {                this.event.tryEmit(event)            }        }    }     private fun AccessibilityEvent.isMainActivityShowed() =        className?.contains(APP_CLASS_NAME) ?: false     override fun onInterrupt() {}     override fun onDestroy() {        scope.cancel()        super.onDestroy()    }     companion object {        private const val APP_CLASS_NAME = \"com.dradefire.securepinonapp.ui.main.MainActivity\"    } <\/code><\/pre>\n<p>\u0421\u043f\u0438\u0441\u043e\u043a \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0442\u044c \u0432\u00a0<em>SharedPreferences<\/em>, \u0434\u043b\u044f \u043a\u043e\u0442\u043e\u0440\u044b\u0445 \u043d\u0430\u043f\u0438\u0448\u0435\u043c \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0439 (\u0434\u043b\u044f \u0432\u043d\u0435\u0434\u0440\u0435\u043d\u0438\u044f \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439 \u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c Koin).<\/p>\n<p>KoinModule.kt<\/p>\n<pre><code class=\"kotlin\">val KoinModule = module {    viewModelOf(::ConfirmViewModel)    viewModelOf(::MainViewModel)     factoryOf(::SharedPreferencesRepository)     singleOf(::Gson) } <\/code><\/pre>\n<p>App.kt<\/p>\n<pre><code class=\"kotlin\">class App : Application() {    override fun onCreate() {        super.onCreate()         startKoin {            androidContext(this@App)            modules(KoinModule)        }    } } <\/code><\/pre>\n<p>SharedPreferencesRepository.kt<\/p>\n<pre><code class=\"kotlin\">class SharedPreferencesRepository(    private val context: Context,    private val gson: Gson, ) {    \/**    * \u0411\u044b\u043b \u043b\u0438 \u0432\u0432\u0435\u0434\u0451\u043d \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u044b\u0439 \u041f\u0418\u041d-\u043a\u043e\u0434 (\u043d\u0443\u0436\u043d\u043e, \u0447\u0442\u043e\u0431\u044b \u043b\u0438\u0448\u043d\u0438\u0439 \u0440\u0430\u0437 \u043d\u0435 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438)    *\/    var isCorrectPin: Boolean?         get() = context.sp.getBoolean(IS_CORRECT_PIN_KEY, false)        set(isCorrectPin) {            context.sp.edit().putBoolean(IS_CORRECT_PIN_KEY, isCorrectPin ?: false).apply()        }     \/**    * \u041f\u0418\u041d-\u043a\u043e\u0434    *\/    var pinCode: String?        get() = context.sp.getString(PIN_KEY, null)        set(pinCode) {            context.sp.edit().putString(PIN_KEY, pinCode).apply()        }    \/**    * \u0421\u043f\u0438\u0441\u043e\u043a \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u0449\u0438\u0442\u0438\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434\u043e\u043c    *\/    var packageIdList: List&lt;String&gt;         get() = gson.fromJson(            context.sp.getString(                PACKAGE_ID_LIST_KEY,                gson.toJson(emptyList&lt;String&gt;()),            ),            List::class.java,        ) as List&lt;String&gt;        set(list) {            context.sp.edit().putString(PACKAGE_ID_LIST_KEY, gson.toJson(list)).apply()        }     private val Context.sp        get() = getSharedPreferences(SECURE_PIN_ON_APP_STORAGE, Context.MODE_PRIVATE)     companion object {        private const val PACKAGE_ID_LIST_KEY = \"PACKAGE_ID_LIST\"        private const val PIN_KEY = \"PIN\"        private const val IS_CORRECT_PIN_KEY = \"IS_CORRECT_PIN\"        private const val SECURE_PIN_ON_APP_STORAGE = \"secure_pin_on_app_storage\"    } } <\/code><\/pre>\n<h2>Corner case \u0441\u00a0\u0443\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u044f\u043c\u0438<\/h2>\n<p>\u0412\u00a0\u043f\u0440\u043e\u0446\u0435\u0441\u0441\u0435 \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u043d\u0430\u0448\u0435\u0433\u043e \u0441\u0435\u0440\u0432\u0438\u0441\u0430 \u043d\u0430\u00a0\u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0438 \u00ab\u0421\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f\u00bb \u044f\u00a0\u043e\u0431\u043d\u0430\u0440\u0443\u0436\u0438\u043b, \u0447\u0442\u043e \u043a\u043e\u0433\u0434\u0430 \u043f\u0440\u0438\u0445\u043e\u0434\u044f\u0442 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f (\u0430\u00a0\u0432\u00a0\u044d\u0442\u043e\u0442 \u043c\u043e\u043c\u0435\u043d\u0442 \u043f\u043e\u044f\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0443\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u0435), \u0442\u043e\u00a0\u043d\u0435\u043e\u0436\u0438\u0434\u0430\u043d\u043d\u043e \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u044d\u043a\u0440\u0430\u043d \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438.<\/p>\n<p>\u0414\u043b\u044f \u0440\u0435\u0448\u0435\u043d\u0438\u044f \u044d\u0442\u043e\u0439 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u043c\u044b\u00a0\u043f\u0440\u043e\u0441\u0442\u043e \u0431\u0443\u0434\u0435\u043c \u0438\u0433\u043d\u043e\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c <em>TYPE_NOTIFICATION_STATE_CHANGED<\/em>.<\/p>\n<p>PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">   init {        event            .filter { !isPinCodeNotExist &amp;&amp; !isCorrectPin &amp;&amp; it.eventType != TYPE_NOTIFICATION_STATE_CHANGED }            .onEach { event -&gt;               ***            }.catch {                Log.e(\"ERROR\", it.message, it)            }.launchIn(scope)    } <\/code><\/pre>\n<h2>\u0427\u0443\u0442\u044c-\u0447\u0443\u0442\u044c \u0443\u043b\u0443\u0447\u0448\u0438\u043c UX<\/h2>\n<p>\u0410\u00a0\u0442\u043e\u0447\u043d\u0435\u0435\u00a0\u2014 \u043a\u043e\u0435-\u0447\u0442\u043e \u043f\u043e\u043b\u043d\u043e\u0441\u0442\u044c\u044e \u043f\u0435\u0440\u0435\u043f\u0438\u0448\u0435\u043c.<\/p>\n<p>\u041a\u0430\u0436\u0434\u044b\u0439 \u0440\u0430\u0437 \u043f\u0440\u0438 \u043e\u0442\u043a\u0440\u044b\u0442\u0438\u0438 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e \u043c\u043e\u0436\u0435\u0442 \u043d\u0430\u0434\u043e\u0435\u0441\u0442\u044c \u0432\u0432\u043e\u0434\u0438\u0442\u044c \u041f\u0418\u041d: \u0441\u0432\u0435\u0440\u043d\u0443\u043b \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u0432\u044b\u0448\u0435\u043b \u043d\u0430\u00a0\u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u044d\u043a\u0440\u0430\u043d \u0438\u00a0\u043e\u0431\u0440\u0430\u0442\u043d\u043e \u0438\u00a0\u0442.\u043f.<br \/> \u041c\u044b\u00a0\u0434\u0430\u0434\u0438\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e \u0432\u0437\u0434\u043e\u0445\u043d\u0443\u0442\u044c \u0441\u0432\u043e\u0431\u043e\u0434\u043d\u0435\u0435: \u0442\u0435\u043f\u0435\u0440\u044c, \u0435\u0441\u043b\u0438 \u043e\u043d\u00a0\u0432\u0432\u0435\u043b \u041f\u0418\u041d \u0434\u043b\u044f \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u0435\u043c\u0443 \u043d\u0435\u00a0\u043f\u0440\u0438\u0434\u0435\u0442\u0441\u044f \u0432\u0432\u043e\u0434\u0438\u0442\u044c \u041f\u0418\u041d, \u043f\u043e\u043a\u0430 \u043e\u043d\u00a0\u043d\u0435\u00a0\u043e\u0442\u043a\u0440\u043e\u0435\u0442 \u0434\u0440\u0443\u0433\u043e\u0435 \u0437\u0430\u0449\u0438\u0449\u0435\u043d\u043d\u043e\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<p>\u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u043c\u044b\u00a0\u0438\u0437\u0431\u0430\u0432\u043b\u044f\u0435\u043c\u0441\u044f \u043e\u0442\u00a0\u0437\u0430\u0432\u044f\u0437\u043a\u0438 \u043d\u0430\u00a0<em>Launcher-\u044b<\/em>.<\/p>\n<p>PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">   private var lastPinnedAppPackageName: String? = null     init {        event            .filter { event -&gt;                !isPinCodeNotExist &amp;&amp; !isCorrectPin &amp;&amp;                        event.eventType != TYPE_NOTIFICATION_STATE_CHANGED || event.packageName != lastPinnedAppPackageName            }            .onEach { event -&gt;                spRepository.isCorrectPin = false                lastPinnedAppPackageName = event.packageName.toString()                  val startActivityIntent = Intent(this, ConfirmActivity::class.java)                    .apply {                        setFlags(                            Intent.FLAG_ACTIVITY_NEW_TASK                                    or Intent.FLAG_ACTIVITY_CLEAR_TASK                                    or Intent.FLAG_ACTIVITY_NO_ANIMATION                                    or Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS,                        )                    }                startActivity(startActivityIntent)            }.catch {                Log.e(\"ERROR\", it.message, it)            }.launchIn(scope)    }     override fun onCreate() {        super.onCreate()        startForeground()    }     override fun onAccessibilityEvent(event: AccessibilityEvent?) {        if (event != null) {            if (packageForPinList.contains(event.packageName) || event.isMainActivityShowed()) {                this.event.tryEmit(event)            }        }    } <\/code><\/pre>\n<h2>\u041e\u0441\u043d\u043e\u0432\u043e\u0439 \u044d\u043a\u0440\u0430\u043d<\/h2>\n<p>\u0423\u00a0\u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0433\u043e \u044d\u043a\u0440\u0430\u043d\u0430 \u0435\u0441\u0442\u044c \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0437\u0430\u0434\u0430\u0447:<\/p>\n<ol>\n<li>\n<p>\u0417\u0430\u043f\u0440\u043e\u0441\u0438\u0442\u044c \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u044f (\u043d\u0430\u00a0\u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0443 \u0443\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u0439 \u0438\u00a0\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u043d\u0430\u0448\u0435\u0433\u043e \u0441\u0435\u0440\u0432\u0438\u0441\u0430);<\/p>\n<\/li>\n<li>\n<p>\u0414\u0430\u0442\u044c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0438\u00a0\u0443\u0431\u0440\u0430\u0442\u044c \u0431\u043b\u043e\u043a \u0441\u00a0\u043b\u044e\u0431\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u043d\u0430\u00a0\u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0435;<\/p>\n<\/li>\n<li>\n<p>\u0414\u0430\u0442\u044c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0441\u043c\u0435\u043d\u0438\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434.<\/p>\n<\/li>\n<\/ol>\n<p>MainActivity.kt<\/p>\n<pre><code class=\"kotlin\">class MainActivity : ComponentActivity() {    private val spRepository by inject&lt;SharedPreferencesRepository&gt;()    private val viewModel by viewModel&lt;MainViewModel&gt;()     override fun onRequestPermissionsResult(        requestCode: Int,        permissions: Array&lt;String&gt;,        grantResults: IntArray    ) {        checkPermission()        super.onRequestPermissionsResult(requestCode, permissions, grantResults)    }     override fun onStart() {        checkPermission()        super.onStart()    }     override fun onCreate(savedInstanceState: Bundle?) {        super.onCreate(savedInstanceState)         \/\/ \u0415\u0441\u043b\u0438 \u041f\u0418\u041d-\u043a\u043e\u0434 \u043d\u0435 \u0441\u0443\u0449\u0435\u0441\u0442\u0432\u0443\u0435\u0442 - \u0434\u0430\u0435\u043c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0437\u0430\u0434\u0430\u0442\u044c \u0435\u0433\u043e        if (spRepository.pinCode == null) {            openConfirmActivityWithSettingPinCode()        }         \/\/ \u0414\u043e\u0441\u0442\u0430\u0451\u043c \u0441\u043f\u0438\u0441\u043e\u043a \u0432\u0441\u0435\u0445 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u043d\u0430 \u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0435        val intent = Intent(Intent.ACTION_MAIN).apply {            addCategory(Intent.CATEGORY_LAUNCHER)        }        val applicationList = packageManager.queryIntentActivities(            intent,            PackageManager.MATCH_ALL,        ).distinctBy { it.activityInfo.packageName }         val packageIdListInit = spRepository.packageIdList        val appInfoListInit = applicationList.mapNotNull {            val activityInfo = it.activityInfo             if (activityInfo.packageName == APP_PACKAGE_ID) { \/\/ \u0422\u0435\u043a\u0443\u0449\u0435\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u043d\u0435 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c                null            } else {                ApplicationInfo(                    icon = activityInfo.applicationInfo.loadIcon(packageManager)                        .toBitmap(),                    name = activityInfo.applicationInfo.loadLabel(packageManager)                        .toString(),                    packageId = activityInfo.packageName,                    isSecured = packageIdListInit.contains(activityInfo.packageName),                )            }        }         setContent {            MaterialTheme {                var appInfoList = remember {                    appInfoListInit.toMutableStateList()                }                 val isAccessibilityGranted by viewModel.isAccessibilityGranted.collectAsState()                val isNotificationGranted by viewModel.isNotificationGranted.collectAsState()                 if (!isAccessibilityGranted || !isNotificationGranted) {                    Dialog(onDismissRequest = {                        \/\/ block                    }) {                        Card(                            modifier = Modifier                                .fillMaxWidth()                                .padding(20.dp),                            shape = RoundedCornerShape(16.dp),                        ) {                            Text(                                text = \"\u041d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u044b \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u0435 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u044f:\",                                modifier = Modifier                                    .fillMaxWidth()                                    .padding(4.dp),                                textAlign = TextAlign.Center,                            )                             if (!isAccessibilityGranted) {                                OutlinedButton(                                    modifier = Modifier                                        .fillMaxWidth()                                        .padding(4.dp),                                    onClick = {  \/\/ \u0417\u0430\u043f\u0440\u0430\u0448\u0438\u0432\u0430\u0435\u043c \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u0435 \u043d\u0430 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0421\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u044b\u0445 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0435\u0439                                        val openSettings =                                            Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS)                                        openSettings.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_NO_HISTORY)                                        startActivity(openSettings)                                    },                                ) {                                    Text(text = \"\u0421\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u044b\u0435 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438\")                                }                            }                             if (!isNotificationGranted) {                                OutlinedButton(                                    modifier = Modifier                                        .fillMaxWidth()                                        .padding(4.dp),                                    onClick = {  \/\/ \u0417\u0430\u043f\u0440\u0430\u0448\u0438\u0432\u0430\u0435\u043c \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u0435 \u043d\u0430 \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0443\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u0439 (\u043d\u0443\u0436\u043d\u043e \u0437\u0430\u043f\u0440\u0430\u0448\u0438\u0432\u0430\u0442\u044c \u0441 33 API)                                        if (Build.VERSION.SDK_INT &gt;= Build.VERSION_CODES.TIRAMISU) {                                            ActivityCompat.requestPermissions(                                                this@MainActivity,                                                arrayOf(POST_NOTIFICATIONS),                                                1,                                            )                                        }                                    },                                ) {                                    Text(text = \"\u0423\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u044f\")                                }                            }                          }                    }                }                 Screen(                     ***                 )            }        }    }      \/**      * \u041f\u0440\u043e\u0432\u0435\u0440\u043a\u0430 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u044b\u0445 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u0439:      * 1. \u0421\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u044b\u0435 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 (AccessibilityService)      * 2. \u0423\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u044f      *\/    private fun checkPermission() {        val isAccessibilityGranted = isAccessibilityServiceEnabled()        viewModel.setAccessibilityPermission(isAccessibilityGranted)         val isNotificationGranted = if (Build.VERSION.SDK_INT &gt;= Build.VERSION_CODES.TIRAMISU) {            ContextCompat.checkSelfPermission(                this@MainActivity,                POST_NOTIFICATIONS,            ) == PackageManager.PERMISSION_GRANTED        } else {            true        }        viewModel.setNotificationPermission(isNotificationGranted)    }     private fun openConfirmActivityWithSettingPinCode() {        val startActivityIntent = Intent(applicationContext, ConfirmActivity::class.java)            .apply {                setFlags(                    Intent.FLAG_ACTIVITY_NEW_TASK                            or Intent.FLAG_ACTIVITY_CLEAR_TASK                            or Intent.FLAG_ACTIVITY_NO_ANIMATION                            or Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS,                )                putExtra(\"isSettingPinCode\", true) \/\/ \u0417\u0430\u0434\u0430\u0435\u043c \u043d\u043e\u0432\u044b\u0439 \u041f\u0418\u041d-\u043a\u043e\u0434            }        startActivity(startActivityIntent)    }     data class ApplicationInfo(        val icon: Bitmap,        val name: String,        val packageId: String,        val isSecured: Boolean,    ) } <\/code><\/pre>\n<p>AccessibilityServiceUtils.kt<\/p>\n<pre><code class=\"kotlin\">\/\/ Copied from https:\/\/mhrpatel12.medium.com\/android-accessibility-service-the-unexplored-goldmine-d336b0f33e30 fun Context.isAccessibilityServiceEnabled(): Boolean {     var accessibilityEnabled = 0     val service: String = packageName + \"\/\" + PinAccessibilityService::class.java.canonicalName     try {         accessibilityEnabled = Settings.Secure.getInt(             applicationContext.contentResolver,             Settings.Secure.ACCESSIBILITY_ENABLED,         )     } catch (e: SettingNotFoundException) {         Log.e(             \"ACCESSIBILITY_ENABLED_LOG\",             \"Error finding setting, default accessibility to not found: \" + e.message,         )     }     val mStringColonSplitter = SimpleStringSplitter(':')     if (accessibilityEnabled == 1) {         val settingValue: String = Settings.Secure.getString(             applicationContext.contentResolver,             Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES,         )         mStringColonSplitter.setString(settingValue)         while (mStringColonSplitter.hasNext()) {             val accessibilityService = mStringColonSplitter.next()              if (accessibilityService.equals(service, ignoreCase = true)) {                 return true             }         }     }     return false } <\/code><\/pre>\n<p>MainViewModel.kt<\/p>\n<pre><code class=\"kotlin\">class MainViewModel(    private val sharedPreferencesRepository: SharedPreferencesRepository, ) : ViewModel() {    private val _isAccessibilityGranted = MutableStateFlow(false)    val isAccessibilityGranted = _isAccessibilityGranted.asStateFlow()     private val _isNotificationGranted = MutableStateFlow(false)    val isNotificationGranted = _isNotificationGranted.asStateFlow()     fun setAccessibilityPermission(isGranted: Boolean) {        _isAccessibilityGranted.update { isGranted }    }     fun setNotificationPermission(isGranted: Boolean) {        _isNotificationGranted.update { isGranted }    }     fun onSwitchClick(packageId: String, checked: Boolean) {        val packageIdList = sharedPreferencesRepository.packageIdList.toMutableSet()         if (checked) {            packageIdList.add(packageId)        } else {            packageIdList.remove(packageId)        }         sharedPreferencesRepository.packageIdList = packageIdList.toList()    } } <\/code><\/pre>\n<p>\u0418\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044f \u043e\u00a0\u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f\u0445 \u0434\u043b\u044f \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438 \u0441\u0430\u043c\u0430 \u0441\u0435\u0431\u044f \u043d\u0435\u00a0\u043e\u0431\u043d\u043e\u0432\u0438\u0442, \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u0440\u0430\u0437 \u0432\u00a04\u00a0\u0441\u0435\u043a\u0443\u043d\u0434\u044b \u043d\u0430\u0448 \u0441\u0435\u0440\u0432\u0438\u0441 \u0431\u0443\u0434\u0435\u0442 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0442\u044c \u0438\u00a0\u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0442\u044c \u0441\u043f\u0438\u0441\u043e\u043a \u0432\u043e\u0442 \u0442\u0430\u043a\u0438\u043c \u043e\u0431\u0440\u0430\u0437\u043e\u043c.<\/p>\n<p>PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">   init {        scope.launch {            while (isActive) {                delay(4000)                packageForPinList = spRepository.packageIdList            }        } ***    } <\/code><\/pre>\n<p>\u0412\u00a0\u0438\u0442\u043e\u0433\u0435 \u043d\u0430\u0448 \u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u044d\u043a\u0440\u0430\u043d \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0432\u043e\u0442 \u0442\u0430\u043a: <\/p>\n<figure class=\"\"><img decoding=\"async\" src=\"https:\/\/imgur.com\/GC7COZ1.png\" alt=\"\u041e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u044d\u043a\u0440\u0430\u043d\" data-src=\"https:\/\/imgur.com\/GC7COZ1.png\"\/><\/p>\n<div><figcaption>\u041e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u044d\u043a\u0440\u0430\u043d<\/figcaption><\/div>\n<\/figure>\n<p>\u0410\u00a0\u044d\u0442\u043e \u0434\u0438\u0430\u043b\u043e\u0433 \u0437\u0430\u043f\u0440\u043e\u0441\u0430 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u0439 \u043d\u0430\u00a0\u043e\u0441\u043d\u043e\u0432\u043d\u043e\u043c \u044d\u043a\u0440\u0430\u043d\u0435: <\/p>\n<figure class=\"\"><img decoding=\"async\" src=\"https:\/\/imgur.com\/2Jew5nt.png\" alt=\"\u041e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u044d\u043a\u0440\u0430\u043d\" data-src=\"https:\/\/imgur.com\/2Jew5nt.png\"\/><\/p>\n<div><figcaption>\u041e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u044d\u043a\u0440\u0430\u043d<\/figcaption><\/div>\n<\/figure>\n<h2>\u042d\u043a\u0440\u0430\u043d \u0441\u00a0\u0432\u0432\u043e\u0434\u043e\u043c \u041f\u0418\u041d-\u043a\u043e\u0434\u0430<\/h2>\n<p>\u042d\u0442\u043e \u0442\u043e\u0442 \u044d\u043a\u0440\u0430\u043d \u043d\u0430\u00a0\u043a\u043e\u0442\u043e\u0440\u043e\u043c \u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f. \u041e\u043d\u00a0\u043d\u0435\u00a0\u043e\u0441\u043e\u0431\u043e \u0445\u0438\u0442\u0440\u044b\u0439: \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0432\u0432\u043e\u0434\u0438\u0442 4\u00a0\u0446\u0438\u0444\u0440\u044b \u0438\u00a0\u043b\u0438\u0431\u043e \u043f\u0440\u043e\u0445\u043e\u0434\u0438\u0442 \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0443, \u043b\u0438\u0431\u043e \u043f\u0440\u043e\u0431\u0443\u0435\u0442 \u0434\u0430\u043b\u044c\u0448\u0435. \u042d\u0442\u043e\u0442\u00a0\u0436\u0435 \u044d\u043a\u0440\u0430\u043d, \u0432\u00a0\u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0442\u00a0\u0441\u0438\u0442\u0443\u0430\u0446\u0438\u0438, \u0431\u0443\u0434\u0435\u0442 \u043e\u0436\u0438\u0434\u0430\u0442\u044c \u043a\u0430\u043a \u0432\u0432\u043e\u0434 \u0442\u0435\u043a\u0443\u0449\u0435\u0433\u043e \u041f\u0418\u041d-\u043a\u043e\u0434\u0430 \u0442\u0430\u043a \u0438\u00a0\u043d\u043e\u0432\u043e\u0433\u043e.<\/p>\n<p>ConfirmActivity.kt<\/p>\n<pre><code class=\"kotlin\">class ConfirmActivity : ComponentActivity() {     override fun onCreate(savedInstanceState: Bundle?) {        super.onCreate(savedInstanceState)         setContent {            val viewModel = koinViewModel&lt;ConfirmViewModel&gt;()            val pinCode by viewModel.pinCode.collectAsState()            val isSettingPinCode =                intent.getBooleanExtra(\"isSettingPinCode\", false) || viewModel.isPinCodeNotExist             LaunchedEffect(Unit) {                viewModel.closeActivityEvent.collect {                    finishAndRemoveTask()                }            }             BackHandler {                \/\/ block button            }             MaterialTheme {                Column(                    modifier = Modifier.fillMaxSize(),                    verticalArrangement = Arrangement.Center,                ) {                    BlockScreen(                        onButtonClick = {                            viewModel.onButtonClick(it, isSettingPinCode)                        },                        pinCodeLength = pinCode.length,                        isPinValid = viewModel.isPinValid,                        title = if (isSettingPinCode) \"Set PIN\" else \"Enter PIN\",                    )                }            }        }    }    *** } <\/code><\/pre>\n<p>ConfirmViewModel.kt<\/p>\n<pre><code class=\"kotlin\">class ConfirmViewModel(    private val sharedPreferencesRepository: SharedPreferencesRepository, ) : ViewModel() {    private val correctPinCode = sharedPreferencesRepository.pinCode    val isPinCodeNotExist = sharedPreferencesRepository.pinCode.isNullOrEmpty()     private val _closeActivityEvent = MutableSharedFlow&lt;Unit&gt;(        extraBufferCapacity = 1,        onBufferOverflow = BufferOverflow.DROP_OLDEST,    )    val closeActivityEvent = _closeActivityEvent.asSharedFlow()     private val _pinCode = MutableStateFlow(\"\")    val pinCode = _pinCode.asStateFlow()    val isPinValid        get() = _pinCode.value == correctPinCode     fun onButtonClick(event: ButtonClickEvent, isSettingPinCode: Boolean) {        when (event) {            is ButtonClickEvent.Number -&gt; {                _pinCode.update { it + event.number }                if (_pinCode.value.length &gt;= 4) {                    handleEnteredFullPinCode(isSettingPinCode)                }            }             ButtonClickEvent.Delete -&gt; {                if (_pinCode.value.isNotEmpty()) {                    _pinCode.update { it.substring(0, it.length - 1) }                }            }        }    }     private fun handleEnteredFullPinCode(isSettingPinCode: Boolean) {        if (isSettingPinCode) {            sharedPreferencesRepository.pinCode = _pinCode.value            onSuccessPinEnter()        } else if (isPinValid) {            onSuccessPinEnter()        } else {            _pinCode.update { \"\" }        }    }     private fun onSuccessPinEnter() {        sharedPreferencesRepository.isCorrectPin = true        _closeActivityEvent.tryEmit(Unit)    }     sealed interface ButtonClickEvent {        data class Number(val number: Int) : ButtonClickEvent        data object Delete : ButtonClickEvent    } } <\/code><\/pre>\n<p>\u0412\u043e\u0442 \u0442\u0430\u043a\u043e\u0439 \u044d\u043a\u0440\u0430\u043d \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438 \u0443\u00a0\u043d\u0430\u0441 \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0441\u044f: <\/p>\n<figure class=\"\"><img decoding=\"async\" src=\"https:\/\/imgur.com\/vuePxq6.png\" alt=\"\u042d\u043a\u0440\u0430\u043d \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438\" data-src=\"https:\/\/imgur.com\/vuePxq6.png\"\/><\/p>\n<div><figcaption>\u042d\u043a\u0440\u0430\u043d \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438<\/figcaption><\/div>\n<\/figure>\n<h2>\u041b\u0443\u0447\u0448\u0430\u044f \u0437\u0430\u0449\u0438\u0442\u0430\u00a0\u2014 \u044d\u0442\u043e \u043d\u0430\u043f\u0430\u0434\u0435\u043d\u0438\u0435 (\u043d\u0430\u00a0\u043f\u0440\u0430\u0432\u0430 \u0430\u0434\u043c\u0438\u043d\u0430, \u043d\u0435 \u043f\u0443\u0442\u0430\u0442\u044c \u0441 root)<\/h2>\n<p>\u0410\u00a0\u0442\u0435\u043f\u0435\u0440\u044c \u0432\u0438\u0448\u0435\u043d\u043a\u0430 \u043d\u0430\u00a0\u0442\u043e\u0440\u0442\u0435 \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u0438 \u043d\u0430\u0448\u0435\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f.<\/p>\n<p>\u00ab\u041d\u043e\u00a0\u043c\u044b\u00a0\u0436\u0435 \u043f\u043e\u0441\u0442\u0430\u0432\u0438\u043b\u0438 \u041f\u0418\u041d-\u043a\u043e\u0434 \u043d\u0430\u00a0\u043d\u0430\u0448\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u0447\u0442\u043e \u0435\u0449\u0451 \u043d\u0430\u0434\u043e?\u00bb\u00a0\u2014 \u0441\u043f\u0440\u043e\u0441\u0438\u0442 \u043d\u0435\u043e\u043f\u044b\u0442\u043d\u044b\u0439 \u0447\u0438\u0442\u0430\u0442\u044c.<br \/> \u0410\u00a0\u044f\u00a0\u0437\u0430\u0434\u0430\u043c \u043e\u0442\u0432\u0435\u0442\u043d\u044b\u0439 \u0432\u043e\u043f\u0440\u043e\u0441: \u00ab\u0410\u00a0\u0435\u0441\u043b\u0438 \u0447\u0435\u043b\u043e\u0432\u0435\u043a \u0437\u0430\u0445\u043e\u0447\u0435\u0442 \u0443\u0434\u0430\u043b\u0438\u0442\u044c \u043d\u0430\u0448\u0435 \u043f\u0440\u0435\u0434\u043b\u043e\u0436\u0435\u043d\u0438\u0435?\u00bb<br \/> \u0421\u0434\u0435\u043b\u0430\u0435\u0442 \u043e\u043d\u00a0\u044d\u0442\u043e \u043e\u0447\u0435\u043d\u044c \u0431\u044b\u0441\u0442\u0440\u043e \u0438\u00a0\u043f\u0440\u043e\u0441\u0442\u043e. <\/p>\n<figure class=\"\"><img decoding=\"async\" src=\"https:\/\/imgur.com\/iN0TSDp.gif\" alt=\"\u041f\u043e\u043b\u043d\u043e\u0435 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u0435\" data-src=\"https:\/\/imgur.com\/iN0TSDp.gif\"\/><\/p>\n<div><figcaption>\u041f\u043e\u043b\u043d\u043e\u0435 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u0435<\/figcaption><\/div>\n<\/figure>\n<p>\u0420\u0435\u0448\u0430\u0442\u044c \u0434\u0430\u043d\u043d\u0443\u044e \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0443 \u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u0447\u0435\u0440\u0435\u0437 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u0435 \u043f\u0440\u0430\u0432 \u0430\u0434\u043c\u0438\u043d\u0430 \u043d\u0430\u00a0\u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<p>\u041d\u043e\u00a0\u044d\u0442\u043e\u0433\u043e \u0431\u0443\u0434\u0435\u0442 \u043d\u0435\u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u043e, \u0442\u0430\u043a \u043a\u0430\u043a \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0441\u043c\u043e\u0436\u0435\u0442 \u0432\u00a0\u043b\u044e\u0431\u043e\u0439 \u043c\u043e\u043c\u0435\u043d\u0442 \u0443\u0431\u0440\u0430\u0442\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0438\u0437\u00a0\u0430\u0434\u043c\u0438\u043d\u043e\u0432.<br \/> \u041f\u043e\u0442\u043e\u043c\u0443 \u043c\u044b:<\/p>\n<ol>\n<li>\n<p>\u041f\u0440\u0438\u043d\u0443\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u043f\u043e\u0441\u0442\u0430\u0432\u0438\u043c \u041f\u0418\u041d-\u043a\u043e\u0434 \u043d\u0430\u00a0\u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 (\u0432\u00a0\u043d\u0430\u0448\u0435\u043c \u0441\u043b\u0443\u0447\u0430\u0435, \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0442\u0430\u043a \u0438\u00a0\u043d\u0430\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f\u00a0\u2014 \u00ab\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438\u00bb);<\/p>\n<\/li>\n<li>\n<p>\u0412\u00a0\u0441\u043b\u0443\u0447\u0430\u0435, \u0435\u0441\u043b\u0438 \u0443\u00a0\u043d\u0430\u0448\u0435\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u0431\u0443\u0434\u0443\u0442 \u0437\u0430\u0431\u0440\u0430\u043d\u044b \u043f\u0440\u0430\u0432\u0430 \u0430\u0434\u043c\u0438\u043d\u0430, \u0442\u043e\u00a0\u043c\u044b\u00a0\u043f\u0440\u043e\u0441\u0442\u043e \u0437\u0430\u0431\u043b\u043e\u043a\u0438\u0440\u0443\u0435\u043c \u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u043e (\u0430\u043d\u0430\u043b\u043e\u0433\u0438\u0447\u043d\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e \u043a\u043d\u043e\u043f\u043a\u0438 \u0432\u044b\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u044f).<\/p>\n<\/li>\n<\/ol>\n<p>\u041f\u043e\u0441\u043b\u0435 \u0442\u0430\u043a\u0438\u0445 \u043c\u0430\u043d\u0438\u043f\u0443\u043b\u044f\u0446\u0438\u0439, \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e, \u0447\u0442\u043e\u0431\u044b \u0443\u0434\u0430\u043b\u0438\u0442\u044c \u043d\u0430\u0448\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043f\u0440\u0438\u0434\u0435\u0442\u0441\u044f \u0432\u0432\u043e\u0434\u0438\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434\u044b \u0438\u00a0\u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f, \u0438\u00a0\u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0430, \u0447\u0442\u043e \u043e\u0431\u0435\u0441\u043f\u0435\u0447\u0438\u0432\u0430\u0435\u0442 \u043d\u0435\u043f\u043b\u043e\u0445\u0443\u044e (\u0445\u043e\u0442\u044c \u0438\u00a0\u043d\u0435\u00a0\u0438\u0434\u0435\u0430\u043b\u044c\u043d\u0443\u044e) \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u044c.<\/p>\n<p>\u0414\u043b\u044f \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u044f \u043f\u0440\u0430\u0432 \u0430\u0434\u043c\u0438\u043d\u0430 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u043d\u0430\u043f\u0438\u0441\u0430\u0442\u044c \u0441\u0432\u043e\u0439 <em>DeviceAdminReceiver<\/em>\u00a0\u2014 \u043f\u043e\u0434\u043a\u043b\u0430\u0441\u0441 <em>BroadcastReceiver-a<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u043f\u0435\u0440\u0435\u0445\u0432\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f, \u0430\u00a0\u0442\u0430\u043a\u0436\u0435 \u0441\u043f\u043e\u0441\u043e\u0431\u043d\u044b\u0439 \u0432\u044b\u043f\u043e\u043b\u043d\u044f\u0442\u044c \u0440\u044f\u0434 \u043f\u0440\u0438\u0432\u0438\u043b\u0435\u0433\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0445 \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0439, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u043c\u0435\u043d\u044f\u0442\u044c \u043f\u0430\u0440\u043e\u043b\u044c \u0438\u043b\u0438 \u043e\u0447\u0438\u0449\u0430\u0442\u044c \u0434\u0430\u043d\u043d\u044b\u0435 \u043d\u0430\u00a0\u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0435.<\/p>\n<p>AndroidManifest.xml<\/p>\n<pre><code class=\"xml\">       &lt;receiver            android:name=\".receiver.AdminReceiver\"            android:label=\"@string\/app_name\"            android:permission=\"android.permission.BIND_DEVICE_ADMIN\"            android:exported=\"false\"&gt;            &lt;meta-data                android:name=\"android.app.device_admin\"                android:resource=\"@xml\/device_admin\"\/&gt;            &lt;intent-filter&gt;                &lt;action android:name=\"android.app.action.DEVICE_ADMIN_ENABLED\"\/&gt;            &lt;\/intent-filter&gt;        &lt;\/receiver&gt; <\/code><\/pre>\n<p>\u041c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0435 \u0434\u043b\u044f <em>AdminReceiver<\/em>; \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0431\u0443\u0434\u0435\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u0444\u043e\u0440\u0441\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438 \u044d\u043a\u0440\u0430\u043d\u0430 (<em>force-lock<\/em>).<br \/> device_admin.xml<\/p>\n<pre><code class=\"xml\">&lt;?xml version=\"1.0\" encoding=\"utf-8\"?&gt; &lt;device-admin&gt;    &lt;uses-policies&gt;        &lt;force-lock \/&gt;    &lt;\/uses-policies&gt; &lt;\/device-admin&gt; <\/code><\/pre>\n<p>AdminReceiver.kt<\/p>\n<pre><code class=\"kotlin\">class AdminReceiver : DeviceAdminReceiver() {    override fun onReceive(context: Context, intent: Intent) {        val action = intent.action        when (action) {            ACTION_DEVICE_ADMIN_DISABLED,            ACTION_DEVICE_ADMIN_DISABLE_REQUESTED -&gt; {                val dpm = context.getSystemService(DevicePolicyManager::class.java)                dpm.lockNow()            }        }    } } <\/code><\/pre>\n<p>\u0417\u0430\u043f\u0440\u0430\u0448\u0438\u0432\u0430\u0435\u043c \u043f\u0440\u0430\u0432\u0430 \u0430\u0434\u043c\u0438\u043d\u0430 \u0443\u00a0\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f:<br \/> MainActivity.kt<\/p>\n<pre><code class=\"kotlin\">   private val dpm by lazy { getSystemService(DEVICE_POLICY_SERVICE) as DevicePolicyManager }    private val adminReceiver by lazy { ComponentName(applicationContext, AdminReceiver::class.java) }    ***    override fun onCreate(savedInstanceState: Bundle?) {        super.onCreate(savedInstanceState)        ***        setContent {            MaterialTheme {                ***                val isAdminGranted by viewModel.isAdminGranted.collectAsState()                 if (!isAccessibilityGranted || !isNotificationGranted || !isAdminGranted) {                    Dialog(onDismissRequest = {                        \/\/ block                    }) {                        Card(                            modifier = Modifier                                .fillMaxWidth()                                .padding(20.dp),                            shape = RoundedCornerShape(16.dp),                        ) {                            ***                            if (!isAdminGranted) {                                OutlinedButton(                                    modifier = Modifier                                        .fillMaxWidth()                                        .padding(4.dp),                                    onClick = {                                        val adminAskIntent = Intent(ACTION_ADD_DEVICE_ADMIN).apply {                                            putExtra(EXTRA_DEVICE_ADMIN, adminReceiver)                                            putExtra(                                                EXTRA_ADD_EXPLANATION,                                                \"\u041d\u0435 \u043f\u0440\u043e\u0442\u0438\u0432, \u0435\u0441\u043b\u0438 \u044f \u043f\u043e\u0437\u0430\u0438\u043c\u0441\u0442\u0432\u0443\u044e \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u0430\u0434\u043c\u0438\u043d\u0441\u043a\u0438\u0445 \u043f\u0440\u0430\u0432?\",                                            )                                        }                                        startActivity(adminAskIntent)                                    },                                ) {                                    Text(text = \"\u041f\u0440\u0430\u0432\u0430 \u0430\u0434\u043c\u0438\u043d\u0430\")                                }                            }                        }                    }                }                 Screen(                    ***                )            }        }    } ***     \/**      * \u041f\u0440\u043e\u0432\u0435\u0440\u043a\u0430 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u044b\u0445 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u0439:      * 1. \u0421\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u044b\u0435 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 (AccessibilityService)      * 2. \u0423\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u044f      * 3. \u041f\u0440\u0430\u0432\u0430 \u0430\u0434\u043c\u0438\u043d\u0430      *\/    private fun checkPermission() {        val isAccessibilityGranted = isAccessibilityServiceEnabled()        viewModel.setAccessibilityPermission(isAccessibilityGranted)         val isAdminGranted = dpm.isAdminActive(adminReceiver)        viewModel.setAdminPermission(isAdminGranted)         val isNotificationGranted = if (Build.VERSION.SDK_INT &gt;= Build.VERSION_CODES.TIRAMISU) {            ContextCompat.checkSelfPermission(                this@MainActivity,                POST_NOTIFICATIONS,            ) == PackageManager.PERMISSION_GRANTED        } else {            true        }        viewModel.setNotificationPermission(isNotificationGranted)    } } <\/code><\/pre>\n<p>MainViewModel.kt<\/p>\n<pre><code class=\"kotlin\">    private val _isAdminGranted = MutableStateFlow(false)     val isAdminGranted = _isAdminGranted.asStateFlow()      fun setAdminPermission(isGranted: Boolean) {         _isAdminGranted.update { isGranted }     } <\/code><\/pre>\n<p>\u0422\u0430\u043a\u0436\u0435 \u043d\u0430\u043c \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0434\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u044c \u0444\u0443\u043d\u043a\u0446\u0438\u043e\u043d\u0430\u043b <em>PinAccessibilityService<\/em>. \u0412\u00a0\u043d\u0435\u043c \u043c\u044b\u00a0\u0440\u0430\u0437 \u0432\u00a015\u00a0\u0441\u0435\u043a\u0443\u043d\u0434 \u0431\u0443\u0434\u0435\u043c \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0442\u044c, \u0435\u0441\u0442\u044c\u00a0\u043b\u0438 \u043f\u0440\u0430\u0432\u0430 \u0430\u0434\u043c\u0438\u043d\u0430. \u0415\u0441\u043b\u0438 \u0435\u0441\u0442\u044c, \u0442\u043e\u00a0\u0431\u0443\u0434\u0435\u043c \u0443\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0443 \u043d\u0430\u00a0\u00ab\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438\u00bb:<br \/> PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">   private var lastPinnedAppPackageName: String? = null    private var settingsList = listOf&lt;String&gt;()     private val dpm by lazy { getSystemService(DEVICE_POLICY_SERVICE) as DevicePolicyManager }    private val adminReceiver by lazy {        ComponentName(            applicationContext,            AdminReceiver::class.java        )    }     init {        scope.launch {            while (isActive) {                delay(15_000)                if (dpm.isAdminActive(adminReceiver)) {                    val intent = Intent(ACTION_SETTINGS)                    settingsList = packageManager                        .queryIntentActivities(intent, PackageManager.MATCH_ALL)                        .distinctBy { it.activityInfo.packageName }                        .map { it.activityInfo.packageName }                }            }        }         scope.launch {            while (isActive) {                delay(4000)                packageForPinList = spRepository.packageIdList + settingsList            }        }        ***    } <\/code><\/pre>\n<p>\u0412\u00a0\u0438\u0442\u043e\u0433\u0435 \u043c\u044b\u00a0\u0434\u043e\u0431\u0438\u043b\u0438\u0441\u044c \u0442\u043e\u0433\u043e, \u0447\u0442\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e, \u0447\u0442\u043e\u0431\u044b \u0443\u0434\u0430\u043b\u0438\u0442\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f \u043f\u0440\u0438\u043b\u043e\u0436\u0438\u0442\u044c \u043d\u0435\u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0443\u0441\u0438\u043b\u0438\u044f: <\/p>\n<figure class=\"\"><img decoding=\"async\" src=\"https:\/\/imgur.com\/AwslVlM.png\" alt=\"\u041f\u043e\u043b\u043d\u043e\u0435 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u0435\" data-src=\"https:\/\/imgur.com\/AwslVlM.png\"\/><\/p>\n<div><figcaption>\u041f\u043e\u043b\u043d\u043e\u0435 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u0435<\/figcaption><\/div>\n<\/figure>\n<h2>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435<\/h2>\n<p>\u0426\u0435\u043b\u044c\u044e \u0434\u0430\u043d\u043d\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0438 \u0431\u044b\u043b\u043e \u043f\u043e\u043a\u0430\u0437\u0430\u0442\u044c, \u0447\u0442\u043e \u0434\u043b\u044f \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u043f\u0440\u043e\u0433\u0440\u0430\u043c\u043c\u043d\u043e\u0439 \u0445\u043e\u0442\u0435\u043b\u043a\u0438 \u043d\u0443\u0436\u043d\u043e \u0436\u0435\u043b\u0430\u043d\u0438\u0435 \u0440\u0430\u0437\u043e\u0431\u0440\u0430\u0442\u044c\u0441\u044f \u0432\u00a0\u0438\u043d\u0442\u0435\u0440\u0435\u0441\u0443\u044e\u0449\u0435\u0439 \u0432\u0430\u0441 \u0442\u0435\u043c\u0435&#8230; \u043d\u0443\u00a0\u0438\u00a0\u043d\u0430\u043b\u0438\u0447\u0438\u0435 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 \u0432\u00a0\u0434\u0430\u043d\u043d\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u0435 \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u044d\u0442\u0443 \u0445\u043e\u0442\u0435\u043b\u043a\u0443.<\/p>\n<p>\u041c\u043e\u0436\u043d\u043e\u00a0\u043b\u0438 \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u043b\u0443\u0447\u0448\u0435? \u0415\u0441\u0442\u0435\u0441\u0442\u0432\u0435\u043d\u043d\u043e.<br \/> \u0412\u00a0\u0441\u0442\u0430\u0442\u044c\u0435 \u044f\u00a0\u043d\u0435\u00a0\u0443\u0447\u0435\u043b \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u043a\u0443 \u0441\u043f\u0435\u0446\u0438\u0444\u0438\u0447\u0435\u0441\u043a\u0438\u0445 \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u0432 (\u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, <em>com.google.android.googlequicksearchbox<\/em>), \u0432\u0441\u0435\u0432\u043e\u0437\u043c\u043e\u0436\u043d\u044b\u0435 \u00ab\u043f\u0435\u0442\u043b\u0438 \u0441\u00a0\u043e\u0442\u043a\u0440\u044b\u0442\u0438\u0435\u043c \u044d\u043a\u0440\u0430\u043d\u0430 \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438\u00bb, \u043e\u043f\u0442\u0438\u043c\u0438\u0437\u0430\u0446\u0438\u0438, \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0433\u0440\u0430\u0444\u0438\u0447\u0435\u0441\u043a\u0438\u0439 \u043f\u0430\u0440\u043e\u043b\u044c, \u0441\u0431\u0440\u043e\u0441 \u041f\u0418\u041d-\u043a\u043e\u0434\u0430 \u0432\u00a0\u0441\u043b\u0443\u0447\u0430\u0435, \u0435\u0441\u043b\u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0437\u0430\u0431\u044b\u043b \u043f\u0430\u0440\u043e\u043b\u044c \u0438\u00a0\u0442.\u043f. \u2014\u00a0\u044d\u0442\u043e \u043d\u0435\u00a0\u0432\u0445\u043e\u0434\u0438\u043b\u043e \u0432\u00a0\u043c\u043e\u0438 \u043f\u043b\u0430\u043d\u044b, \u043d\u043e\u00a0\u044f\u00a0\u043d\u0430\u0434\u0435\u044e\u0441\u044c, \u0447\u0442\u043e \u0434\u0430\u043d\u043d\u0430\u044f \u0441\u0442\u0430\u0442\u044c\u044f \u043e\u043a\u0430\u0436\u0435\u0442\u0441\u044f \u0432\u0430\u043c \u043f\u043e\u043b\u0435\u0437\u043d\u043e\u0439.<br \/> \u0412\u0435\u0441\u044c \u043a\u043e\u0434 \u0434\u043b\u044f \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0438 \u043c\u043e\u0436\u043d\u043e \u043d\u0430\u0439\u0442\u0438 \u0432\u043e\u0442 \u0437\u0434\u0435\u0441\u044c (<a href=\"https:\/\/github.com\/DradeFire\/SecurePinOnApp-For-Hubr\" rel=\"noopener noreferrer nofollow\">\u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043a\u043e\u0434<\/a>).<\/p>\n<p>\u0410\u00a0\u044d\u0442\u043e \u043f\u043e\u043b\u0435\u0437\u043d\u044b\u0435 \u0441\u0441\u044b\u043b\u043a\u0438 \u043d\u0430\u00a0\u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u044e:<br \/> \u2014\u00a0<a href=\"https:\/\/developer.android.com\/work\/device-admin\" rel=\"noopener noreferrer nofollow\">Device administration overview\u00a0| Android Developers<\/a><br \/> \u2014\u00a0<a href=\"https:\/\/developer.android.com\/guide\/topics\/ui\/accessibility\/service\" rel=\"noopener noreferrer nofollow\">Create your own accessibility service\u00a0| Android Developers<\/a><\/p>\n<\/div>\n<\/div>\n<\/div>\n<p><!----><!----><\/div>\n<p><!----><!----><br \/> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/834768\/\"> https:\/\/habr.com\/ru\/articles\/834768\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<div><!--[--><!--]--><\/div>\n<div id=\"post-content-body\">\n<div>\n<div class=\"article-formatted-body article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<h2>\u041f\u0440\u0435\u0434\u044b\u0441\u0442\u043e\u0440\u0438\u044f<\/h2>\n<p>\u0415\u0449\u0451 \u0441\u00a0\u0434\u0435\u0442\u0441\u0442\u0432\u0430 \u043c\u043e\u0439 \u043e\u0442\u0435\u0446 \u043f\u0440\u0438\u0443\u0447\u0438\u043b \u043c\u0435\u043d\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0430\u043d\u0442\u0438\u0432\u0438\u0440\u0443\u0441\u0430\u043c\u0438. \u0421\u043e\u0431\u043b\u044e\u0434\u0430\u044f \u0442\u0440\u0430\u0434\u0438\u0446\u0438\u0438, \u044f\u00a0\u043a\u0443\u043f\u0438\u043b \u0441\u0435\u0431\u0435 \u043f\u043e\u0434\u043f\u0438\u0441\u043a\u0443 \u043d\u0430\u00a0\u0430\u043d\u0442\u0438\u0432\u0438\u0440\u0443\u0441 \u0434\u043b\u044f \u0410\u043d\u0434\u0440\u043e\u0438\u0434\u0430. \u041e\u043a\u0430\u0437\u0430\u043b\u043e\u0441\u044c, \u0432\u00a0\u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0438 \u0435\u0441\u0442\u044c \u043a\u0440\u0430\u0439\u043d\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u0430\u044f \u0444\u0438\u0447\u0430\u00a0\u2014 \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u041f\u0418\u041d-\u043a\u043e\u0434\u0430 \u0434\u043b\u044f \u0434\u0440\u0443\u0433\u0438\u0445 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u043d\u0430\u00a0\u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0435. \u0418\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0439 \u043e\u043d\u0430 \u0431\u044b\u043b\u0430 \u0434\u043b\u044f \u043c\u0435\u043d\u044f \u0442\u0435\u043c, \u0447\u0442\u043e\u00a0\u044f, \u043a\u0430\u043a \u043c\u043e\u0431\u0438\u043b\u044c\u043d\u044b\u0439 \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a, \u043d\u0435\u00a0\u0438\u043c\u0435\u043b \u043d\u0438\u00a0\u043c\u0430\u043b\u0435\u0439\u0448\u0435\u0433\u043e \u043f\u043e\u043d\u044f\u0442\u0438\u044f, \u043a\u0430\u043a \u043f\u043e\u0434\u043e\u0431\u043d\u043e\u0435 \u043c\u043e\u0436\u043d\u043e \u0441\u0434\u0435\u043b\u0430\u0442\u044c. \u0418\u00a0\u0432\u043e\u0442 \u0442\u0435\u043f\u0435\u0440\u044c, \u043f\u043e\u0441\u043b\u0435 \u043d\u0435\u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0445 \u0440\u0430\u0441\u043a\u043e\u043f\u043e\u043a \u0438\u00a0\u043f\u0440\u043e\u0434\u0435\u043b\u0430\u043d\u043d\u043e\u0439 \u0440\u0430\u0431\u043e\u0442\u044b, \u044f\u00a0\u0434\u0435\u043b\u044e\u0441\u044c \u0441\u0432\u043e\u0438\u043c \u043e\u043f\u044b\u0442\u043e\u043c.<\/p>\n<h2>\u041f\u043b\u0430\u043d<\/h2>\n<p>\u041f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0434\u043e\u043b\u0436\u043d\u043e \u0443\u043c\u0435\u0442\u044c:<\/p>\n<ol>\n<li>\n<p>\u0420\u0430\u0441\u043f\u043e\u0437\u043d\u0430\u0432\u0430\u0442\u044c, \u043a\u043e\u0433\u0434\u0430 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u041f\u0418\u041d-\u043a\u043e\u0434\u043e\u043c;<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434 \u043f\u0440\u0438 \u043e\u0442\u043a\u0440\u044b\u0442\u0438\u0438 \u0441\u0430\u043c\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f (\u0432\u00a0\u0440\u0430\u043c\u043a\u0430\u0445 \u00ab\u0441\u0430\u043c\u043e\u0437\u0430\u0449\u0438\u0442\u044b\u00bb);<\/p>\n<\/li>\n<li>\n<p>\u0412\u044b\u0431\u0438\u0440\u0430\u0442\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u0434\u043b\u044f \u0431\u043b\u043e\u043a\u0430;<\/p>\n<\/li>\n<li>\n<p>\u0421\u043e\u0437\u0434\u0430\u0432\u0430\u0442\u044c\/\u043c\u0435\u043d\u044f\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434;<\/p>\n<\/li>\n<li>\n<p>\u0417\u0430\u0449\u0438\u0449\u0430\u0442\u044c \u0441\u0435\u0431\u044f \u043e\u0442\u00a0\u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f.<\/p>\n<\/li>\n<\/ol>\n<h2>\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u0438\u0435 \u0442\u0435\u043a\u0443\u0449\u0435\u0433\u043e \u0437\u0430\u043f\u0443\u0449\u0435\u043d\u043d\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f<\/h2>\n<p>\u041f\u0435\u0440\u0432\u044b\u043c \u0434\u0435\u043b\u043e\u043c \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0431\u044b\u043b\u043e \u0440\u0435\u0448\u0438\u0442\u044c \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0443\u00a0\u2014 \u00ab\u043a\u0430\u043a \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c, \u0447\u0442\u043e \u043e\u0442\u043a\u0440\u044b\u043b\u043e\u0441\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043d\u0443\u0436\u043d\u043e \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u0430\u0442\u044c?\u00bb.<\/p>\n<p>\u0421\u043d\u0430\u0447\u0430\u043b\u0430 \u043c\u043e\u0439 \u0432\u0437\u0433\u043b\u044f\u0434 \u043f\u0430\u043b \u043d\u0430\u00a0\u0432\u0430\u0440\u0438\u0430\u043d\u0442 \u0441\u00a0\u043e\u0442\u043b\u043e\u0432\u043e\u043c \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0445 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439 \u0447\u0435\u0440\u0435\u0437 <em>BroadcastReceiver<\/em>.<br \/> \u041a\u00a0\u0441\u043e\u0436\u0430\u043b\u0435\u043d\u0438\u044e, \u043e\u043a\u0430\u0437\u0430\u043b\u043e\u0441\u044c \u0447\u0442\u043e \u043d\u0435\u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e \u043e\u0442\u043b\u043e\u0432\u0438\u0442\u044c \u0438\u043d\u0442\u0435\u043d\u0442 \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f.<\/p>\n<p>\u0412\u0442\u043e\u0440\u044b\u043c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u044b\u043c \u0440\u0435\u0448\u0435\u043d\u0438\u0435\u043c \u0431\u044b\u043b\u043e \u0440\u0435\u0433\u0443\u043b\u044f\u0440\u043d\u043e (\u0440\u0430\u0437 \u0432\u00a0\u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0441\u0435\u043a\u0443\u043d\u0434) \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0442\u044c \u0437\u0430\u043f\u0443\u0449\u0435\u043d\u043d\u044b\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f, \u0447\u0435\u0440\u0435\u0437 <em>ActivityManager.runningAppProcesses()<\/em>.<br \/> \u041d\u043e, \u043d\u0430\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u044f\u00a0\u043f\u043e\u043d\u0438\u043c\u0430\u044e, \u0441\u0435\u0439\u0447\u0430\u0441 \u0434\u0430\u043d\u043d\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u044b \u0442\u0435\u043a\u0443\u0449\u0435\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f, \u0447\u0442\u043e \u043d\u0430\u043c \u043d\u0435\u00a0\u043f\u043e\u0434\u0445\u043e\u0434\u0438\u0442.<\/p>\n<p>\u0427\u0442\u043e\u00a0\u0436, \u043e\u0441\u0442\u0430\u0451\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e&#8230;\u0441\u0442\u0430\u0440\u044b\u0439 \u0434\u043e\u0431\u0440\u044b\u0439 <em>AccessibilityService<\/em><\/p>\n<p><em>AccessibilityService<\/em>\u00a0\u2014 \u043e\u0447\u0435\u043d\u044c \u043c\u043e\u0449\u043d\u044b\u0439 \u0438\u00a0\u043e\u0434\u043d\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u043e \u043e\u043f\u0430\u0441\u043d\u044b\u0439 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442. \u0415\u0433\u043e \u00ab\u043f\u043e\u043b\u0435\u0437\u043d\u044b\u0435 \u0444\u0443\u043d\u043a\u0446\u0438\u0438\u00bb (\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u0438\u0435 \u0438\u00a0\u043e\u0437\u0432\u0443\u0447\u0438\u0432\u0430\u043d\u0438\u0435 \u0442\u0435\u043a\u0441\u0442\u0430 \u043d\u0430\u00a0\u044d\u043a\u0440\u0430\u043d\u0435, \u043d\u0430\u0436\u0430\u0442\u0438\u0435 \u043d\u0430\u00a0\u043a\u043d\u043e\u043f\u043a\u0438, \u0441\u0432\u0430\u0439\u043f\u044b \u0438\u00a0\u0442.\u0434.) \u0434\u043b\u044f \u043b\u044e\u0434\u0435\u0439 \u0441\u00a0\u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u043d\u044b\u043c\u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044f\u043c\u0438 \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432\u00a0\u0441\u0430\u043c\u044b\u0445 \u0443\u0436\u0430\u0441\u043d\u044b\u0445 \u0446\u0435\u043b\u044f\u0445, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0432\u00a0\u043d\u0430\u0448\u0438\u0445. \u0422\u0430\u043a \u0438\u00a0\u043f\u043e\u0441\u0442\u0443\u043f\u0438\u043c.<\/p>\n<h2>\u041d\u0430\u0447\u0430\u043b\u044c\u043d\u043e\u0435 \u0440\u0435\u0448\u0435\u043d\u0438\u0435<\/h2>\n<p>\u0414\u043b\u044f \u043d\u0430\u0447\u0430\u043b\u0430 \u043c\u044b\u00a0\u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0432\u00a0\u043c\u0430\u043d\u0438\u0444\u0435\u0441\u0442 \u043d\u0430\u0448 \u0441\u0435\u0440\u0432\u0438\u0441 \u0438\u00a0\u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u044b\u0435 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u0438\u044f:<\/p>\n<p>AndroidManifest.xml<\/p>\n<pre><code class=\"xml\">   &lt;uses-permission        android:name=\"android.permission.BIND_ACCESSIBILITY_SERVICE\"        tools:ignore=\"ProtectedPermissions\" \/&gt;    &lt;uses-permission        android:name=\"android.permission.QUERY_ALL_PACKAGES\"        tools:ignore=\"QueryAllPackagesPermission\" \/&gt;    &lt;uses-permission android:name=\"android.permission.FOREGROUND_SERVICE\" \/&gt;    &lt;uses-permission android:name=\"android.permission.FOREGROUND_SERVICE_SPECIAL_USE\" \/&gt;    &lt;uses-permission android:name=\"android.permission.POST_NOTIFICATIONS\" \/&gt;   ***        &lt;service            android:name=\".service.PinAccessibilityService\"            android:exported=\"true\"            android:foregroundServiceType=\"specialUse\"            android:permission=\"android.permission.BIND_ACCESSIBILITY_SERVICE\"&gt;            &lt;property                android:name=\"android.app.PROPERTY_SPECIAL_USE_FGS_SUBTYPE\"                android:value=\"pin_code_for_another_apps\" \/&gt;             &lt;intent-filter&gt;                &lt;action android:name=\"android.accessibilityservice.AccessibilityService\" \/&gt;            &lt;\/intent-filter&gt;             &lt;meta-data                android:name=\"android.accessibilityservice\"                android:resource=\"@xml\/accessibilityservice\" \/&gt;        &lt;\/service&gt; <\/code><\/pre>\n<p>\u041c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0435 \u0434\u043b\u044f <em>PinAccessibilityService<\/em>.<br \/> accessibilityservice.xml<\/p>\n<pre><code class=\"xml\">&lt;?xml version=\"1.0\" encoding=\"utf-8\"?&gt; &lt;accessibility-service xmlns:android=\"http:\/\/schemas.android.com\/apk\/res\/android\"    android:accessibilityEventTypes=\"typeAllMask\"    android:canRequestEnhancedWebAccessibility=\"true\"    android:notificationTimeout=\"100\"    android:accessibilityFeedbackType=\"feedbackGeneric\" android:accessibilityFlags=\"flagRetrieveInteractiveWindows|flagIncludeNotImportantViews\"    android:canRetrieveWindowContent=\"true\"\/&gt; <\/code><\/pre>\n<p>\u041a\u0430\u043a\u0438\u0435 \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u044b\u0435 \u0437\u0430\u0434\u0430\u0447\u0438 \u0434\u043e\u043b\u0436\u0435\u043d \u0440\u0435\u0448\u0430\u0442\u044c \u043d\u0430\u0448 \u0441\u0435\u0440\u0432\u0438\u0441:<\/p>\n<ul>\n<li>\n<p>\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0442\u044c, \u0447\u0442\u043e \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u043c\u0438 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u0449\u0438\u0442\u0438\u0442\u044c;<\/p>\n<\/li>\n<li>\n<p>\u041e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0442\u044c, \u0447\u0442\u043e \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u043c\u0438 \u043d\u0430\u0448\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u0447\u0442\u043e\u0431\u044b \u043e\u043d\u043e \u0437\u0430\u0449\u0438\u0449\u0430\u043b\u043e \u0441\u0430\u043c\u043e \u0441\u0435\u0431\u044f;<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u0432\u0432\u043e\u0434\u043e\u043c \u041f\u0418\u041d-\u043a\u043e\u0434\u0430.<\/p>\n<\/li>\n<\/ul>\n<p>\u041f\u0440\u0438\u0441\u0442\u0443\u043f\u0438\u043c \u043a\u00a0\u043d\u0430\u043f\u0438\u0441\u0430\u043d\u0438\u044e \u0441\u0430\u043c\u043e\u0433\u043e <em>AccessibilityService<\/em>.<\/p>\n<p>\u041f\u0440\u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0435 \u0441\u0435\u0440\u0432\u0438\u0441\u0430 \u043c\u044b\u00a0\u0437\u0430\u043a\u0440\u0435\u043f\u043b\u044f\u0435\u043c \u0443\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u0435, \u0447\u0442\u043e\u0431\u044b \u0441\u043b\u0435\u0434\u0438\u0442\u044c, \u0436\u0438\u0432\u00a0\u043b\u0438 \u043d\u0430\u0448 \u0441\u0435\u0440\u0432\u0438\u0441 (<em>AccessibilityService<\/em> \u043c\u043e\u0436\u0435\u0442 \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0432\u00a0\u0444\u043e\u043d\u0435 \u0431\u0435\u0437 \u0437\u0430\u043f\u0443\u0441\u043a\u0430 \u043a\u0430\u043a <em>ForegroundService<\/em>).<\/p>\n<p>PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">class PinAccessibilityService : AccessibilityService() {     override fun onCreate() {        super.onCreate()        startForeground()    }     private fun startForeground() {        val channelId = createNotificationChannel()         val notification = NotificationCompat.Builder(this, channelId)            .setContentTitle(\"Pin On App\")            .setContentText(\"Apps protected\")            .setOngoing(true)            .setSmallIcon(R.mipmap.ic_launcher)            .setPriority(PRIORITY_HIGH)            .setCategory(Notification.CATEGORY_SERVICE)            .build()        startForeground(101, notification)    }     private fun createNotificationChannel(): String {        val channelId = \"pin_on_app_service\"        val channelName = \"PinOnApp\"        val channel = NotificationChannel(            channelId,            channelName,            NotificationManager.IMPORTANCE_HIGH,        ).apply {            lightColor = Color.BLUE            lockscreenVisibility = Notification.VISIBILITY_PRIVATE        }         val service = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager        service.createNotificationChannel(channel)         return channelId    } } <\/code><\/pre>\n<p>\u0414\u0430\u043b\u0435\u0435, \u043c\u044b\u00a0\u043f\u044b\u0442\u0430\u0435\u043c\u0441\u044f \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c, \u0447\u0442\u043e \u0437\u0430\u00a0\u044d\u043a\u0440\u0430\u043d \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u043c\u0438\u00a0\u0438, \u0432\u00a0\u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043e\u0442\u00a0\u044d\u0442\u043e\u0433\u043e, \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u0435\u043c \u0438\u043b\u0438 \u043d\u0435\u00a0\u0431\u043b\u043e\u043a\u0438\u0440\u0443\u0435\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f.<\/p>\n<p>\u041b\u043e\u0433\u0438\u043a\u0430 \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0430\u044f:<\/p>\n<ol>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f <em>Launcher<\/em>\u00a0\u2014 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0431\u044b\u043b \u043d\u0430\u00a0\u043e\u0441\u043d\u043e\u0432\u043d\u043e\u043c \u044d\u043a\u0440\u0430\u043d\u0435, \u0430\u00a0\u0437\u043d\u0430\u0447\u0438\u0442 \u043f\u0440\u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0435 \u0437\u0430\u0449\u0438\u0449\u0435\u043d\u043d\u043e\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u0432\u0432\u043e\u0434\u043e\u043c \u041f\u0418\u041d-\u043a\u043e\u0434\u0430.<\/p>\n<\/li>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043d\u0443\u0436\u043d\u043e \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u0430\u0442\u044c, \u0438\u00a0\u041f\u0418\u041d \u043d\u0435\u00a0\u0431\u044b\u043b \u0432\u0432\u0435\u0434\u0435\u043d\u00a0\u2014 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c \u044d\u043a\u0440\u0430\u043d \u0441\u00a0\u0432\u0432\u043e\u0434\u043e\u043c \u041f\u0418\u041d-\u043a\u043e\u0434\u0430. \u041d\u0443\u0436\u043d\u043e\u0435 \u043d\u0430\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u043c \u043f\u043e\u00a0\u0435\u0433\u043e <em>packageName<\/em>.<\/p>\n<\/li>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u043f\u043e\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0437\u0430\u0449\u0438\u0442\u0443 \u043d\u0430\u00a0\u043d\u0430\u0448\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435, \u0442\u043e\u00a0\u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u043f\u043e\u043f\u0430\u0434\u0430\u0442\u044c \u0432\u00a0\u0446\u0438\u043a\u043b \u0438\u00a0\u0431\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u043e \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0432\u0432\u043e\u0434\u0430 \u041f\u0418\u041d\u0430, \u0442.\u043a.\u00a0\u043c\u044b\u00a0\u043e\u0440\u0438\u0435\u043d\u0442\u0438\u0440\u0443\u0435\u043c\u0441\u044f \u043d\u0430\u00a0<em>packageName<\/em> \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439, \u0430\u00a0<em>packageName<\/em> \u0443\u00a0\u044d\u043a\u0440\u0430\u043d\u0430 \u0431\u043b\u043e\u043a\u0430 \u0438\u00a0\u043d\u0430\u0448\u0435\u0433\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u043e\u0434\u0438\u043d\u0430\u043a\u043e\u0432\u044b\u0439. \u0414\u043b\u044f \u0440\u0435\u0448\u0435\u043d\u0438\u044f \u044d\u0442\u043e\u0439 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u043c\u044b\u00a0\u0437\u0430\u0432\u044f\u0436\u0435\u043c\u0441\u044f \u043d\u0430\u00a0\u0443\u043d\u0438\u043a\u0430\u043b\u044c\u043d\u044b\u0439 \u0438\u0434\u0435\u043d\u0442\u0438\u0444\u0438\u043a\u0430\u0442\u043e\u0440 \u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0433\u043e \u044d\u043a\u0440\u0430\u043d\u0430.<\/p>\n<\/li>\n<\/ol>\n<p><em>Launcher-\u044b<\/em> \u043d\u0430\u00a0\u0440\u0430\u0437\u043d\u044b\u0445\u00a0\u041e\u0421 \u0440\u0430\u0437\u043d\u044b\u0435, \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u043c\u044b\u00a0\u0437\u0430\u0440\u0430\u043d\u0435\u0435 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0441\u043f\u0438\u0441\u043e\u043a \u0432\u0441\u0435\u0445 <em>Launcher-\u043e\u0432<\/em> \u043d\u0430\u00a0\u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0435.<\/p>\n<p>PinAccessibilityService.kt<\/p>\n<pre><code class=\"kotlin\">   private val scope = CoroutineScope(Dispatchers.Main + SupervisorJob())    private val event = MutableSharedFlow&lt;AccessibilityEvent&gt;(       extraBufferCapacity = 1,       onBufferOverflow = BufferOverflow.DROP_OLDEST,    )     private val spRepository by inject&lt;SharedPreferencesRepository&gt;()    private var packageForPinList = spRepository.packageIdList     private var launcherList = listOf&lt;String&gt;()     private val isPinCodeNotExist        get() = spRepository.pinCode.isNullOrEmpty()    private val isCorrectPin        get() = spRepository.isCorrectPin == true     init {        event            .filter { !isPinCodeNotExist &amp;&amp; !isCorrectPin }            .onEach { _ -&gt;                spRepository.isCorrectPin = false                 val startActivityIntent = Intent(applicationContext, ConfirmActivity::class.java)                    .apply {                        setFlags(                            Intent.FLAG_ACTIVITY_NEW_TASK                                    or Intent.FLAG_ACTIVITY_CLEAR_TASK                                    or Intent.FLAG_ACTIVITY_NO_ANIMATION                                    or Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS,                        )                    }                startActivity(startActivityIntent)            }.catch {                Log.e(\"ERROR\", it.message, it)            }.launchIn(scope)    }     override fun onCreate() {        super.onCreate()        startForeground()        setLauncherListOnDevice()    }     private fun setLauncherListOnDevice() {        val i = Intent(Intent.ACTION_MAIN).apply {            addCategory(Intent.CATEGORY_HOME)        }        launcherList = packageManager.queryIntentActivities(i, 0).map {            it.activityInfo.packageName        }    }     override fun onAccessibilityEvent(event: AccessibilityEvent?) {        if (event != null) {            if (launcherList.contains(event.packageName)) {                spRepository.isCorrectPin = false            } else if (packageForPinList.contains(event.packageName) || event.isMainActivityShowed()) {                this.event.tryEmit(event)            }        }    }     private fun AccessibilityEvent.isMainActivityShowed() =        className?.contains(APP_CLASS_NAME) ?: false     override fun onInterrupt() {}     override fun onDestroy() {        scope.cancel()        super.onDestroy()    }     companion object {        private const val APP_CLASS_NAME = \"com.dradefire.securepinonapp.ui.main.MainActivity\"    } <\/code><\/pre>\n<p>\u0421\u043f\u0438\u0441\u043e\u043a \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0442\u044c \u0432\u00a0<em>SharedPreferences<\/em>, \u0434\u043b\u044f \u043a\u043e\u0442\u043e\u0440\u044b\u0445 \u043d\u0430\u043f\u0438\u0448\u0435\u043c \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0439 (\u0434\u043b\u044f \u0432\u043d\u0435\u0434\u0440\u0435\u043d\u0438\u044f \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439 \u043c\u044b\u00a0\u0431\u0443\u0434\u0435\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c Koin).<\/p>\n<p>KoinModule.kt<\/p>\n<pre><code class=\"kotlin\">val KoinModule = module {    viewModelOf(::ConfirmViewModel)    viewModelOf(::MainViewModel)     factoryOf(::SharedPreferencesRepository)     singleOf(::Gson) } <\/code><\/pre>\n<p>App.kt<\/p>\n<pre><code class=\"kotlin\">class App : Application() {    override fun onCreate() {        super.onCreate()         startKoin {            androidContext(this@App)            modules(KoinModule)        }    } } <\/code><\/pre>\n<p>SharedPreferencesRepository.kt<\/p>\n<pre><code class=\"kotlin\">class SharedPreferencesRepository(    private val context: Context,    private val gson: Gson, ) {    \/**    * \u0411\u044b\u043b \u043b\u0438 \u0432\u0432\u0435\u0434\u0451\u043d \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u044b\u0439 \u041f\u0418\u041d-\u043a\u043e\u0434 (\u043d\u0443\u0436\u043d\u043e, \u0447\u0442\u043e\u0431\u044b \u043b\u0438\u0448\u043d\u0438\u0439 \u0440\u0430\u0437 \u043d\u0435 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u044d\u043a\u0440\u0430\u043d \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u0438)    *\/    var isCorrectPin: Boolean?         get() = context.sp.getBoolean(IS_CORRECT_PIN_KEY, false)        set(isCorrectPin) {            context.sp.edit().putBoolean(IS_CORRECT_PIN_KEY, isCorrectPin ?: false).apply()        }     \/**    * \u041f\u0418\u041d-\u043a\u043e\u0434    *\/    var pinCode: String?        get() = context.sp.getString(PIN_KEY, null)        set(pinCode) {            context.sp.edit().putString(PIN_KEY, pinCode).apply()        }    \/**    * \u0421\u043f\u0438\u0441\u043e\u043a \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u0449\u0438\u0442\u0438\u0442\u044c \u041f\u0418\u041d-\u043a\u043e\u0434\u043e\u043c    *\/    var packageIdList: List&lt;String&gt;         get() = gson.fromJson(            context.sp.getString(                PACKAGE_ID_LIST_KEY,                gson.toJson(emptyList&lt;String&gt;()),            ),            List::class.java,        ) as List&lt;String&gt;        set(list) <\/code><\/pre>\n<\/div>\n<\/div>\n<\/div>\n<\/div>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-428678","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/428678","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=428678"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/428678\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=428678"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=428678"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=428678"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}