{"id":479834,"date":"2026-05-15T08:00:20","date_gmt":"2026-05-15T08:00:20","guid":{"rendered":"https:\/\/savepearlharbor.com\/?p=479834"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=479834","title":{"rendered":"Mission Impossible: \u043a\u0430\u043a \u0434\u043e\u0431\u0438\u0442\u044c\u0441\u044f 0 \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439 \u0432 \u0441\u043b\u043e\u0436\u043d\u043e\u043c \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u043e\u043c UI"},"content":{"rendered":"<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/a5c\/99d\/8c1\/a5c99d8c1f5684799a229f1bdfad90ba.jpeg\" alt=\"Image by\u00a0author\" title=\"Image by\u00a0author\" width=\"1000\" height=\"416\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/a5c\/99d\/8c1\/a5c99d8c1f5684799a229f1bdfad90ba.jpeg 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/a5c\/99d\/8c1\/a5c99d8c1f5684799a229f1bdfad90ba.jpeg 781w\" loading=\"lazy\" decode=\"async\"\/><\/p>\n<div><figcaption>Image by\u00a0author<\/figcaption><\/div>\n<\/figure>\n<p><strong>\u042d\u0442\u043e \u043f\u0435\u0440\u0435\u0432\u043e\u0434 \u043c\u043e\u0435\u0439 \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0438, \u043e\u043f\u0443\u0431\u043b\u0438\u043a\u043e\u0432\u0430\u043d\u043d\u043e\u0439 \u0440\u0430\u043d\u0435\u0435 \u0432 ProAndroidDev.<\/strong><\/p>\n<p>\u0421\u043e\u0442\u043d\u044f \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443 \u043f\u0440\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0435 \u2014 \u044d\u0442\u043e \u043f\u0440\u0438\u0433\u043e\u0432\u043e\u0440. \u041f\u0440\u0438\u0433\u043e\u0432\u043e\u0440 \u0431\u0430\u0442\u0430\u0440\u0435\u0435 \u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0430, \u043f\u043b\u0430\u0432\u043d\u043e\u0441\u0442\u0438 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0439 \u0438 \u0432\u0430\u0448\u0435\u0439 \u0440\u0435\u043f\u0443\u0442\u0430\u0446\u0438\u0438 \u043a\u0430\u043a \u0438\u043d\u0436\u0435\u043d\u0435\u0440\u0430. \u041c\u044b \u043f\u0440\u0438\u0432\u044b\u043a\u043b\u0438 \u043c\u044b\u0441\u043b\u0438\u0442\u044c \u0432\u044b\u0441\u043e\u043a\u043e\u0443\u0440\u043e\u0432\u043d\u0435\u0432\u044b\u043c\u0438 \u0430\u0431\u0441\u0442\u0440\u0430\u043a\u0446\u0438\u044f\u043c\u0438: \u0437\u0430\u043a\u0438\u043d\u0443\u0442\u044c <code>LazyColumn<\/code>, \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043f\u0430\u0440\u0443 <code>Modifier.padding<\/code> \u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u0438\u0442\u044c \u0432 \u043f\u0440\u043e\u0434\u0430\u043a\u0448\u0435\u043d. \u041d\u043e \u0447\u0442\u043e \u0434\u0435\u043b\u0430\u0442\u044c, \u043a\u043e\u0433\u0434\u0430 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u044b \u043d\u0430\u0447\u0438\u043d\u0430\u044e\u0442 &#171;\u0437\u0430\u0445\u043b\u0435\u0431\u044b\u0432\u0430\u0442\u044c\u0441\u044f&#187;, \u0430 Layout Inspector \u0433\u043e\u0440\u0438\u0442 \u043a\u0440\u0430\u0441\u043d\u044b\u043c \u043e\u0442 \u0438\u0437\u0431\u044b\u0442\u043e\u0447\u043d\u044b\u0445 \u043e\u0442\u0440\u0438\u0441\u043e\u0432\u043e\u043a?<\/p>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0441\u043d\u0438\u043c\u0435\u043c \u0440\u043e\u0437\u043e\u0432\u044b\u0435 \u043e\u0447\u043a\u0438 \u0430\u0431\u0441\u0442\u0440\u0430\u043a\u0446\u0438\u0439. \u0415\u0441\u043b\u0438 \u0443\u0431\u0440\u0430\u0442\u044c \u0432\u0441\u044e &#171;\u043c\u0430\u0433\u0438\u044e&#187; \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u043e\u0432 Compose, \u043e\u0441\u0442\u0430\u043d\u0443\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e <code>Layout<\/code> \u0438 <code>Modifier<\/code>. \u0414\u0430\u0436\u0435 <code>Text<\/code> \u043f\u043e\u0434 \u043a\u0430\u043f\u043e\u0442\u043e\u043c \u2014 \u044d\u0442\u043e <code>BasicText<\/code>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043e\u043f\u0438\u0440\u0430\u0435\u0442\u0441\u044f \u043d\u0430 \u0444\u0443\u043d\u0434\u0430\u043c\u0435\u043d\u0442 <code>Layout<\/code>.<\/p>\n<p>\u0421\u0435\u0433\u043e\u0434\u043d\u044f \u043c\u044b \u0440\u0430\u0437\u0431\u0435\u0440\u0435\u043c \u0445\u0430\u0440\u0434\u043a\u043e\u0440\u043d\u044b\u0439 \u043a\u0435\u0439\u0441 \u0438\u0437 \u043c\u043e\u0435\u0433\u043e \u043f\u0440\u043e\u0435\u043a\u0442\u0430 <strong>MicroMod<\/strong>: \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u0431\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u043e\u0439 \u0441\u0435\u0442\u043a\u0438 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432, \u0440\u0430\u0441\u043f\u043e\u043b\u043e\u0436\u0435\u043d\u043d\u044b\u0445 \u043f\u043e \u0441\u043f\u0438\u0440\u0430\u043b\u0438, \u0441 \u0430\u0431\u0441\u043e\u043b\u044e\u0442\u043d\u043e \u043d\u0443\u043b\u0435\u0432\u044b\u043c \u043f\u043e\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u0435\u043c \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439 \u043f\u0440\u0438 \u0430\u043a\u0442\u0438\u0432\u043d\u043e\u043c \u0441\u043a\u0440\u043e\u043b\u043b\u0435. \u041d\u0438\u043a\u0430\u043a\u043e\u0439 \u0435\u0440\u0443\u043d\u0434\u044b \u0432 \u0434\u0443\u0445\u0435 &#171;\u043f\u0440\u043e\u0441\u0442\u043e \u0434\u043e\u0431\u0430\u0432\u044c @Stable&#187;. \u0422\u043e\u043b\u044c\u043a\u043e \u0433\u043b\u0443\u0431\u043e\u043a\u0430\u044f \u0440\u0430\u0431\u043e\u0442\u0430 \u0441 \u0444\u0430\u0437\u0430\u043c\u0438 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430, \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 <code>LazyLayout<\/code> \u0438 \u0443\u043c\u043d\u043e\u0435 \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435\u043c.<\/p>\n<p>\u041f\u0440\u0435\u0436\u0434\u0435 \u0447\u0435\u043c \u043f\u0435\u0440\u0435\u0439\u0442\u0438 \u043a \u043f\u0440\u0430\u043a\u0442\u0438\u043a\u0435, \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u043a\u0440\u0435\u043f\u0438\u0442\u044c \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u044e\u044e \u043c\u0435\u0445\u0430\u043d\u0438\u043a\u0443 \u0444\u0430\u0437\u044b Layout. \u041f\u0440\u043e\u0446\u0435\u0441\u0441 \u043a\u043e\u043c\u043f\u043e\u043d\u043e\u0432\u043a\u0438 \u0441\u043e\u0441\u0442\u043e\u0438\u0442 \u0438\u0437 \u0442\u0440\u0435\u0445 \u0448\u0430\u0433\u043e\u0432: <strong>Constraints<\/strong> (\u041e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u044f), <strong>Measurement<\/strong> (\u0418\u0437\u043c\u0435\u0440\u0435\u043d\u0438\u0435) \u0438 <strong>Placement<\/strong> (\u0420\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u0435). \u0412\u0435\u0447\u043d\u044b\u0439 \u0441\u043f\u043e\u0440 \u043e \u0442\u043e\u043c, \u043f\u0435\u0440\u0435\u0434\u0430\u0435\u0442\u0441\u044f \u043b\u0438 \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 &#171;\u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437&#187; \u0438\u043b\u0438 &#171;\u0441\u043d\u0438\u0437\u0443 \u0432\u0432\u0435\u0440\u0445&#187;, \u043d\u0435 \u0438\u043c\u0435\u0435\u0442 \u043e\u0434\u043d\u043e\u0437\u043d\u0430\u0447\u043d\u043e\u0433\u043e \u043e\u0442\u0432\u0435\u0442\u0430, \u0442\u0430\u043a \u043a\u0430\u043a \u044d\u0442\u0438 \u044d\u0442\u0430\u043f\u044b \u0432\u0437\u0430\u0438\u043c\u043e\u0441\u0432\u044f\u0437\u0430\u043d\u044b:<\/p>\n<ol>\n<li>\n<p><strong>Constraints<\/strong> \u043f\u0435\u0440\u0435\u0434\u0430\u044e\u0442\u0441\u044f \u043e\u0442 \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044f \u043a \u0434\u0435\u0442\u044f\u043c (\u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437).<\/p>\n<\/li>\n<li>\n<p><strong>Measurement<\/strong> (\u0440\u0430\u0441\u0447\u0435\u0442 \u0440\u0430\u0437\u043c\u0435\u0440\u043e\u0432) \u0438\u0434\u0435\u0442 \u0432 \u043e\u0431\u0440\u0430\u0442\u043d\u043e\u043c \u043f\u043e\u0440\u044f\u0434\u043a\u0435 (\u0441\u043d\u0438\u0437\u0443 \u0432\u0432\u0435\u0440\u0445).<\/p>\n<\/li>\n<li>\n<p><strong>Placement<\/strong> (\u043f\u043e\u0437\u0438\u0446\u0438\u043e\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435) \u0444\u0438\u043d\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0441\u043d\u043e\u0432\u0430 \u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437.<\/p>\n<\/li>\n<\/ol>\n<hr\/>\n<h4>\u0410\u043d\u0430\u0442\u043e\u043c\u0438\u044f \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430: \u043f\u043e\u0447\u0435\u043c\u0443 Compose \u043b\u0430\u0433\u0430\u0435\u0442<\/h4>\n<p>\u041e\u0442\u0440\u0438\u0441\u043e\u0432\u043a\u0430 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430 \u0432 Compose \u043f\u0440\u043e\u0445\u043e\u0434\u0438\u0442 \u0447\u0435\u0440\u0435\u0437 \u0442\u0440\u0438 \u0441\u0442\u0440\u043e\u0433\u043e \u0440\u0435\u0433\u043b\u0430\u043c\u0435\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0445 \u044d\u0442\u0430\u043f\u0430:<\/p>\n<ol>\n<li>\n<p><strong>Composition:<\/strong> \u0427\u0442\u043e \u043c\u044b \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c? Compose \u0441\u0442\u0440\u043e\u0438\u0442 \u0434\u0435\u0440\u0435\u0432\u043e UI.<\/p>\n<\/li>\n<li>\n<p><strong>Layout:<\/strong> \u0413\u0434\u0435 \u044d\u0442\u043e \u043d\u0430\u0445\u043e\u0434\u0438\u0442\u0441\u044f? \u0412\u043a\u043b\u044e\u0447\u0430\u0435\u0442 \u0448\u0430\u0433\u0438 Measurement (\u0441\u043d\u0438\u0437\u0443 \u0432\u0432\u0435\u0440\u0445) \u0438 Placement (\u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437).<\/p>\n<\/li>\n<li>\n<p><strong>Drawing:<\/strong> \u041a\u0430\u043a \u044d\u0442\u043e \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442? \u041e\u0442\u0440\u0438\u0441\u043e\u0432\u043a\u0430 \u043f\u0438\u043a\u0441\u0435\u043b\u0435\u0439 \u043d\u0430 \u044d\u043a\u0440\u0430\u043d\u0435.<\/p>\n<\/li>\n<\/ol>\n<p>\u0411\u043e\u043b\u044c\u0448\u0438\u043d\u0441\u0442\u0432\u043e \u043f\u0440\u043e\u0431\u043b\u0435\u043c \u0441 \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c\u044e (\u0442\u0435 \u0441\u0430\u043c\u044b\u0435 \u043b\u0430\u0433\u0438 \u043f\u0440\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0435) \u0432\u043e\u0437\u043d\u0438\u043a\u0430\u044e\u0442, \u043a\u043e\u0433\u0434\u0430 \u043c\u044b \u0437\u0430\u0441\u0442\u0430\u0432\u043b\u044f\u0435\u043c Compose \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0442\u044c\u0441\u044f \u043d\u0430 \u0444\u0430\u0437\u0443 Composition \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u043f\u0438\u043a\u0441\u0435\u043b\u044f. \u0414\u0430\u0436\u0435 \u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c\u044b \u0432\u0440\u043e\u0434\u0435 <em>Donut-hole skipping<\/em> (\u043f\u0440\u043e\u043f\u0443\u0441\u043a \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0438 \u0432\u043d\u0443\u0442\u0440\u0438 \u0444\u0443\u043d\u043a\u0446\u0438\u0438, \u0435\u0441\u043b\u0438 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u043d\u0435 \u0438\u0437\u043c\u0435\u043d\u0438\u043b\u0438\u0441\u044c) \u0438\u043b\u0438 \u043d\u043e\u0432\u044b\u0439 <em>Strong Skipping Mode<\/em> \u043d\u0435 \u0441\u043f\u0430\u0441\u0443\u0442, \u0435\u0441\u043b\u0438 \u0432\u044b \u0447\u0438\u0442\u0430\u0435\u0442\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u0441\u043a\u0440\u043e\u043b\u043b\u0430 \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e \u0432 \u0442\u0435\u043b\u0435 <code>@Composable<\/code> \u0444\u0443\u043d\u043a\u0446\u0438\u0438.<\/p>\n<hr\/>\n<h4>\u041b\u044f\u043c\u0431\u0434\u0430-\u043c\u043e\u0434\u0438\u0444\u0438\u043a\u0430\u0442\u043e\u0440\u044b \u0438 \u0441\u043f\u0443\u0441\u043a \u0434\u043e Canvas<\/h4>\n<p>\u041a\u043b\u0430\u0441\u0441\u0438\u0447\u0435\u0441\u043a\u0438\u0439 \u043f\u0440\u0438\u043c\u0435\u0440 \u0434\u0435\u043b\u0435\u0433\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u0439: \u0432\u043c\u0435\u0441\u0442\u043e <code>Modifier.offset(x = scrollState.value)<\/code>, \u0433\u0440\u0430\u043c\u043e\u0442\u043d\u044b\u0439 \u0438\u043d\u0436\u0435\u043d\u0435\u0440 \u043d\u0430\u043f\u0438\u0448\u0435\u0442 <code>Modifier.offset { IntOffset(x = scrollState.value, 0) }<\/code>. \u041b\u044f\u043c\u0431\u0434\u0430 \u043e\u0442\u043a\u043b\u0430\u0434\u044b\u0432\u0430\u0435\u0442 \u0447\u0442\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f \u0434\u043e \u0444\u0430\u0437\u044b Layout (\u0430 \u0438\u043c\u0435\u043d\u043d\u043e \u2014 Placement). \u0424\u0430\u0437\u0430 Composition \u043f\u0440\u0438 \u044d\u0442\u043e\u043c \u043e\u0441\u0442\u0430\u0435\u0442\u0441\u044f \u043d\u0435\u0442\u0440\u043e\u043d\u0443\u0442\u043e\u0439.<\/p>\n<p>\u0410 \u043a\u043e\u0433\u0434\u0430 \u0441\u0442\u043e\u0438\u0442 \u043e\u043f\u0443\u0441\u043a\u0430\u0442\u044c\u0441\u044f \u0434\u043e <code>Canvas<\/code>? \u0415\u0441\u043b\u0438 \u0432\u0430\u0448 UI \u2014 \u044d\u0442\u043e \u0433\u0440\u0430\u0444\u0438\u043a \u0438\u043b\u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u0447\u0430\u0441\u0442\u0438\u0446 \u0431\u0435\u0437 \u0441\u043b\u043e\u0436\u043d\u043e\u0433\u043e \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u0435\u0433\u043e \u0441\u0442\u0435\u0439\u0442\u0430 \u0438 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0445 \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0445 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432. \u0412 <code>Canvas<\/code> \u043c\u044b \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u043c \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e \u0432 \u0444\u0430\u0437\u0435 Drawing. \u041d\u043e \u0435\u0441\u043b\u0438 \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u044b \u043f\u043e\u043b\u043d\u043e\u0446\u0435\u043d\u043d\u044b\u0435 \u043d\u043e\u0434\u044b Compose (\u043a\u0430\u0440\u0442\u043e\u0447\u043a\u0438, \u043a\u043d\u043e\u043f\u043a\u0438, \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f), \u043d\u0430\u0448\u0435 \u043e\u0440\u0443\u0436\u0438\u0435 \u2014 \u043d\u0438\u0437\u043a\u043e\u0443\u0440\u043e\u0432\u043d\u0435\u0432\u044b\u0439 <code>LazyLayout<\/code>.<\/p>\n<hr\/>\n<h4>\u041f\u0440\u0430\u043a\u0442\u0438\u043a\u0430: \u0411\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u0430\u044f \u0441\u043f\u0438\u0440\u0430\u043b\u044c \u043d\u0430 LazyLayout<\/h4>\n<p><strong>\u0417\u0430\u0434\u0430\u0447\u0430:<\/strong> \u0420\u0430\u0441\u043f\u043e\u043b\u043e\u0436\u0438\u0442\u044c \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b \u043d\u0435 \u0441\u043f\u0438\u0441\u043a\u043e\u043c \u0438 \u043d\u0435 \u0441\u0435\u0442\u043a\u043e\u0439, \u0430 \u043f\u043e \u0431\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u043e\u0439 \u0441\u043f\u0438\u0440\u0430\u043b\u0438.<\/p>\n<p>\u041c\u0430\u0442\u0435\u043c\u0430\u0442\u0438\u043a\u0430 \u0440\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u044f (\u0440\u0430\u0441\u0447\u0435\u0442 \u043a\u043e\u043e\u0440\u0434\u0438\u043d\u0430\u0442 X \u0438 Y \u043f\u043e \u0438\u043d\u0434\u0435\u043a\u0441\u0443 n) \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0442\u0430\u043a:<\/p>\n<pre><code class=\"kotlin\">private fun getSpiralCoordinates(n: Int): Pair&lt;Int, Int&gt; {    if (n == 0) return Pair(0, 0)    val k = ceil((sqrt(n.toDouble() + 1) - 1) \/ 2).toInt()    val t = 2 * k    val m = (2 * k + 1) * (2 * k + 1)    return when {        n &gt;= m - t -&gt; Pair(k - (m - n), -k)        n &gt;= m - 2 * t -&gt; Pair(-k, -k + (m - t - n))        n &gt;= m - 3 * t -&gt; Pair(-k + (m - 2 * t - n), k)        else -&gt; Pair(k, k - (m - 3 * t - n))    }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:87px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0427\u0442\u043e\u0431\u044b \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u043b\u0435\u043d\u0438\u0432\u0443\u044e \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0443 \u043c\u043d\u043e\u0436\u0435\u0441\u0442\u0432\u0430 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 \u0441 \u043f\u043e\u0440\u043e\u0433\u043e\u043c \u0432\u0438\u0434\u0438\u043c\u043e\u0441\u0442\u0438, \u043d\u0430\u043c \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 \u043f\u0440\u043e\u0432\u0430\u0439\u0434\u0435\u0440.<\/p>\n<h3>\u0428\u0430\u0433 1: \u041f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u043a\u0430 State \u0438 Provider<\/h3>\n<p>\u0418\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 <code>LazyLayoutItemProvider<\/code> \u0442\u0440\u0435\u0431\u0443\u0435\u0442 \u043e\u0442 \u043d\u0430\u0441 \u0437\u043d\u0430\u043d\u0438\u044f \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u0430 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 \u0438 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0434\u043b\u044f \u0438\u0445 \u043e\u0442\u0440\u0438\u0441\u043e\u0432\u043a\u0438. \u041d\u043e \u043c\u044b \u043f\u043e\u0439\u0434\u0435\u043c \u0434\u0430\u043b\u044c\u0448\u0435 \u2014 \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e \u0442\u043e\u0447\u043d\u043e \u0437\u043d\u0430\u0442\u044c, \u043a\u0430\u043a\u0438\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b \u043f\u043e\u043f\u0430\u0434\u0430\u044e\u0442 \u0432 Viewport.<\/p>\n<pre><code class=\"kotlin\">typealias ComposableItemContent = @Composable (ListItem) -&gt; Unitdata class LazyLayoutItemContent(    val item: ListItem,    val itemContent: ComposableItemContent)class ItemProvider(    private val itemsState: State&lt;List&lt;LazyLayoutItemContent&gt;&gt;) : LazyLayoutItemProvider {    override val itemCount        get() = itemsState.value.size    @Composable    override fun Item(index: Int, key: Any) {        val item = itemsState.value.getOrNull(index)        item?.itemContent?.invoke(item.item)    }    fun getItemIndexesInRange(boundaries: ViewBoundaries, stepX: Int, stepY: Int): List&lt;Int&gt; {        val result = mutableListOf&lt;Int&gt;()        itemsState.value.forEachIndexed { index, itemContent -&gt;            val item = itemContent.item            val realX = item.coordinates.x * stepX            val realY = item.coordinates.y * stepY            if (realX in boundaries.fromX..boundaries.toX &amp;&amp;                realY in boundaries.fromY..boundaries.toY            ) result.add(index)        }        return result    }        fun getItem(index: Int): ListItem? = itemsState.value.getOrNull(index)?.item}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h3>\u0428\u0430\u0433 2: \u041e\u043f\u0442\u0438\u043c\u0438\u0437\u0430\u0446\u0438\u044f \u0447\u0435\u0440\u0435\u0437 derivedStateOf \u0438 rememberUpdatedState<\/h3>\n<p>\u0427\u0442\u043e\u0431\u044b \u043d\u0430\u0448 <code>ItemProvider<\/code> \u0440\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u043b \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f DSL, \u043d\u043e \u043d\u0435 \u0432\u044b\u0437\u044b\u0432\u0430\u043b \u043b\u0438\u0448\u043d\u0438\u0445 \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439 \u0432\u0441\u0435\u0439 \u043e\u0431\u0435\u0440\u0442\u043a\u0438, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0445\u0438\u0440\u0443\u0440\u0433\u0438\u0447\u0435\u0441\u043a\u0438 \u0442\u043e\u0447\u043d\u043e\u0435 \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435\u043c:<\/p>\n<pre><code class=\"kotlin\">@Composablefun rememberItemProvider(customLazyListScope: CustomLazyListScope.() -&gt; Unit): ItemProvider {    val customLazyListScopeState = rememberUpdatedState(customLazyListScope)    return remember {        ItemProvider(            itemsState = derivedStateOf {                val layoutScope = CustomLazyListScopeImpl().apply(customLazyListScopeState.value)                layoutScope.items            }        )    }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0417\u0434\u0435\u0441\u044c <code>derivedStateOf<\/code> \u2014 \u043d\u0430\u0448 \u043b\u0443\u0447\u0448\u0438\u0439 \u0434\u0440\u0443\u0433. \u041e\u043d \u043a\u044d\u0448\u0438\u0440\u0443\u0435\u0442 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u0439 DSL. \u0415\u0441\u043b\u0438 \u0432\u0445\u043e\u0434\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u043d\u0435 \u0438\u0437\u043c\u0435\u043d\u0438\u043b\u0438\u0441\u044c, Compose \u0434\u0430\u0436\u0435 \u043d\u0435 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u0442 \u0432 \u044d\u0442\u0443 \u0441\u0442\u043e\u0440\u043e\u043d\u0443 \u043d\u0430 \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u043c \u0446\u0438\u043a\u043b\u0435.<\/p>\n<h3>\u0428\u0430\u0433 3: \u0423\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0441\u043a\u0440\u043e\u043b\u043b\u043e\u043c \u0431\u0435\u0437 \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0438<\/h3>\n<p>\u0412 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u043e\u043c Layout \u043c\u044b \u0441\u0430\u043c\u0438 \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0435\u043c \u0436\u0435\u0441\u0442\u044b. \u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c <code>LazyLayoutState<\/code>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0445\u0440\u0430\u043d\u0438\u0442 \u043e\u0444\u0444\u0441\u0435\u0442:<\/p>\n<pre><code class=\"kotlin\">@Stableclass LazyLayoutState(initialOffset: IntOffset = IntOffset.Zero) {    private val _offsetState = mutableStateOf(initialOffset)    val offsetState: State&lt;IntOffset&gt; get() = _offsetState        fun onDrag(offset: IntOffset) {        val x = _offsetState.value.x - offset.x        val y = _offsetState.value.y - offset.y        _offsetState.value = IntOffset(x, y)    }    \/\/ ... \u0440\u0430\u0441\u0447\u0435\u0442 \u0433\u0440\u0430\u043d\u0438\u0446 \u0432\u0438\u0434\u0438\u043c\u043e\u0441\u0442\u0438}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h3>\u0428\u0430\u0433 4: Placement \u2014 \u0442\u0430\u043c, \u0433\u0434\u0435 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u043c\u0430\u0433\u0438\u044f &#171;\u043d\u0443\u043b\u044f&#187;<\/h3>\n<p>\u0421\u0430\u043c\u0430\u044f \u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u0430\u044f \u0447\u0430\u0441\u0442\u044c: \u043c\u044b \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c <code>measure<\/code> \u0442\u043e\u043b\u044c\u043a\u043e \u0434\u043b\u044f \u0432\u0438\u0434\u0438\u043c\u044b\u0445 \u0438\u043d\u0434\u0435\u043a\u0441\u043e\u0432, \u0430 \u0440\u0430\u0441\u0447\u0435\u0442 \u0440\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u044f (placement) \u0434\u0435\u043b\u0430\u0435\u043c &#171;\u043d\u0430 \u043b\u0435\u0442\u0443&#187;.<\/p>\n<pre><code class=\"kotlin\">fun Placeable.PlacementScope.placeItem(    state: LazyLayoutState,    listItem: ListItem,    placeables: List&lt;Placeable&gt;,    gridStepX: Int,    gridStepY: Int) {    val xPosition = (listItem.coordinates.x * gridStepX) - state.offsetState.value.x    val yPosition = (listItem.coordinates.y * gridStepY) - state.offsetState.value.y    placeables.forEach { placeable -&gt;        placeable.placeRelative(xPosition, yPosition)    }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u043e\u0447\u0435\u043c\u0443 \u044d\u0442\u043e \u0434\u0430\u0435\u0442 <strong>0 \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439<\/strong>? \u041c\u044b \u0447\u0438\u0442\u0430\u0435\u043c <code>state.offsetState.value<\/code> \u0432\u043d\u0443\u0442\u0440\u0438 \u043b\u044f\u043c\u0431\u0434\u044b \u0440\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0438 <code>layout()<\/code>. \u041a\u043e\u0433\u0434\u0430 \u043e\u0444\u0444\u0441\u0435\u0442 \u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f (\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442), Compose \u0438\u043d\u0432\u0430\u043b\u0438\u0434\u0438\u0440\u0443\u0435\u0442 <strong>\u0442\u043e\u043b\u044c\u043a\u043e \u0444\u0430\u0437\u0443 Placement<\/strong>. \u0415\u043c\u0443 \u043d\u0435 \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u043d\u043e\u0432\u043e \u043f\u0435\u0440\u0435\u0441\u0447\u0438\u0442\u044b\u0432\u0430\u0442\u044c \u0440\u0430\u0437\u043c\u0435\u0440\u044b \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0445 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 (Measurement), \u0438 \u0443\u0436 \u0442\u0435\u043c \u0431\u043e\u043b\u0435\u0435 \u043f\u0435\u0440\u0435\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c \u0434\u0435\u0440\u0435\u0432\u043e UI (Composition).<\/p>\n<hr\/>\n<h4>\u0411\u0435\u043d\u0447\u043c\u0430\u0440\u043a: Layout Inspector \u043d\u0435 \u0432\u0440\u0435\u0442<\/h4>\n<p>\u0417\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c Layout Inspector \u0438 \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u043c \u0430\u0433\u0440\u0435\u0441\u0441\u0438\u0432\u043d\u043e \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442\u044c \u0441\u043f\u0438\u0440\u0430\u043b\u044c.<\/p>\n<p><strong>\u0420\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442:<\/strong><\/p>\n<figure class=\"\"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/3b9\/50e\/fa4\/3b950efa41e98854e14533ce8873b248.gif\" alt=\"MicroMod preview\" title=\"MicroMod preview\" width=\"358\" height=\"750\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/3b9\/50e\/fa4\/3b950efa41e98854e14533ce8873b248.gif 780w,&#10;       https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/3b9\/50e\/fa4\/3b950efa41e98854e14533ce8873b248.gif 781w\" loading=\"lazy\" decode=\"async\"\/><\/p>\n<div><figcaption><strong>MicroMod preview<\/strong><\/figcaption><\/div>\n<\/figure>\n<h4>Layout Inspector:<\/h4>\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/d98\/166\/d68\/d98166d683b2f479b467d989f3d03a27.gif\" alt=\"Smooth scrolling without unnecessary calculations\" title=\"Smooth scrolling without unnecessary calculations\" width=\"710\" height=\"380\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/d98\/166\/d68\/d98166d683b2f479b467d989f3d03a27.gif 780w,&#10;       https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/d98\/166\/d68\/d98166d683b2f479b467d989f3d03a27.gif 781w\" loading=\"lazy\" decode=\"async\"\/><\/p>\n<div><figcaption><strong>Smooth scrolling without unnecessary calculations<\/strong><\/figcaption><\/div>\n<\/figure>\n<ul>\n<li>\n<p><strong>Recomposition Count: 0.<\/strong><\/p>\n<\/li>\n<li>\n<p><strong>Skipped Count:<\/strong> \u041c\u0430\u043a\u0441\u0438\u043c\u0430\u043b\u044c\u043d\u044b\u0439 \u0434\u043b\u044f \u0432\u0441\u0435\u0445 \u043d\u043e\u0434.<\/p>\n<\/li>\n<\/ul>\n<p>\u0412\u044b\u043d\u043e\u0441 \u043b\u043e\u0433\u0438\u043a\u0438 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u0439 \u0437\u0430 \u043f\u0440\u0435\u0434\u0435\u043b\u044b \u0444\u0430\u0437\u044b \u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0438 \u0438 \u0433\u0440\u0430\u043c\u043e\u0442\u043d\u043e\u0435 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0435\u0439 <code>SubcomposeLayoutState<\/code> (\u043d\u0430 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u0431\u0430\u0437\u0438\u0440\u0443\u0435\u0442\u0441\u044f <code>LazyLayout<\/code>) \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u0434\u043e\u0431\u0438\u0442\u044c\u0441\u044f \u0438\u0434\u0435\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0444\u0440\u0435\u0439\u043c\u0440\u0435\u0439\u0442\u0430 \u0434\u0430\u0436\u0435 \u043d\u0430 \u0441\u043b\u0430\u0431\u044b\u0445 \u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0430\u0445.<\/p>\n<h4>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435<\/h4>\n<p>\u041c\u044b \u0441\u043e\u0431\u0440\u0430\u043b\u0438 \u0432\u044b\u0441\u043e\u043a\u043e\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0439 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442. \u0414\u0430, \u0432 \u043e\u0442\u043b\u0438\u0447\u0438\u0435 \u043e\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f \u0433\u043e\u0442\u043e\u0432\u043e\u0433\u043e <code>LazyColumn<\/code>, \u043d\u0430\u043c \u043f\u0440\u0438\u0448\u043b\u043e\u0441\u044c \u0432\u0440\u0443\u0447\u043d\u0443\u044e \u0441\u0447\u0438\u0442\u0430\u0442\u044c \u0433\u0440\u0430\u043d\u0438\u0446\u044b, \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u0436\u0435\u0441\u0442\u044b \u0438 \u0434\u0438\u0440\u0438\u0436\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0444\u0430\u0437\u0430\u043c\u0438 Measure \u0438 Place. \u041d\u043e \u0438\u043c\u0435\u043d\u043d\u043e \u044d\u0442\u043e \u043e\u0442\u043b\u0438\u0447\u0430\u0435\u0442 \u043e\u0431\u044b\u0447\u043d\u043e\u0433\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u0444\u0440\u0435\u0439\u043c\u0432\u043e\u0440\u043a\u0430 \u043e\u0442 \u0438\u043d\u0436\u0435\u043d\u0435\u0440\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u043e\u043d\u0438\u043c\u0430\u0435\u0442 \u0435\u0433\u043e \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u043e\u0441\u0442\u0438.<\/p>\n<p>\u0411\u0430\u043b\u0430\u043d\u0441 \u043c\u0435\u0436\u0434\u0443 \u0447\u0438\u0441\u0442\u044b\u043c \u043a\u043e\u0434\u043e\u043c \u0438 \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c\u044e \u0432\u0441\u0435\u0433\u0434\u0430 \u0445\u0440\u0443\u043f\u043e\u043a. \u0415\u0441\u043b\u0438 \u0432\u0430\u043c \u043d\u0443\u0436\u0435\u043d \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0439 \u0441\u043f\u0438\u0441\u043e\u043a \u2014 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0439\u0442\u0435 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u044b. \u041d\u043e \u0435\u0441\u043b\u0438 \u0432\u0430\u0448\u0430 \u0437\u0430\u0434\u0430\u0447\u0430, \u043a\u0430\u043a \u0438 \u0432 \u043f\u0440\u043e\u0435\u043a\u0442\u0435 <strong>MicroMod<\/strong> (\u043f\u043e\u0441\u0442\u0440\u043e\u0435\u043d\u043d\u043e\u043c \u043d\u0430 \u043c\u0438\u043a\u0440\u043e\u043c\u043e\u0434\u0443\u043b\u044c\u043d\u043e\u0439 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0435 \u0441 Navigation3), \u0442\u0440\u0435\u0431\u0443\u0435\u0442 \u0431\u0435\u0441\u043a\u043e\u043c\u043f\u0440\u043e\u043c\u0438\u0441\u0441\u043d\u043e\u0439 \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u0438 \u043e\u0442 \u043d\u0435\u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u043e\u0433\u043e UI \u2014 \u0441\u043f\u0443\u0441\u043a\u0430\u0439\u0442\u0435\u0441\u044c \u043d\u0430 \u0443\u0440\u043e\u0432\u0435\u043d\u044c <code>LazyLayout<\/code>. \u041a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u0444\u0430\u0437\u0430\u043c\u0438 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430 \u2014 \u044d\u0442\u043e \u0432\u0430\u0448\u0430 \u0441\u0443\u043f\u0435\u0440\u0441\u0438\u043b\u0430. \u0418\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0439\u0442\u0435 \u0435\u0451 \u0441 \u0443\u043c\u043e\u043c.<\/p>\n<p>\u041f\u043e\u043b\u043d\u0443\u044e \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e \u0441\u043c\u043e\u0442\u0440\u0438\u0442\u0435 \u043d\u0430 <a href=\"https:\/\/github.com\/shamil-RS\/MicroMod\" rel=\"noopener noreferrer nofollow\">Git<\/a><\/p>\n<\/div>\n<p>\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\/1035388\/\">https:\/\/habr.com\/ru\/articles\/1035388\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>Image by\u00a0author\u042d\u0442\u043e \u043f\u0435\u0440\u0435\u0432\u043e\u0434 \u043c\u043e\u0435\u0439 \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0438, \u043e\u043f\u0443\u0431\u043b\u0438\u043a\u043e\u0432\u0430\u043d\u043d\u043e\u0439 \u0440\u0430\u043d\u0435\u0435 \u0432 ProAndroidDev.\u0421\u043e\u0442\u043d\u044f \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443 \u043f\u0440\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0435 \u2014 \u044d\u0442\u043e \u043f\u0440\u0438\u0433\u043e\u0432\u043e\u0440. \u041f\u0440\u0438\u0433\u043e\u0432\u043e\u0440 \u0431\u0430\u0442\u0430\u0440\u0435\u0435 \u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0430, \u043f\u043b\u0430\u0432\u043d\u043e\u0441\u0442\u0438 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0439 \u0438 \u0432\u0430\u0448\u0435\u0439 \u0440\u0435\u043f\u0443\u0442\u0430\u0446\u0438\u0438 \u043a\u0430\u043a \u0438\u043d\u0436\u0435\u043d\u0435\u0440\u0430. \u041c\u044b \u043f\u0440\u0438\u0432\u044b\u043a\u043b\u0438 \u043c\u044b\u0441\u043b\u0438\u0442\u044c \u0432\u044b\u0441\u043e\u043a\u043e\u0443\u0440\u043e\u0432\u043d\u0435\u0432\u044b\u043c\u0438 \u0430\u0431\u0441\u0442\u0440\u0430\u043a\u0446\u0438\u044f\u043c\u0438: \u0437\u0430\u043a\u0438\u043d\u0443\u0442\u044c LazyColumn, \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043f\u0430\u0440\u0443 Modifier.padding \u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u0438\u0442\u044c \u0432 \u043f\u0440\u043e\u0434\u0430\u043a\u0448\u0435\u043d. \u041d\u043e \u0447\u0442\u043e \u0434\u0435\u043b\u0430\u0442\u044c, \u043a\u043e\u0433\u0434\u0430 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u044b \u043d\u0430\u0447\u0438\u043d\u0430\u044e\u0442 &#171;\u0437\u0430\u0445\u043b\u0435\u0431\u044b\u0432\u0430\u0442\u044c\u0441\u044f&#187;, \u0430 Layout Inspector \u0433\u043e\u0440\u0438\u0442 \u043a\u0440\u0430\u0441\u043d\u044b\u043c \u043e\u0442 \u0438\u0437\u0431\u044b\u0442\u043e\u0447\u043d\u044b\u0445 \u043e\u0442\u0440\u0438\u0441\u043e\u0432\u043e\u043a?\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0441\u043d\u0438\u043c\u0435\u043c \u0440\u043e\u0437\u043e\u0432\u044b\u0435 \u043e\u0447\u043a\u0438 \u0430\u0431\u0441\u0442\u0440\u0430\u043a\u0446\u0438\u0439. \u0415\u0441\u043b\u0438 \u0443\u0431\u0440\u0430\u0442\u044c \u0432\u0441\u044e &#171;\u043c\u0430\u0433\u0438\u044e&#187; \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u043e\u0432 Compose, \u043e\u0441\u0442\u0430\u043d\u0443\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e Layout \u0438 Modifier. \u0414\u0430\u0436\u0435 Text \u043f\u043e\u0434 \u043a\u0430\u043f\u043e\u0442\u043e\u043c \u2014 \u044d\u0442\u043e BasicText, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043e\u043f\u0438\u0440\u0430\u0435\u0442\u0441\u044f \u043d\u0430 \u0444\u0443\u043d\u0434\u0430\u043c\u0435\u043d\u0442 Layout.\u0421\u0435\u0433\u043e\u0434\u043d\u044f \u043c\u044b \u0440\u0430\u0437\u0431\u0435\u0440\u0435\u043c \u0445\u0430\u0440\u0434\u043a\u043e\u0440\u043d\u044b\u0439 \u043a\u0435\u0439\u0441 \u0438\u0437 \u043c\u043e\u0435\u0433\u043e \u043f\u0440\u043e\u0435\u043a\u0442\u0430 MicroMod: \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u0431\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u043e\u0439 \u0441\u0435\u0442\u043a\u0438 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432, \u0440\u0430\u0441\u043f\u043e\u043b\u043e\u0436\u0435\u043d\u043d\u044b\u0445 \u043f\u043e \u0441\u043f\u0438\u0440\u0430\u043b\u0438, \u0441 \u0430\u0431\u0441\u043e\u043b\u044e\u0442\u043d\u043e \u043d\u0443\u043b\u0435\u0432\u044b\u043c \u043f\u043e\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u0435\u043c \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439 \u043f\u0440\u0438 \u0430\u043a\u0442\u0438\u0432\u043d\u043e\u043c \u0441\u043a\u0440\u043e\u043b\u043b\u0435. \u041d\u0438\u043a\u0430\u043a\u043e\u0439 \u0435\u0440\u0443\u043d\u0434\u044b \u0432 \u0434\u0443\u0445\u0435 &#171;\u043f\u0440\u043e\u0441\u0442\u043e \u0434\u043e\u0431\u0430\u0432\u044c @Stable&#187;. \u0422\u043e\u043b\u044c\u043a\u043e \u0433\u043b\u0443\u0431\u043e\u043a\u0430\u044f \u0440\u0430\u0431\u043e\u0442\u0430 \u0441 \u0444\u0430\u0437\u0430\u043c\u0438 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430, \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 LazyLayout \u0438 \u0443\u043c\u043d\u043e\u0435 \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435\u043c.\u041f\u0440\u0435\u0436\u0434\u0435 \u0447\u0435\u043c \u043f\u0435\u0440\u0435\u0439\u0442\u0438 \u043a \u043f\u0440\u0430\u043a\u0442\u0438\u043a\u0435, \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u043a\u0440\u0435\u043f\u0438\u0442\u044c \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u044e\u044e \u043c\u0435\u0445\u0430\u043d\u0438\u043a\u0443 \u0444\u0430\u0437\u044b Layout. \u041f\u0440\u043e\u0446\u0435\u0441\u0441 \u043a\u043e\u043c\u043f\u043e\u043d\u043e\u0432\u043a\u0438 \u0441\u043e\u0441\u0442\u043e\u0438\u0442 \u0438\u0437 \u0442\u0440\u0435\u0445 \u0448\u0430\u0433\u043e\u0432: Constraints (\u041e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u044f), Measurement (\u0418\u0437\u043c\u0435\u0440\u0435\u043d\u0438\u0435) \u0438 Placement (\u0420\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u0435). \u0412\u0435\u0447\u043d\u044b\u0439 \u0441\u043f\u043e\u0440 \u043e \u0442\u043e\u043c, \u043f\u0435\u0440\u0435\u0434\u0430\u0435\u0442\u0441\u044f \u043b\u0438 \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 &#171;\u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437&#187; \u0438\u043b\u0438 &#171;\u0441\u043d\u0438\u0437\u0443 \u0432\u0432\u0435\u0440\u0445&#187;, \u043d\u0435 \u0438\u043c\u0435\u0435\u0442 \u043e\u0434\u043d\u043e\u0437\u043d\u0430\u0447\u043d\u043e\u0433\u043e \u043e\u0442\u0432\u0435\u0442\u0430, \u0442\u0430\u043a \u043a\u0430\u043a \u044d\u0442\u0438 \u044d\u0442\u0430\u043f\u044b \u0432\u0437\u0430\u0438\u043c\u043e\u0441\u0432\u044f\u0437\u0430\u043d\u044b:Constraints \u043f\u0435\u0440\u0435\u0434\u0430\u044e\u0442\u0441\u044f \u043e\u0442 \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044f \u043a \u0434\u0435\u0442\u044f\u043c (\u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437).Measurement (\u0440\u0430\u0441\u0447\u0435\u0442 \u0440\u0430\u0437\u043c\u0435\u0440\u043e\u0432) \u0438\u0434\u0435\u0442 \u0432 \u043e\u0431\u0440\u0430\u0442\u043d\u043e\u043c \u043f\u043e\u0440\u044f\u0434\u043a\u0435 (\u0441\u043d\u0438\u0437\u0443 \u0432\u0432\u0435\u0440\u0445).Placement (\u043f\u043e\u0437\u0438\u0446\u0438\u043e\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435) \u0444\u0438\u043d\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0441\u043d\u043e\u0432\u0430 \u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437.\u0410\u043d\u0430\u0442\u043e\u043c\u0438\u044f \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430: \u043f\u043e\u0447\u0435\u043c\u0443 Compose \u043b\u0430\u0433\u0430\u0435\u0442\u041e\u0442\u0440\u0438\u0441\u043e\u0432\u043a\u0430 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430 \u0432 Compose \u043f\u0440\u043e\u0445\u043e\u0434\u0438\u0442 \u0447\u0435\u0440\u0435\u0437 \u0442\u0440\u0438 \u0441\u0442\u0440\u043e\u0433\u043e \u0440\u0435\u0433\u043b\u0430\u043c\u0435\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0445 \u044d\u0442\u0430\u043f\u0430:Composition: \u0427\u0442\u043e \u043c\u044b \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c? Compose \u0441\u0442\u0440\u043e\u0438\u0442 \u0434\u0435\u0440\u0435\u0432\u043e UI.Layout: \u0413\u0434\u0435 \u044d\u0442\u043e \u043d\u0430\u0445\u043e\u0434\u0438\u0442\u0441\u044f? \u0412\u043a\u043b\u044e\u0447\u0430\u0435\u0442 \u0448\u0430\u0433\u0438 Measurement (\u0441\u043d\u0438\u0437\u0443 \u0432\u0432\u0435\u0440\u0445) \u0438 Placement (\u0441\u0432\u0435\u0440\u0445\u0443 \u0432\u043d\u0438\u0437).Drawing: \u041a\u0430\u043a \u044d\u0442\u043e \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442? \u041e\u0442\u0440\u0438\u0441\u043e\u0432\u043a\u0430 \u043f\u0438\u043a\u0441\u0435\u043b\u0435\u0439 \u043d\u0430 \u044d\u043a\u0440\u0430\u043d\u0435.\u0411\u043e\u043b\u044c\u0448\u0438\u043d\u0441\u0442\u0432\u043e \u043f\u0440\u043e\u0431\u043b\u0435\u043c \u0441 \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c\u044e (\u0442\u0435 \u0441\u0430\u043c\u044b\u0435 \u043b\u0430\u0433\u0438 \u043f\u0440\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0435) \u0432\u043e\u0437\u043d\u0438\u043a\u0430\u044e\u0442, \u043a\u043e\u0433\u0434\u0430 \u043c\u044b \u0437\u0430\u0441\u0442\u0430\u0432\u043b\u044f\u0435\u043c Compose \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0442\u044c\u0441\u044f \u043d\u0430 \u0444\u0430\u0437\u0443 Composition \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u043f\u0438\u043a\u0441\u0435\u043b\u044f. \u0414\u0430\u0436\u0435 \u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c\u044b \u0432\u0440\u043e\u0434\u0435 Donut-hole skipping (\u043f\u0440\u043e\u043f\u0443\u0441\u043a \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0438 \u0432\u043d\u0443\u0442\u0440\u0438 \u0444\u0443\u043d\u043a\u0446\u0438\u0438, \u0435\u0441\u043b\u0438 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u043d\u0435 \u0438\u0437\u043c\u0435\u043d\u0438\u043b\u0438\u0441\u044c) \u0438\u043b\u0438 \u043d\u043e\u0432\u044b\u0439 Strong Skipping Mode \u043d\u0435 \u0441\u043f\u0430\u0441\u0443\u0442, \u0435\u0441\u043b\u0438 \u0432\u044b \u0447\u0438\u0442\u0430\u0435\u0442\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u0441\u043a\u0440\u043e\u043b\u043b\u0430 \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e \u0432 \u0442\u0435\u043b\u0435 @Composable \u0444\u0443\u043d\u043a\u0446\u0438\u0438.\u041b\u044f\u043c\u0431\u0434\u0430-\u043c\u043e\u0434\u0438\u0444\u0438\u043a\u0430\u0442\u043e\u0440\u044b \u0438 \u0441\u043f\u0443\u0441\u043a \u0434\u043e Canvas\u041a\u043b\u0430\u0441\u0441\u0438\u0447\u0435\u0441\u043a\u0438\u0439 \u043f\u0440\u0438\u043c\u0435\u0440 \u0434\u0435\u043b\u0435\u0433\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u0439: \u0432\u043c\u0435\u0441\u0442\u043e Modifier.offset(x = scrollState.value), \u0433\u0440\u0430\u043c\u043e\u0442\u043d\u044b\u0439 \u0438\u043d\u0436\u0435\u043d\u0435\u0440 \u043d\u0430\u043f\u0438\u0448\u0435\u0442 Modifier.offset { IntOffset(x = scrollState.value, 0) }. \u041b\u044f\u043c\u0431\u0434\u0430 \u043e\u0442\u043a\u043b\u0430\u0434\u044b\u0432\u0430\u0435\u0442 \u0447\u0442\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f \u0434\u043e \u0444\u0430\u0437\u044b Layout (\u0430 \u0438\u043c\u0435\u043d\u043d\u043e \u2014 Placement). \u0424\u0430\u0437\u0430 Composition \u043f\u0440\u0438 \u044d\u0442\u043e\u043c \u043e\u0441\u0442\u0430\u0435\u0442\u0441\u044f \u043d\u0435\u0442\u0440\u043e\u043d\u0443\u0442\u043e\u0439.\u0410 \u043a\u043e\u0433\u0434\u0430 \u0441\u0442\u043e\u0438\u0442 \u043e\u043f\u0443\u0441\u043a\u0430\u0442\u044c\u0441\u044f \u0434\u043e Canvas? \u0415\u0441\u043b\u0438 \u0432\u0430\u0448 UI \u2014 \u044d\u0442\u043e \u0433\u0440\u0430\u0444\u0438\u043a \u0438\u043b\u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u0430 \u0447\u0430\u0441\u0442\u0438\u0446 \u0431\u0435\u0437 \u0441\u043b\u043e\u0436\u043d\u043e\u0433\u043e \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u0435\u0433\u043e \u0441\u0442\u0435\u0439\u0442\u0430 \u0438 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0445 \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0445 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432. \u0412 Canvas \u043c\u044b \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u043c \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e \u0432 \u0444\u0430\u0437\u0435 Drawing. \u041d\u043e \u0435\u0441\u043b\u0438 \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u044b \u043f\u043e\u043b\u043d\u043e\u0446\u0435\u043d\u043d\u044b\u0435 \u043d\u043e\u0434\u044b Compose (\u043a\u0430\u0440\u0442\u043e\u0447\u043a\u0438, \u043a\u043d\u043e\u043f\u043a\u0438, \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f), \u043d\u0430\u0448\u0435 \u043e\u0440\u0443\u0436\u0438\u0435 \u2014 \u043d\u0438\u0437\u043a\u043e\u0443\u0440\u043e\u0432\u043d\u0435\u0432\u044b\u0439 LazyLayout.\u041f\u0440\u0430\u043a\u0442\u0438\u043a\u0430: \u0411\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u0430\u044f \u0441\u043f\u0438\u0440\u0430\u043b\u044c \u043d\u0430 LazyLayout\u0417\u0430\u0434\u0430\u0447\u0430: \u0420\u0430\u0441\u043f\u043e\u043b\u043e\u0436\u0438\u0442\u044c \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b \u043d\u0435 \u0441\u043f\u0438\u0441\u043a\u043e\u043c \u0438 \u043d\u0435 \u0441\u0435\u0442\u043a\u043e\u0439, \u0430 \u043f\u043e \u0431\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u043e\u0439 \u0441\u043f\u0438\u0440\u0430\u043b\u0438.\u041c\u0430\u0442\u0435\u043c\u0430\u0442\u0438\u043a\u0430 \u0440\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u044f (\u0440\u0430\u0441\u0447\u0435\u0442 \u043a\u043e\u043e\u0440\u0434\u0438\u043d\u0430\u0442 X \u0438 Y \u043f\u043e \u0438\u043d\u0434\u0435\u043a\u0441\u0443 n) \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0442\u0430\u043a:private fun getSpiralCoordinates(n: Int): Pair&lt;Int, Int&gt; {    if (n == 0) return Pair(0, 0)    val k = ceil((sqrt(n.toDouble() + 1) &#8212; 1) \/ 2).toInt()    val t = 2 * k    val m = (2 * k + 1) * (2 * k + 1)    return when {        n &gt;= m &#8212; t -&gt; Pair(k &#8212; (m &#8212; n), -k)        n &gt;= m &#8212; 2 * t -&gt; Pair(-k, -k + (m &#8212; t &#8212; n))        n &gt;= m &#8212; 3 * t -&gt; Pair(-k + (m &#8212; 2 * t &#8212; n), k)        else -&gt; Pair(k, k &#8212; (m &#8212; 3 * t &#8212; n))    }}\u0427\u0442\u043e\u0431\u044b \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u043b\u0435\u043d\u0438\u0432\u0443\u044e \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0443 \u043c\u043d\u043e\u0436\u0435\u0441\u0442\u0432\u0430 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 \u0441 \u043f\u043e\u0440\u043e\u0433\u043e\u043c \u0432\u0438\u0434\u0438\u043c\u043e\u0441\u0442\u0438, \u043d\u0430\u043c \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 \u043f\u0440\u043e\u0432\u0430\u0439\u0434\u0435\u0440.\u0428\u0430\u0433 1: \u041f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u043a\u0430 State \u0438 Provider\u0418\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 LazyLayoutItemProvider \u0442\u0440\u0435\u0431\u0443\u0435\u0442 \u043e\u0442 \u043d\u0430\u0441 \u0437\u043d\u0430\u043d\u0438\u044f \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u0430 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 \u0438 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0434\u043b\u044f \u0438\u0445 \u043e\u0442\u0440\u0438\u0441\u043e\u0432\u043a\u0438. \u041d\u043e \u043c\u044b \u043f\u043e\u0439\u0434\u0435\u043c \u0434\u0430\u043b\u044c\u0448\u0435 \u2014 \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e \u0442\u043e\u0447\u043d\u043e \u0437\u043d\u0430\u0442\u044c, \u043a\u0430\u043a\u0438\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b \u043f\u043e\u043f\u0430\u0434\u0430\u044e\u0442 \u0432 Viewport.typealias ComposableItemContent = @Composable (ListItem) -&gt; Unitdata class LazyLayoutItemContent(    val item: ListItem,    val itemContent: ComposableItemContent)class ItemProvider(    private val itemsState: State&lt;List&lt;LazyLayoutItemContent&gt;&gt;) : LazyLayoutItemProvider {    override val itemCount        get() = itemsState.value.size    @Composable    override fun Item(index: Int, key: Any) {        val item = itemsState.value.getOrNull(index)        item?.itemContent?.invoke(item.item)    }    fun getItemIndexesInRange(boundaries: ViewBoundaries, stepX: Int, stepY: Int): List&lt;Int&gt; {        val result = mutableListOf&lt;Int&gt;()        itemsState.value.forEachIndexed { index, itemContent -&gt;            val item = itemContent.item            val realX = item.coordinates.x * stepX            val realY = item.coordinates.y * stepY            if (realX in boundaries.fromX..boundaries.toX &amp;&amp;                realY in boundaries.fromY..boundaries.toY            ) result.add(index)        }        return result    }        fun getItem(index: Int): ListItem? = itemsState.value.getOrNull(index)?.item}\u0428\u0430\u0433 2: \u041e\u043f\u0442\u0438\u043c\u0438\u0437\u0430\u0446\u0438\u044f \u0447\u0435\u0440\u0435\u0437 derivedStateOf \u0438 rememberUpdatedState\u0427\u0442\u043e\u0431\u044b \u043d\u0430\u0448 ItemProvider \u0440\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u043b \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f DSL, \u043d\u043e \u043d\u0435 \u0432\u044b\u0437\u044b\u0432\u0430\u043b \u043b\u0438\u0448\u043d\u0438\u0445 \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439 \u0432\u0441\u0435\u0439 \u043e\u0431\u0435\u0440\u0442\u043a\u0438, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0445\u0438\u0440\u0443\u0440\u0433\u0438\u0447\u0435\u0441\u043a\u0438 \u0442\u043e\u0447\u043d\u043e\u0435 \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435\u043c:@Composablefun rememberItemProvider(customLazyListScope: CustomLazyListScope.() -&gt; Unit): ItemProvider {    val customLazyListScopeState = rememberUpdatedState(customLazyListScope)    return remember {        ItemProvider(            itemsState = derivedStateOf {                val layoutScope = CustomLazyListScopeImpl().apply(customLazyListScopeState.value)                layoutScope.items            }        )    }}\u0417\u0434\u0435\u0441\u044c derivedStateOf \u2014 \u043d\u0430\u0448 \u043b\u0443\u0447\u0448\u0438\u0439 \u0434\u0440\u0443\u0433. \u041e\u043d \u043a\u044d\u0448\u0438\u0440\u0443\u0435\u0442 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u0439 DSL. \u0415\u0441\u043b\u0438 \u0432\u0445\u043e\u0434\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u043d\u0435 \u0438\u0437\u043c\u0435\u043d\u0438\u043b\u0438\u0441\u044c, Compose \u0434\u0430\u0436\u0435 \u043d\u0435 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u0442 \u0432 \u044d\u0442\u0443 \u0441\u0442\u043e\u0440\u043e\u043d\u0443 \u043d\u0430 \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u043c \u0446\u0438\u043a\u043b\u0435.\u0428\u0430\u0433 3: \u0423\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0441\u043a\u0440\u043e\u043b\u043b\u043e\u043c \u0431\u0435\u0437 \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0438\u0412 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u043e\u043c Layout \u043c\u044b \u0441\u0430\u043c\u0438 \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0435\u043c \u0436\u0435\u0441\u0442\u044b. \u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c LazyLayoutState, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0445\u0440\u0430\u043d\u0438\u0442 \u043e\u0444\u0444\u0441\u0435\u0442:@Stableclass LazyLayoutState(initialOffset: IntOffset = IntOffset.Zero) {    private val _offsetState = mutableStateOf(initialOffset)    val offsetState: State&lt;IntOffset&gt; get() = _offsetState        fun onDrag(offset: IntOffset) {        val x = _offsetState.value.x &#8212; offset.x        val y = _offsetState.value.y &#8212; offset.y        _offsetState.value = IntOffset(x, y)    }    \/\/ &#8230; \u0440\u0430\u0441\u0447\u0435\u0442 \u0433\u0440\u0430\u043d\u0438\u0446 \u0432\u0438\u0434\u0438\u043c\u043e\u0441\u0442\u0438}\u0428\u0430\u0433 4: Placement \u2014 \u0442\u0430\u043c, \u0433\u0434\u0435 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u043c\u0430\u0433\u0438\u044f &#171;\u043d\u0443\u043b\u044f&#187;\u0421\u0430\u043c\u0430\u044f \u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u0430\u044f \u0447\u0430\u0441\u0442\u044c: \u043c\u044b \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c measure \u0442\u043e\u043b\u044c\u043a\u043e \u0434\u043b\u044f \u0432\u0438\u0434\u0438\u043c\u044b\u0445 \u0438\u043d\u0434\u0435\u043a\u0441\u043e\u0432, \u0430 \u0440\u0430\u0441\u0447\u0435\u0442 \u0440\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u044f (placement) \u0434\u0435\u043b\u0430\u0435\u043c &#171;\u043d\u0430 \u043b\u0435\u0442\u0443&#187;.fun Placeable.PlacementScope.placeItem(    state: LazyLayoutState,    listItem: ListItem,    placeables: List&lt;Placeable&gt;,    gridStepX: Int,    gridStepY: Int) {    val xPosition = (listItem.coordinates.x * gridStepX) &#8212; state.offsetState.value.x    val yPosition = (listItem.coordinates.y * gridStepY) &#8212; state.offsetState.value.y    placeables.forEach { placeable -&gt;        placeable.placeRelative(xPosition, yPosition)    }}\u041f\u043e\u0447\u0435\u043c\u0443 \u044d\u0442\u043e \u0434\u0430\u0435\u0442 0 \u0440\u0435\u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0439? \u041c\u044b \u0447\u0438\u0442\u0430\u0435\u043c state.offsetState.value \u0432\u043d\u0443\u0442\u0440\u0438 \u043b\u044f\u043c\u0431\u0434\u044b \u0440\u0430\u0437\u043c\u0435\u0449\u0435\u043d\u0438\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u0438 layout(). \u041a\u043e\u0433\u0434\u0430 \u043e\u0444\u0444\u0441\u0435\u0442 \u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f (\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442), Compose \u0438\u043d\u0432\u0430\u043b\u0438\u0434\u0438\u0440\u0443\u0435\u0442 \u0442\u043e\u043b\u044c\u043a\u043e \u0444\u0430\u0437\u0443 Placement. \u0415\u043c\u0443 \u043d\u0435 \u043d\u0443\u0436\u043d\u043e \u0437\u0430\u043d\u043e\u0432\u043e \u043f\u0435\u0440\u0435\u0441\u0447\u0438\u0442\u044b\u0432\u0430\u0442\u044c \u0440\u0430\u0437\u043c\u0435\u0440\u044b \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0445 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 (Measurement), \u0438 \u0443\u0436 \u0442\u0435\u043c \u0431\u043e\u043b\u0435\u0435 \u043f\u0435\u0440\u0435\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c \u0434\u0435\u0440\u0435\u0432\u043e UI (Composition).\u0411\u0435\u043d\u0447\u043c\u0430\u0440\u043a: Layout Inspector \u043d\u0435 \u0432\u0440\u0435\u0442\u0417\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c Layout Inspector \u0438 \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u043c \u0430\u0433\u0440\u0435\u0441\u0441\u0438\u0432\u043d\u043e \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442\u044c \u0441\u043f\u0438\u0440\u0430\u043b\u044c.\u0420\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442:MicroMod previewLayout Inspector:Smooth scrolling without unnecessary calculationsRecomposition Count: 0.Skipped Count: \u041c\u0430\u043a\u0441\u0438\u043c\u0430\u043b\u044c\u043d\u044b\u0439 \u0434\u043b\u044f \u0432\u0441\u0435\u0445 \u043d\u043e\u0434.\u0412\u044b\u043d\u043e\u0441 \u043b\u043e\u0433\u0438\u043a\u0438 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u0439 \u0437\u0430 \u043f\u0440\u0435\u0434\u0435\u043b\u044b \u0444\u0430\u0437\u044b \u043a\u043e\u043c\u043f\u043e\u0437\u0438\u0446\u0438\u0438 \u0438 \u0433\u0440\u0430\u043c\u043e\u0442\u043d\u043e\u0435 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0435\u0439 SubcomposeLayoutState (\u043d\u0430 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u0431\u0430\u0437\u0438\u0440\u0443\u0435\u0442\u0441\u044f LazyLayout) \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u0434\u043e\u0431\u0438\u0442\u044c\u0441\u044f \u0438\u0434\u0435\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0444\u0440\u0435\u0439\u043c\u0440\u0435\u0439\u0442\u0430 \u0434\u0430\u0436\u0435 \u043d\u0430 \u0441\u043b\u0430\u0431\u044b\u0445 \u0443\u0441\u0442\u0440\u043e\u0439\u0441\u0442\u0432\u0430\u0445.\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435\u041c\u044b \u0441\u043e\u0431\u0440\u0430\u043b\u0438 \u0432\u044b\u0441\u043e\u043a\u043e\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0439 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442. \u0414\u0430, \u0432 \u043e\u0442\u043b\u0438\u0447\u0438\u0435 \u043e\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f \u0433\u043e\u0442\u043e\u0432\u043e\u0433\u043e LazyColumn, \u043d\u0430\u043c \u043f\u0440\u0438\u0448\u043b\u043e\u0441\u044c \u0432\u0440\u0443\u0447\u043d\u0443\u044e \u0441\u0447\u0438\u0442\u0430\u0442\u044c \u0433\u0440\u0430\u043d\u0438\u0446\u044b, \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u0436\u0435\u0441\u0442\u044b \u0438 \u0434\u0438\u0440\u0438\u0436\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0444\u0430\u0437\u0430\u043c\u0438 Measure \u0438 Place. \u041d\u043e \u0438\u043c\u0435\u043d\u043d\u043e \u044d\u0442\u043e \u043e\u0442\u043b\u0438\u0447\u0430\u0435\u0442 \u043e\u0431\u044b\u0447\u043d\u043e\u0433\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u0444\u0440\u0435\u0439\u043c\u0432\u043e\u0440\u043a\u0430 \u043e\u0442 \u0438\u043d\u0436\u0435\u043d\u0435\u0440\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u043e\u043d\u0438\u043c\u0430\u0435\u0442 \u0435\u0433\u043e \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u043e\u0441\u0442\u0438.\u0411\u0430\u043b\u0430\u043d\u0441 \u043c\u0435\u0436\u0434\u0443 \u0447\u0438\u0441\u0442\u044b\u043c \u043a\u043e\u0434\u043e\u043c \u0438 \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c\u044e \u0432\u0441\u0435\u0433\u0434\u0430 \u0445\u0440\u0443\u043f\u043e\u043a. \u0415\u0441\u043b\u0438 \u0432\u0430\u043c \u043d\u0443\u0436\u0435\u043d \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0439 \u0441\u043f\u0438\u0441\u043e\u043a \u2014 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0439\u0442\u0435 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u044b. \u041d\u043e \u0435\u0441\u043b\u0438 \u0432\u0430\u0448\u0430 \u0437\u0430\u0434\u0430\u0447\u0430, \u043a\u0430\u043a \u0438 \u0432 \u043f\u0440\u043e\u0435\u043a\u0442\u0435 MicroMod (\u043f\u043e\u0441\u0442\u0440\u043e\u0435\u043d\u043d\u043e\u043c \u043d\u0430 \u043c\u0438\u043a\u0440\u043e\u043c\u043e\u0434\u0443\u043b\u044c\u043d\u043e\u0439 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0435 \u0441 Navigation3), \u0442\u0440\u0435\u0431\u0443\u0435\u0442 \u0431\u0435\u0441\u043a\u043e\u043c\u043f\u0440\u043e\u043c\u0438\u0441\u0441\u043d\u043e\u0439 \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u0438 \u043e\u0442 \u043d\u0435\u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u043e\u0433\u043e UI \u2014 \u0441\u043f\u0443\u0441\u043a\u0430\u0439\u0442\u0435\u0441\u044c \u043d\u0430 \u0443\u0440\u043e\u0432\u0435\u043d\u044c LazyLayout. \u041a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u0444\u0430\u0437\u0430\u043c\u0438 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430 \u2014 \u044d\u0442\u043e \u0432\u0430\u0448\u0430 \u0441\u0443\u043f\u0435\u0440\u0441\u0438\u043b\u0430. \u0418\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0439\u0442\u0435 \u0435\u0451 \u0441 \u0443\u043c\u043e\u043c.\u041f\u043e\u043b\u043d\u0443\u044e \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e \u0441\u043c\u043e\u0442\u0440\u0438\u0442\u0435 \u043d\u0430 Git\u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 https:\/\/habr.com\/ru\/articles\/1035388\/<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-479834","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/479834","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=479834"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/479834\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=479834"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=479834"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=479834"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}