{"id":446016,"date":"2025-01-24T21:00:31","date_gmt":"2025-01-24T21:00:31","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=446016"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=446016","title":{"rendered":"<span>\u041f\u0438\u0448\u0435\u043c \u0438\u0434\u0435\u0430\u043b\u044c\u043d\u0443\u044e mobile-first \u0433\u0430\u043b\u0435\u0440\u0435\u044e<\/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<h3>\u041f\u0430\u0440\u0430 \u0441\u043b\u043e\u0432 \u043e \u0441\u0435\u0431\u0435<\/h3>\n<p>\u041f\u0440\u0438\u0432\u0435\u0442, \u043c\u0435\u043d\u044f \u0437\u043e\u0432\u0443\u0442 \u0410\u0440\u0442\u0443\u0440, \u0438 \u044f \u043b\u044e\u0431\u043b\u044e \u043f\u043b\u0430\u0432\u043d\u044b\u0435 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u044b, CSS \u0438 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u043f\u043e\u0434\u0445\u043e\u0434\u044b. \u0421\u0435\u0433\u043e\u0434\u043d\u044f \u0445\u043e\u0447\u0443 \u043f\u043e\u0433\u043e\u0432\u043e\u0440\u0438\u0442\u044c \u0441 \u0432\u0430\u043c\u0438 \u043e \u0437\u0430\u0434\u0430\u0447\u0435, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u043d\u0430\u0441\u0442\u0438\u0433\u0430\u0435\u0442, \u043f\u043e\u0436\u0430\u043b\u0443\u0439, \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0444\u0440\u043e\u043d\u0442\u0435\u043d\u0434 \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0430: \u043e \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u0438 \u0433\u0430\u043b\u0435\u0440\u0435\u0438.<\/p>\n<h3>\u041f\u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u0437\u0430\u0434\u0430\u0447\u0438<\/h3>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0442\u0430\u043a, \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e \u044d\u0442\u043e \u0438 \u043d\u0435 \u0433\u0430\u043b\u0435\u0440\u0435\u044f \u0432\u043e\u0432\u0441\u0435. \u041d\u0430\u0437\u0432\u0430\u043d\u0438\u0435 \u043d\u0435 \u0442\u0430\u043a \u0432\u0430\u0436\u043d\u043e, \u0432\u0430\u0436\u043d\u0430 \u0437\u0430\u0434\u0430\u0447\u0430. \u041e\u043f\u0438\u0441\u0430\u0442\u044c \u0435\u0451 \u043c\u043e\u0436\u043d\u043e \u0442\u0430\u043a: \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u0430 mobile-first \u0441\u0432\u0430\u0439\u043f\u0430\u043b\u043a\u0430 \u0441\u043b\u0430\u0439\u0434\u043e\u0432. \u0421\u0432\u0430\u0439\u043f\u044b \u0434\u043e\u043b\u0436\u043d\u044b \u0432\u044b\u0433\u043b\u044f\u0434\u0435\u0442\u044c \u043c\u0430\u043a\u0441\u0438\u043c\u0430\u043b\u044c\u043d\u043e \u043d\u0430\u0442\u0438\u0432\u043d\u043e \u0438 \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0431\u0435\u0437 \u0441\u044a\u0435\u0434\u0430\u043d\u0438\u044f \u043a\u0430\u0434\u0440\u043e\u0432. \u0425\u043e\u0442\u044c \u043e\u043d\u0430 \u0438 mobile-first, \u0434\u0435\u0441\u043a\u0442\u043e\u043f\u044b \u0441 \u0438\u0445 \u043c\u044b\u0448\u043a\u0430\u043c\u0438 \u0441\u0442\u043e\u0440\u043e\u043d\u043e\u0439 \u043e\u0431\u0445\u043e\u0434\u0438\u0442\u044c \u043d\u0435\u043b\u044c\u0437\u044f.<\/p>\n<h3>\u041a\u0430\u043a \u044d\u0442\u043e \u0434\u0435\u043b\u0430\u043b\u043e\u0441\u044c \u0432\u0435\u043a\u0430\u043c\u0438<\/h3>\n<p>\u0414\u043e\u043b\u0433\u0438\u0435 \u0433\u043e\u0434\u044b, \u0443\u0436\u0435 \u0434\u0430\u0436\u0435 \u0434\u0435\u0441\u044f\u0442\u0438\u043b\u0435\u0442\u0438\u044f, \u043f\u0440\u0438 \u043f\u0440\u043e\u0438\u0437\u043d\u0435\u0441\u0435\u043d\u0438\u0438 \u0434\u0438\u0437\u0430\u0439\u043d\u0435\u0440\u043e\u043c \u0441\u043b\u043e\u0432\u0430 \u00ab\u0441\u0432\u0430\u0439\u043f\u00bb, \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u0441\u0440\u0430\u0437\u0443 \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u0433\u0440\u0443\u0441\u0442\u043d\u0435\u043b, \u043e\u0441\u043e\u0437\u043d\u0430\u0432\u0430\u044f, \u0447\u0442\u043e \u0441\u0435\u0439\u0447\u0430\u0441 \u043f\u0440\u0438\u0434\u0435\u0442\u0441\u044f \u0438\u043c\u0435\u0442\u044c \u0434\u0435\u043b\u043e \u0441\u043e \u0432\u0441\u044f\u043a\u043e\u0433\u043e \u0440\u043e\u0434\u0430 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u044f\u043c\u0438 \u043d\u0430 JS, \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0441 touch-\u0441\u043e\u0431\u044b\u0442\u0438\u044f\u043c\u0438 \u0438 \u0432\u0441\u043f\u043e\u043c\u0438\u043d\u0430\u0442\u044c \u043a\u0430\u043a \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e position \u0432 CSS (\u0434\u0435\u0441\u044f\u0442\u043a\u0438 \u0441\u043e\u0431\u0435\u0441\u043e\u0432 \u0443\u0431\u0435\u0434\u0438\u043b\u0438 \u043c\u0435\u043d\u044f \u0432 \u0442\u043e\u043c, \u0447\u0442\u043e \u0434\u0430\u043b\u0435\u043a\u043e \u043d\u0435 \u0432\u0441\u0435 \u043f\u043e\u043d\u0438\u043c\u0430\u044e\u0442, \u043a\u0430\u043a \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u044d\u0442\u043e \u043d\u0430 \u043f\u0435\u0440\u0432\u044b\u0439 \u0432\u0437\u0433\u043b\u044f\u0434 \u043f\u0440\u043e\u0441\u0442\u043e\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e).<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0434\u0438\u0437\u0430\u0439\u043d\u0435\u0440 \u043b\u044e\u0442\u043e\u0432\u0430\u043b, \u0442\u043e \u043d\u0443\u0436\u043d\u043e \u0431\u044b\u043b\u043e \u0435\u0449\u0435 \u0438 \u0438\u0437\u0443\u0447\u0430\u0442\u044c \u043d\u0435\u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0437\u0430\u043a\u043e\u043d\u044b \u0444\u0438\u0437\u0438\u043a\u0438, \u043f\u044b\u0442\u0430\u044f\u0441\u044c \u043f\u043e\u043d\u044f\u0442\u044c, \u043a\u0430\u043a \u043d\u0430 JS \u0432\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u0435\u0441\u0442\u0438 \u044d\u0444\u0444\u0435\u043a\u0442 \u043f\u0440\u0443\u0436\u0438\u043d\u044b: \u0438\u043c\u0435\u043d\u043d\u043e \u0442\u0430\u043a \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0434\u043e\u0448\u0435\u0434\u0448\u0438\u0439 \u0434\u043e \u0433\u0440\u0430\u043d\u0438\u0446\u044b \u0441\u0432\u0430\u0439\u043f \u043d\u0430 \u043f\u043e\u0447\u0442\u0438 \u0432\u0441\u0435\u0445 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u0441\u043c\u0430\u0440\u0442\u0444\u043e\u043d\u0430\u0445.<\/p>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e2f79e4767f9b8c8d14c\" data-style=\"\" id=\"6792e2f79e4767f9b8c8d14c\" width=\"\"><\/div>\n<p>Anyway, \u0432\u043e\u043e\u0440\u0443\u0436\u0438\u0432\u0448\u0438\u0441\u044c \u0437\u0430\u043a\u043e\u043d\u0430\u043c\u0438 \u0434\u0438\u043d\u0430\u043c\u0438\u043a\u0438, \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430\u043c\u0438 position\/left\/right\/transform \u0438 \u0441\u043e\u0431\u044b\u0442\u0438\u044f\u043c\u0438 \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430 pointerevents, \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438 \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u044e\u0442\u0441\u044f \u0437\u0430 \u0434\u0435\u043b\u043e. \u0421\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0435\u043c, \u043e\u0434\u0438\u043d \u0438\u0437 \u043d\u0438\u0445 \u043f\u043b\u044e\u043d\u0435\u0442 \u0438 \u0437\u0430\u0433\u0443\u0433\u043b\u0438\u0442 \u00abthe most popular swipeable gallery on JS\u00bb, \u0432\u0442\u043e\u0440\u043e\u0439 \u2014\u00a0\u0434\u043e\u0432\u0435\u0434\u0435\u0442 \u043d\u0430\u0447\u0430\u0442\u043e\u0435 \u0434\u043e \u043a\u043e\u043d\u0446\u0430 \u0438 \u0437\u0430\u043c\u0435\u0442\u0438\u0442, \u0447\u0442\u043e \u043f\u0440\u0438 \u0432\u0441\u0435\u0445 \u0435\u0433\u043e \u0443\u0441\u0438\u043b\u0438\u044f\u0445 \u044d\u0444\u0444\u0435\u043a\u0442\u0430 \u043d\u0430\u0442\u0438\u0432\u043d\u043e\u0441\u0442\u0438 \u0434\u043e\u0431\u0438\u0442\u044c\u0441\u044f \u043d\u0435 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f.<\/p>\n<p>\u0421\u043b\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u0434\u043e\u0431\u043d\u044b\u0445 \u043f\u043e\u0434\u0445\u043e\u0434\u043e\u0432 \u0437\u0430\u043a\u043b\u044e\u0447\u0430\u0435\u0442\u0441\u044f \u0432 \u0442\u043e\u043c, \u0447\u0442\u043e \u043f\u043e\u043c\u0438\u043c\u043e \u0438\u0437\u0443\u0447\u0435\u043d\u0438\u044f \u0432\u0441\u0435\u0433\u043e \u0432\u044b\u0448\u0435\u043f\u0435\u0440\u0435\u0447\u0438\u0441\u043b\u0435\u043d\u043d\u043e\u0433\u043e \u043d\u0443\u0436\u043d\u043e \u0435\u0449\u0435 \u0438 \u0431\u044b\u0442\u044c \u044d\u043a\u0441\u043f\u0435\u0440\u0442\u043e\u043c \u0432 \u0442\u043e\u043c, \u043a\u0430\u043a \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433 \u0432 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0435, \u0432\u0435\u0434\u044c \u043f\u043e\u0432\u0435\u0441\u0438\u0442\u044c \u0432\u0441\u0435 \u0441\u0432\u043e\u0438 \u043f\u0440\u0435\u043a\u0440\u0430\u0441\u043d\u043e \u043e\u043f\u0438\u0441\u0430\u043d\u043d\u044b\u0435 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u044f \u043d\u0430 pointermove \u2014 \u044d\u0442\u043e \u043a\u0430\u043a \u043f\u0440\u0430\u0432\u0438\u043b\u043e \u0444\u0430\u0442\u0430\u043b\u0438\u0442\u0438 \u0441 \u0442\u043e\u0447\u043a\u0438 \u0437\u0440\u0435\u043d\u0438\u044f \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430.<\/p>\n<p>\u041a\u043e\u0440\u043e\u0447\u0435, \u0447\u0435\u0433\u043e \u043c\u044b \u043c\u0443\u0447\u0430\u0435\u043c\u0441\u044f? \u041c\u044b \u0432 2025 \u0438\u043b\u0438 \u0433\u0434\u0435?<\/p>\n<h3>\u0421\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0439 CSS \u0431\u0435\u0436\u0438\u0442 \u043d\u0430 \u043f\u043e\u043c\u043e\u0449\u044c<\/h3>\n<p>\u041d\u0435 \u0437\u043d\u0430\u044e \u043a\u0430\u043a \u0432\u044b, \u0430 \u044f, \u043a\u043e\u0433\u0434\u0430 \u0432\u0438\u0436\u0443 \u0433\u0430\u043b\u0435\u0440\u0435\u044e, \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u043b\u044f\u044e \u0441\u0435\u0431\u0435 \u0431\u043b\u043e\u043a \u0441\u043e \u0441\u043a\u0440\u043e\u043b\u043b\u043e\u043c. \u0421\u043a\u0440\u043e\u043b\u043b \u043f\u0440\u0435\u043a\u0440\u0430\u0441\u0435\u043d \u0442\u0435\u043c, \u0447\u0442\u043e \u043e\u043d \u0434\u0435\u043b\u0430\u0435\u0442 \u0437\u0430 \u043d\u0430\u0441 \u043c\u043e\u0440\u0435 \u0440\u0430\u0431\u043e\u0442\u044b:<\/p>\n<ul>\n<li>\n<p>\u041e\u0431\u0435\u0441\u043f\u0435\u0447\u0438\u0432\u0430\u0435\u0442 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 \u0431\u0435\u0437 JS<\/p>\n<\/li>\n<li>\n<p>\u041e\u0431\u0435\u0441\u043f\u0435\u0447\u0438\u0432\u0430\u0435\u0442 overscroll \u043d\u0430 \u043a\u0440\u0430\u044f\u0445 \u0431\u0435\u0437 JS<\/p>\n<\/li>\n<li>\n<p>\u041f\u0440\u0438 \u0436\u0435\u043b\u0430\u043d\u0438\u0438 \u0434\u0430\u0435\u0442 \u0441\u043b\u0435\u0434\u0438\u0442\u044c \u043d\u0430 \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441\u043e\u043c \u0447\u0435\u0440\u0435\u0437 DOM-\u0441\u043e\u0431\u044b\u0442\u0438\u044f<\/p>\n<\/li>\n<\/ul>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u043f\u043e\u043f\u0440\u043e\u0431\u0443\u0435\u043c \u0438\u043c\u043f\u043b\u0435\u043c\u0435\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430\u0448\u0443 \u0433\u0430\u043b\u0435\u0440\u0435\u044e \u0447\u0435\u0440\u0435\u0437 \u0433\u043e\u0440\u0438\u0437\u043e\u043d\u0442\u0430\u043b\u044c\u043d\u044b\u0439 \u0441\u043a\u0440\u043e\u043b\u043b.<\/p>\n<h3>\u0413\u0430\u043b\u0435\u0440\u0435\u044f \u043a\u0430\u043a \u0431\u043b\u043e\u043a \u0441 \u0433\u043e\u0440\u0438\u0437\u043e\u043d\u0442\u0430\u043b\u044c\u043d\u044b\u043c \u0441\u043a\u0440\u043e\u043b\u043b\u043e\u043c<\/h3>\n<pre><code class=\"typescript\">import { Children, ReactNode } from \"react\";  export const Gallery = ({ children }: { children?: ReactNode; }) =&gt; {   return (     &lt;div&gt;       &lt;div style={{ overflowX: \"auto\", display: \"flex\" }}&gt;         {Children.map(children, (child, i) =&gt; (           &lt;div style={{ width: \"100%\", flexShrink: 0 }} key={i}&gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<p>\u0412\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0438\u0437\u0443\u043c\u0438\u0442\u0435\u043b\u044c\u043d\u043e, \u043a\u0430\u043a \u043f\u043e \u043c\u043d\u0435, \u043d\u043e \u043e\u0447\u0435\u0432\u0438\u0434\u043d\u043e \u043c\u044b \u0442\u0443\u0442 \u0432\u0441\u0442\u0440\u0435\u0442\u0438\u043c\u0441\u044f \u0441 \u0440\u044f\u0434\u043e\u043c \u043f\u0440\u043e\u0431\u043b\u0435\u043c:<\/p>\n<ol>\n<li>\n<p>\u0412\u0438\u0434\u0438\u043c\u043e\u0441\u0442\u044c \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440\u0430.<\/p>\n<\/li>\n<li>\n<p>\u0421\u043a\u0432\u043e\u0437\u043d\u043e\u0435 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u0435: \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e, \u0447\u0442\u043e\u0431\u044b \u043f\u043e\u0441\u043b\u0435 \u0441\u0432\u0430\u0439\u043f\u0430 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u043b\u0430 \u043f\u043b\u0430\u0432\u043d\u0430\u044f \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u043d\u0430 \u0433\u0440\u0430\u043d\u0438\u0446\u0435 \u043c\u0435\u0436\u0434\u0443 \u0441\u043b\u0430\u0439\u0434\u0430\u043c\u0438.<\/p>\n<\/li>\n<\/ol>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e322ca986ae5034d1d08\" data-style=\"\" id=\"6792e322ca986ae5034d1d08\" width=\"\"><\/div>\n<h3>\u0423\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440<\/h3>\n<p>\u0420\u0430\u043d\u044c\u0448\u0435, \u0447\u0442\u043e\u0431\u044b \u0443\u0431\u0440\u0430\u0442\u044c \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440, \u043d\u0443\u0436\u043d\u043e \u0431\u044b\u043b\u043e \u0445\u0438\u043c\u0438\u0447\u0438\u0442\u044c \u0441 \u0434\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u043c\u0438 HTML-\u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430\u043c\u0438, \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e\u043c <code>overflow<\/code> \u0438 \u0445\u0430\u0440\u0434\u043a\u043e\u0434\u043e\u043c \u0433\u0430\u0431\u0430\u0440\u0438\u0442\u043e\u0432 \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440\u0430.<\/p>\n<p>\u041f\u043e\u0442\u043e\u043c \u0443 \u043d\u0430\u0441 \u043f\u043e\u044f\u0432\u0438\u043b\u0441\u044f \u0437\u0430\u043c\u0435\u0447\u0430\u0442\u0435\u043b\u044c\u043d\u044b\u0439 <code>::-webkit-scrollbar-*<\/code>.<\/p>\n<p>\u0421 \u0434\u0435\u043a\u0430\u0431\u0440\u044f 2024-\u0433\u043e \u0432\u0441\u0435\u043c \u044d\u0442\u0438\u043c \u0437\u0430\u043d\u0438\u043c\u0430\u0442\u044c\u0441\u044f \u043d\u0435 \u043d\u0443\u0436\u043d\u043e: \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scrollbar-width<\/code> \u0441\u0442\u0430\u043b\u043e \u0434\u043e\u0441\u0442\u0443\u043f\u043d\u043e \u0432\u043e <a href=\"https:\/\/caniuse.com\/mdn-css_properties_scrollbar-width_none\" rel=\"noopener noreferrer nofollow\">\u0432\u0441\u0435\u0445 \u043f\u043e\u043f\u0443\u043b\u044f\u0440\u043d\u044b\u0445 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0430\u0445<\/a>:<\/p>\n<pre><code class=\"typescript\">import { Children, ReactNode } from \"react\";  export const Gallery = ({ children }: { children?: ReactNode; }) =&gt; {   return (     &lt;div&gt;       &lt;div          style={{            overflowX: \"auto\",            display: \"flex\",           scrollbarWidth: \"none\"         }}       &gt;         {Children.map(children, (child, i) =&gt; (           &lt;div style={{ width: \"100%\", flexShrink: 0 }} key={i}&gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e342999b83f994c8b5e7\" data-style=\"\" id=\"6792e342999b83f994c8b5e7\" width=\"\"><\/div>\n<h3>\u0423\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u043a\u0432\u043e\u0437\u043d\u043e\u0435 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u0435<\/h3>\n<p>\u0421\u0430\u043c\u043e\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0435. 2024-\u0439 \u043f\u043e\u0434\u0430\u0440\u0438\u043b \u043d\u0430\u043c \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e \u043c\u0430\u0441\u0441\u043e\u0432\u0443\u044e \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0443 \u043d\u0435\u0432\u0438\u0434\u0438\u043c\u043e\u0433\u043e \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440\u0430, \u043d\u043e \u0438 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0443 \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430 \u0441\u0432\u043e\u0439\u0441\u0442\u0432 <code>scroll-snap<\/code>. Snap \u0432 \u043f\u0435\u0440\u0435\u0432\u043e\u0434\u0435 \u0441 \u0430\u043d\u0433\u043b\u0438\u0439\u0441\u043a\u043e\u0433\u043e \u043e\u0437\u043d\u0430\u0447\u0430\u0435\u0442 \u00ab\u0449\u0435\u043b\u0447\u043e\u043a\u00bb. \u042d\u0442\u043e\u0442 \u0449\u0435\u043b\u0447\u043e\u043a \u2014 \u043a\u0430\u043a \u0440\u0430\u0437 \u0442\u043e, \u0447\u0442\u043e \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e.<\/p>\n<h4>\u041f\u0430\u0440\u0443 \u0441\u043b\u043e\u0432 \u043e\u0431 \u044d\u0442\u043e\u043c \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0435.<\/h4>\n<p>\u0415\u0441\u043b\u0438 \u0433\u043e\u0432\u043e\u0440\u0438\u0442\u044c \u043e\u0431\u0449\u043e, \u0442\u043e \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u044d\u0442\u043e\u0433\u043e \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u0433\u0438\u0431\u043a\u043e \u043d\u0430\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c \u043f\u043e\u0432\u0435\u0434\u0435\u043d\u0438\u0435 \u0441\u043a\u0440\u043e\u043b\u043b\u0430, \u043e\u0441\u043e\u0431\u0435\u043d\u043d\u043e \u043a\u043e\u0433\u0434\u0430 \u0432 \u0431\u043b\u043e\u043a\u0435 \u043f\u0440\u0438\u0441\u0443\u0442\u0441\u0442\u0432\u0443\u044e\u0442 \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b. \u041a\u0430\u0436\u0434\u044b\u0439 \u044d\u043b\u0435\u043c\u0435\u043d\u0442 \u043c\u043e\u0436\u0435\u0442 \u0441\u0442\u0430\u0442\u044c \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u043e\u0439 \u0442\u043e\u0447\u043a\u043e\u0439 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0430. \u0417\u0430 \u044d\u0442\u043e \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scroll-snap-type<\/code>.<\/p>\n<p>\u041f\u043e\u043c\u0438\u043c\u043e \u0442\u043e\u0447\u0435\u043a \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u043c\u043e\u0436\u043d\u043e \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0442\u0430\u043a\u0438\u0435 \u0432\u0435\u0449\u0438 \u043a\u0430\u043a \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435 (<code>scroll-snap-padding<\/code>, <code>scroll-snap-margin<\/code>) \u043c\u0435\u0436\u0434\u0443 \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u044b\u043c\u0438 \u0442\u043e\u0447\u043a\u0430\u043c\u0438, \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u0435 \u044d\u0442\u0438\u0445 \u0442\u043e\u0447\u0435\u043a (<code>scroll-snap-align<\/code>) \u0438 \u0442\u043e, \u043c\u043e\u0436\u043d\u043e \u043b\u0438 \u043f\u0440\u043e\u0441\u043a\u0430\u043a\u0438\u0432\u0430\u0442\u044c \u0442\u043e\u0447\u043a\u0438 \u0432 \u0441\u043b\u0443\u0447\u0430\u0435, \u0435\u0441\u043b\u0438 \u0438\u043d\u0442\u0435\u043d\u0441\u0438\u0432\u043d\u043e\u0441\u0442\u044c \u0436\u0435\u0441\u0442\u0430 \u044d\u0442\u043e \u043f\u043e\u0434\u0440\u0430\u0437\u0443\u043c\u0435\u0432\u0430\u0435\u0442 (<code>scroll-snap-stop<\/code>).<\/p>\n<p>\u041c\u044b \u043d\u0435 \u0431\u0443\u0434\u0435\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432\u0441\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430, \u043d\u043e \u0435\u0441\u043b\u0438 \u0432\u0430\u043c \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u0438\u0437\u0443\u0447\u0438\u0442\u044c \u043a\u0430\u043a\u043e\u0435-\u0442\u043e \u0438\u0437 \u043d\u0438\u0445, \u0440\u0435\u043a\u043e\u043c\u0435\u043d\u0434\u0443\u044e <a href=\"https:\/\/developer.mozilla.org\/en-US\/docs\/Web\/CSS\/CSS_scroll_snap\" rel=\"noopener noreferrer nofollow\">\u0441\u0442\u0430\u0442\u044c\u044e \u0438\u0437 MDN<\/a>. \u0422\u0430\u043c \u0438 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0435 \u043f\u0440\u0438\u043c\u0435\u0440\u044b, \u0438 \u043f\u043e\u0434\u0440\u043e\u0431\u043d\u043e\u0435 \u043e\u043f\u0438\u0441\u0430\u043d\u0438\u0435, \u0438 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044f \u043e \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u043d\u043e\u0439 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0435.<\/p>\n<p>\u041d\u0430\u0448\u0430 \u0437\u0430\u0434\u0430\u0447\u0430 \u0441\u0435\u0439\u0447\u0430\u0441 \u2014\u00a0\u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u044b\u0435 \u0442\u043e\u0447\u043a\u0438 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0430. \u0414\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u0432\u043e\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c\u0441\u044f \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430\u043c\u0438 <code>scroll-snap-type<\/code> \u0438 <code>scroll-snap-stop<\/code>.<\/p>\n<h4>\u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u044b\u0435 \u0442\u043e\u0447\u043a\u0438 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0430<\/h4>\n<pre><code class=\"typescript\">&lt;div    style={{      overflowX: \"auto\",      display: \"flex\",     scrollbarWidth: \"none\",     scrollSnapType: \"x mandatory\",     scrollSnapStop: \"always\"   }} &gt;   {Children.map(children, (child, i) =&gt; (     &lt;div style={{ width: \"100%\", flexShrink: 0 }} key={i}&gt;       {child}     &lt;\/div&gt;   ))} &lt;\/div&gt;<\/code><\/pre>\n<p>\u041a\u0430\u043a \u0432\u0438\u0434\u0438\u0442\u0435, \u043d\u0435\u0434\u043e\u0441\u0442\u0430\u044e\u0449\u0430\u044f \u043c\u0435\u0445\u0430\u043d\u0438\u043a\u0430 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u044f \u0433\u0430\u043b\u0435\u0440\u0435\u0438 \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u0442\u0441\u044f \u043f\u0443\u0442\u0435\u043c \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u0431\u0443\u043a\u0432\u0430\u043b\u044c\u043d\u043e \u0434\u0432\u0443\u0445 CSS-\u0441\u0432\u043e\u0439\u0441\u0442\u0432.<\/p>\n<p>\u0417\u043d\u0430\u0447\u0435\u043d\u0438\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 <code>scroll-snap-type: x mandatory<\/code> \u043e\u0437\u043d\u0430\u0447\u0430\u0435\u0442, \u0447\u0442\u043e \u043f\u043e \u043e\u0441\u0438 X \u0441\u043a\u0440\u043e\u043b\u043b \u0434\u043e\u043b\u0436\u0435\u043d \u043e\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043d\u0430 \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u043e\u0439 \u0442\u043e\u0447\u043a\u0435. \u041f\u043e\u043c\u0438\u043c\u043e mandatory, \u0435\u0441\u0442\u044c \u0435\u0449\u0435 proximity, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0434\u0435\u043b\u0435\u0433\u0438\u0440\u0443\u0435\u0442 \u0440\u0435\u0448\u0435\u043d\u0438\u0435 \u043e\u0431 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0435 \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u043e\u043c\u0443 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0443. \u0417\u0434\u0435\u0441\u044c \u043d\u0430\u043c \u044d\u0442\u043e \u043d\u0438 \u043a \u0447\u0435\u043c\u0443. \u041f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u043e\u0439 \u0442\u043e\u0447\u043a\u043e\u0439 \u0441\u0435\u0439\u0447\u0430\u0441 \u044f\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u043b\u0435\u0432\u044b\u0439 \u043a\u0440\u0430\u0439 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0434\u043e\u0447\u0435\u0440\u043d\u0435\u0433\u043e \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430 (\u0441\u043b\u0430\u0439\u0434\u0430).<\/p>\n<p>\u0417\u043d\u0430\u0447\u0435\u043d\u0438\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 <code>scroll-snap-stop: always<\/code> \u0433\u043e\u0432\u043e\u0440\u0438\u0442 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0443 \u043d\u0435 \u043f\u0440\u043e\u0433\u043b\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u0441\u043b\u0430\u0439\u0434, \u0434\u0430\u0436\u0435 \u0435\u0441\u043b\u0438 \u0441\u0432\u0430\u0439\u043f \u0431\u044b\u043b \u043e\u0447\u0435\u043d\u044c \u0441\u0438\u043b\u044c\u043d\u044b\u043c.<\/p>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e35bf97ec4e4dc644e25\" data-style=\"\" id=\"6792e35bf97ec4e4dc644e25\" width=\"\"><\/div>\n<h4>\u041a\u0430\u0441\u0442\u043e\u043c\u043d\u0430\u044f \u0448\u0438\u0440\u0438\u043d\u0430 \u0441\u043b\u0430\u0439\u0434\u0430, \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u0435 \u0438 \u0437\u0430\u0437\u043e\u0440 \u043c\u0435\u0436\u0434\u0443 \u0441\u043b\u0430\u0439\u0434\u0430\u043c\u0438<\/h4>\n<p>\u0421\u043b\u0430\u0439\u0434\u044b \u043d\u0435 \u0432\u0441\u0435\u0433\u0434\u0430 \u0434\u043e\u043b\u0436\u043d\u044b \u0431\u044b\u0442\u044c \u0448\u0438\u0440\u0438\u043d\u043e\u0439 \u0441 \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044f, \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u043f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u044e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e \u0434\u043b\u044f \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0448\u0438\u0440\u0438\u043d\u044b. \u0410 \u0440\u0430\u0437 \u0434\u0430\u0435\u043c \u043f\u043e\u0438\u0433\u0440\u0430\u0442\u044c\u0441\u044f \u0441 \u0448\u0438\u0440\u0438\u043d\u043e\u0439, \u0442\u043e \u0438 \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u0435 \u043e\u0442\u043d\u043e\u0441\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044f \u0442\u043e\u0436\u0435 \u043f\u0440\u0438\u0433\u043e\u0434\u0438\u0442\u0441\u044f. \u0414\u043b\u044f \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u044f \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0443\u043f\u043e\u043c\u044f\u043d\u0443\u0442\u043e\u0435 \u0432\u044b\u0448\u0435 CSS-\u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scroll-snap-align<\/code>. \u041d\u0443 \u0438 \u0434\u0430\u0434\u0438\u043c \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u043c\u0435\u0436\u0434\u0443 \u0441\u043b\u0430\u0439\u0434\u0430\u043c\u0438. \u0422\u0443\u0442 \u043d\u0435 \u0431\u0443\u0434\u0435\u043c \u043d\u0438\u0447\u0435\u0433\u043e \u0432\u044b\u0434\u0443\u043c\u044b\u0432\u0430\u0442\u044c \u0438 \u043f\u0440\u0438\u043c\u0435\u043d\u0438\u043c \u0441\u0442\u0430\u0440\u044b\u0439 \u0441\u043e\u0432\u0435\u0441\u0442\u043a\u0438\u0439 <code>gap<\/code>:<\/p>\n<pre><code class=\"typescript\">import { Children, ReactNode } from \"react\";  export const Gallery = ({   children,    slideWidth = \"100%\",   slideAlign = \"center\",   slidesGap = 0 }: {    children?: ReactNode;   slideWidth?: string | number;   slideAlign?: \"start\" | \"center\" | \"end\";   slidesGap?: string | number; }) =&gt; {   return (     &lt;div&gt;       &lt;div          style={{            overflowX: \"auto\",            display: \"flex\",           scrollbarWidth: \"none\",           scrollSnapType: \"x mandatory\",           scrollSnapStop: \"always\",           gap: slidesGap         }}       &gt;         {Children.map(children, (child, i) =&gt; (           &lt;div              style={{               width: slideWidth,               scrollSnapAlign: slideAlign,               flexShrink: 0,             }}              key={i}           &gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<blockquote>\n<p>\u041f\u0440\u043e\u0448\u0443 \u043e\u0431\u0440\u0430\u0442\u0438\u0442\u044c \u0432\u043d\u0438\u043c\u0430\u043d\u0438\u0435 \u043d\u0430 \u0442\u043e, \u0447\u0442\u043e \u0441\u0432\u0435\u0436\u0435\u0438\u0441\u043f\u0435\u0447\u0435\u043d\u043d\u043e\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scroll-snap-align<\/code> \u0432\u043c\u0435\u0441\u0442\u043e top, left, bottom \u0438 right \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u0435\u0442 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f start \u0438 end. \u042d\u0442\u043e \u0442\u0430\u043a \u043d\u0430\u0437\u044b\u0432\u0430\u0435\u043c\u044b\u0435 logical values: \u043e\u043d\u0438 \u043a\u043e\u043c\u043f\u0430\u043a\u0442\u043d\u0435\u0435 \u0438 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u043d\u0435 \u0434\u0443\u043c\u0430\u0442\u044c \u043e \u043f\u0435\u0440\u0435\u0432\u0435\u0440\u043d\u0443\u0442\u044b\u0445 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430\u0445 (\u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430\u0445 \u043d\u0430 \u0430\u0440\u0430\u0431\u0441\u043a\u043e\u043c \u044f\u0437\u044b\u043a\u0435).<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e, \u0447\u0442\u043e \u0437\u0430 logical values \u0442\u0430\u043a\u0438\u0435, \u0442\u043e \u043c\u043e\u0436\u0435\u0442\u0435 <a href=\"https:\/\/css-tricks.com\/css-logical-properties-and-values\/\" rel=\"noopener noreferrer nofollow\">\u043f\u043e\u0447\u0438\u0442\u0430\u0442\u044c \u0441\u0442\u0430\u0442\u044c\u044e \u043d\u0430 CSS Tricks<\/a> \u0438\u043b\u0438 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c <a href=\"https:\/\/www.youtube.com\/watch?v=UNqH9rT6nqQ\" rel=\"noopener noreferrer nofollow\">\u043c\u043e\u0439 \u0432\u0435\u0431\u0438\u043d\u0430\u0440<\/a>.<\/p>\n<\/blockquote>\n<h3>\u0414\u0430\u0435\u043c \u043c\u0435\u043d\u044f\u0442\u044c \u043d\u0430\u0447\u0430\u043b\u044c\u043d\u044b\u0439 \u0441\u043b\u0430\u0439\u0434<\/h3>\n<p>\u041b\u0430\u0434\u043d\u043e, \u043f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u044e \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u043f\u043e\u043a\u043e\u0434\u0438\u0442\u044c. \u0420\u0430\u0437\u0440\u0430\u0431\u044b \u043f\u043e-\u043b\u044e\u0431\u043e\u043c\u0443 \u0437\u0430\u0445\u043e\u0442\u044f\u0442 \u0440\u0438\u0441\u043e\u0432\u0430\u0442\u044c \u0433\u0430\u043b\u0435\u0440\u0435\u044e \u0441 \u043f\u0435\u0440\u0432\u043e\u043d\u0430\u0447\u0430\u043b\u044c\u043d\u044b\u043c \u0441\u043b\u0430\u0439\u0434\u043e\u043c, \u043e\u0442\u043b\u0438\u0447\u043d\u044b\u043c \u043e\u0442 \u043f\u0435\u0440\u0432\u043e\u0433\u043e. \u0422\u0443\u0442 \u0443\u0436 \u0431\u0435\u0437 TS \u043d\u0435 \u043e\u0431\u043e\u0439\u0442\u0438\u0441\u044c:<\/p>\n<pre><code class=\"typescript\">import { Children, ReactNode, useEffect, useRef } from \"react\";  export const Gallery = ({   children,    slideWidth = \"100%\",    slideAlign = \"center\",   slidesGap = 0,   initialSlideIndex = 0, }: {    children?: ReactNode;   slideWidth?: string | number;   slideAlign?: \"start\" | \"center\" | \"end\";   slidesGap?: string | number;   initialSlideIndex?: number; }) =&gt; {   const scrollContainer = useRef&lt;HTMLDivElement&gt;(null);      useEffect(() =&gt; {     if (initialSlideIndex &gt; 0) {       scrollContainer.current?.children[initialSlideIndex].scrollIntoView({         behavior: \"instant\",         block: \"nearest\",       });     }   }, []);     return (     &lt;div&gt;       &lt;div          style={{            overflowX: \"auto\",            display: \"flex\",           scrollbarWidth: \"none\",           scrollSnapType: \"x mandatory\",           scrollSnapStop: \"always\",           gap: slidesGap         }}         ref={scrollContainer}       &gt;         {Children.map(children, (child, i) =&gt; (           &lt;div              style={{               width: slideWidth,               scrollSnapAlign: slideAlign,               flexShrink: 0,             }}              key={i}           &gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<h4>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u043f\u043e-\u043f\u043e\u0440\u044f\u0434\u043a\u0443<\/h4>\n<p><code>useRef<\/code> \u0438 <code>useEffect<\/code> \u2014\u00a0\u0441\u0443\u0449\u043d\u043e\u0441\u0442\u0438 \u0438\u0437 \u043c\u0438\u0440\u0430 React, \u043e\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043d\u0430\u0434\u043e\u043b\u0433\u043e \u043d\u0430 \u043d\u0438\u0445 \u043d\u0435 \u0445\u043e\u0447\u0435\u0442\u0441\u044f. \u0427\u0435\u0440\u0435\u0437 <code>useRef<\/code> \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0441\u0441\u044b\u043b\u043a\u0443 \u043d\u0430 DOM-\u044d\u043b\u0435\u043c\u0435\u043d\u0442, \u0447\u0435\u0440\u0435\u0437 <code>useEffect<\/code> \u2014\u00a0\u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c \u0441\u043d\u0438\u043f\u043f\u0435\u0442 \u0432 \u043a\u043e\u043b\u043b\u0431\u044d\u043a\u0435 \u043f\u0440\u0438 \u043f\u0435\u0440\u0432\u043e\u043c \u0440\u0435\u043d\u0434\u0435\u0440\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u0430.<\/p>\n<p>\u0418\u0437 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0433\u043e \u0442\u0443\u0442 \u043c\u0435\u0442\u043e\u0434 DOM-\u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430 <code>scrollIntoView<\/code>. \u042d\u0442\u043e \u043c\u0435\u0442\u043e\u0434, \u043f\u0440\u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0435 \u043a\u043e\u0442\u043e\u0440\u043e\u0433\u043e \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044c \u043f\u0440\u043e\u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442 \u0441\u0435\u0431\u044f \u0434\u043e \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430.<\/p>\n<blockquote>\n<p><strong>\u0412\u0430\u0436\u043d\u043e:<\/strong> \u043c\u0435\u0442\u043e\u0434 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0442\u044c \u043d\u0430 \u0434\u043e\u0447\u0435\u0440\u043d\u0435\u043c \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0435, \u0430 \u043d\u0435 \u043d\u0430 \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u0435.<\/p>\n<\/blockquote>\n<p>\u0421\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u0443 \u043c\u0435\u0442\u043e\u0434\u0430 \u043d\u0435 \u043c\u0435\u043d\u0435\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u044b\u0435.<\/p>\n<p><code>behavior<\/code> \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0442\u0438\u043f \u0441\u043a\u0440\u043e\u043b\u043b\u0430: \u043e\u043d \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c \u0438\u043b\u0438 \u043c\u0433\u043d\u043e\u0432\u0435\u043d\u043d\u044b\u043c, \u0438\u043b\u0438 \u043f\u043b\u0430\u0432\u043d\u044b\u043c. \u0414\u043b\u044f \u0440\u0435\u0448\u0435\u043d\u0438\u044f \u043d\u0430\u0448\u0435\u0439 \u0437\u0430\u0434\u0430\u0447\u0438 \u043d\u0443\u0436\u0435\u043d \u0438\u043c\u0435\u043d\u043d\u043e \u043c\u0433\u043d\u043e\u0432\u0435\u043d\u043d\u044b\u0439.<\/p>\n<p><code>block<\/code> \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u0443\u0435\u0442 \u0442\u043e, \u043a \u043a\u0430\u043a\u0438\u043c \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u0441\u043a\u0438\u043c \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430\u043c \u0431\u0443\u0434\u0435\u0442 \u043f\u0440\u0438\u043c\u0435\u043d\u0435\u043d \u0432\u044b\u0437\u043e\u0432. \u0415\u0441\u043b\u0438 \u043d\u0435 \u0443\u043a\u0430\u0437\u0430\u0442\u044c \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u0435 <strong>nearest<\/strong>, \u043c\u044b \u0443\u0432\u0438\u0434\u0438\u043c, \u0447\u0442\u043e \u043f\u0440\u0438 \u0432\u044b\u0437\u043e\u0432\u0435 \u043e\u0442\u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e \u0441\u043a\u0440\u043e\u043b\u043b \u0433\u0430\u043b\u0435\u0440\u0435\u0438, \u043d\u043e \u0438 \u0441\u043a\u0440\u043e\u043b\u043b \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430. \u0410 \u044d\u0442\u043e \u043d\u0430\u043c \u043d\u0435 \u043d\u0443\u0436\u043d\u043e.<\/p>\n<p>\u0421\u043e \u0432\u0441\u0435\u043c\u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044f\u043c\u0438 \u0438 \u043d\u044e\u0430\u043d\u0441\u0430\u043c\u0438 \u044d\u0442\u043e\u0433\u043e \u043c\u0435\u0442\u043e\u0434\u0430 \u043c\u043e\u0436\u043d\u043e \u043e\u0437\u043d\u0430\u043a\u043e\u043c\u0438\u0442\u044c\u0441\u044f \u0432 <a href=\"https:\/\/developer.mozilla.org\/en-US\/docs\/Web\/API\/Element\/scrollIntoView\" rel=\"noopener noreferrer nofollow\">\u0441\u0442\u0430\u0442\u044c\u0435 \u043e\u0442 MDN<\/a>.<\/p>\n<h3>\u041f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0430 \u0434\u0435\u0441\u043a\u0442\u043e\u043f\u0430<\/h3>\n<p>\u0412 \u043d\u0430\u0447\u0430\u043b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 \u044f \u043f\u0438\u0441\u0430\u043b \u043e \u0442\u043e\u043c, \u0447\u0442\u043e \u0445\u043e\u0442\u044c \u043c\u044b \u0438 mobile-first, \u0434\u0435\u0441\u043a\u0442\u043e\u043f\u044b \u043d\u0435\u043b\u044c\u0437\u044f \u0441\u043f\u0438\u0441\u044b\u0432\u0430\u0442\u044c \u0441\u043e \u0441\u0447\u0435\u0442\u043e\u0432. \u0422\u0430\u043c \u0437\u0430\u0447\u0430\u0441\u0442\u0443\u044e \u043d\u0435\u0442 \u0443\u0434\u043e\u0431\u043d\u043e\u0433\u043e \u0441\u043f\u043e\u0441\u043e\u0431\u0430 \u043f\u043e\u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442\u044c \u0432\u0435\u0440\u0442\u0438\u043a\u0430\u043b\u044c\u043d\u043e, \u0434\u0430 \u0438 \u0442\u0430\u043c \u0433\u0434\u0435 \u0442\u0430\u043a \u043c\u043e\u0436\u043d\u043e, \u044d\u0442\u043e \u0432\u0441\u0451 \u0440\u0430\u0432\u043d\u043e \u043d\u0435 \u0441\u0447\u0438\u0442\u0430\u0435\u0442\u0441\u044f \u043f\u0440\u0438\u0432\u044b\u0447\u043d\u044b\u043c UX-\u043f\u0430\u0442\u0442\u0435\u0440\u043d\u043e\u043c. \u041d\u0430 \u0434\u0435\u0441\u043a\u0442\u043e\u043f\u0430\u0445 \u0431\u0430\u043b\u043e\u043c \u043f\u0440\u0430\u0432\u044f\u0442 \u043a\u043b\u0438\u043a\u0438 \u043c\u044b\u0448\u043a\u043e\u0439. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u043f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u044e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0441\u0442\u0440\u0435\u043b\u043a\u0438 \u043f\u043e \u0431\u043e\u043a\u0430\u043c \u0433\u0430\u043b\u0435\u0440\u0435\u0438.<\/p>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0441\u0434\u0435\u043b\u0430\u0435\u043c \u0442\u0430\u043a, \u0447\u0442\u043e\u0431\u044b \u0432\u0438\u0434\u0438\u043c\u043e\u0441\u0442\u044c \u0441\u0442\u0440\u0435\u043b\u043e\u043a \u0431\u044b\u043b\u0430 \u043e\u043f\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e\u0439. \u0422\u0430\u043a \u0436\u0435 \u043f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u044e \u043e\u0442\u0434\u0430\u0442\u044c \u043d\u0430 \u043e\u0442\u043a\u0443\u043f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f\u043c \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u0430 \u0432\u043d\u0435\u0448\u043d\u0438\u0439 \u0432\u0438\u0434 \u0441\u0442\u0440\u0435\u043b\u043e\u043a. \u0417\u0430\u0434\u0430\u0434\u0438\u043c \u0438\u043c \u0442\u043e\u043b\u044c\u043a\u043e \u0431\u0430\u0437\u043e\u0432\u044b\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430.<\/p>\n<pre><code class=\"typescript\">import { Children, ReactNode, useEffect, useRef, CSSProperties } from \"react\";  type Arrow = {   content: ReactNode;   inlineOffset: number | string; };  const arrowStyles: CSSProperties = {   position: \"absolute\",   top: \"50%\",   transform: \"translateY(-50%)\",   zIndex: 2, };  export const Gallery = ({   children,    slideWidth = \"100%\",    slideAlign = \"center\",   slidesGap = 0,   initialSlideIndex = 0,   arrows, }: {    children?: ReactNode;   slideWidth?: string | number;   slideAlign?: \"start\" | \"center\" | \"end\";   slidesGap?: string | number;   initialSlideIndex?: number;   arrows?: [Arrow, Arrow]; }) =&gt; {   const scrollContainer = useRef&lt;HTMLDivElement&gt;(null);   const activeSlideIndex = useRef&lt;number&gt;(initialSlideIndex);      useEffect(() =&gt; {     if (initialSlideIndex &gt; 0) {       scrollContainer.current?.children[initialSlideIndex].scrollIntoView({         behavior: \"instant\",         block: \"nearest\",       });     }   }, []);      const scrollTo = (slideIndex: number) =&gt; {     if (activeSlideIndex.current !== slideIndex) {       scrollContainer.current?.children[slideIndex].scrollIntoView({         behavior: \"smooth\",         block: \"nearest\",       });       activeSlideIndex.current = slideIndex;     }   };    return (     &lt;div&gt;       &lt;div style={{ position: \"relative\" }}&gt;         {arrows &amp;&amp; (           &lt;div             style={{               ...arrowStyles,               insetInlineStart: arrows[0].inlineOffset,             }}             onClick={() =&gt; {               if (activeSlideIndex.current &gt; 0) {                 scrollTo(activeSlideIndex.current - 1);               }             }}           &gt;             {arrows[0].content}           &lt;\/div&gt;         )}         &lt;div            style={{              overflowX: \"auto\",              display: \"flex\",             scrollbarWidth: \"none\",             scrollSnapType: \"x mandatory\",             scrollSnapStop: \"always\",             gap: slidesGap,             position: \"relative\",             zIndex: 1,           }}           ref={scrollContainer}         &gt;           {Children.map(children, (child, i) =&gt; (             &lt;div                style={{                 width: slideWidth,                 scrollSnapAlign: slideAlign,                 flexShrink: 0,               }}                key={i}             &gt;               {child}             &lt;\/div&gt;           ))}         &lt;\/div&gt;           {arrows &amp;&amp; (           &lt;div             style={{               ...arrowStyles,               insetInlineEnd: arrows[1].inlineOffset,             }}             onClick={() =&gt; {               if (activeSlideIndex.current + 1 &lt; Children.count(children)) {                 scrollTo(activeSlideIndex.current + 1);               }             }}           &gt;             {arrows[1].content}           &lt;\/div&gt;         )}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<p>\u041e\u0449\u0443\u0449\u0435\u043d\u0438\u0435, \u043a\u0430\u043a \u0431\u0443\u0434\u0442\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u043b\u043e\u0441\u044c \u043c\u043d\u043e\u0433\u043e \u043a\u043e\u0434\u0430, \u043d\u043e \u043d\u0430 \u0441\u0430\u043c\u043e\u043c \u0434\u0435\u043b\u0435 \u043e\u043d \u043f\u0440\u043e\u0441\u0442 \u043a\u0430\u043a \u0442\u0440\u0438 \u043a\u043e\u043f\u0435\u0439\u043a\u0438:<\/p>\n<ul>\n<li>\n<p>\u0414\u0430\u0435\u043c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u0440\u043e\u0431\u0440\u0430\u0441\u044b\u0432\u0430\u0442\u044c \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>arrows<\/code>, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u044f\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u043c\u0430\u0441\u0441\u0438\u0432\u043e\u043c \u0441 \u0434\u0432\u0443\u043c\u044f \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430\u043c\u0438. \u041a\u0430\u0436\u0434\u044b\u0439 \u0438\u0437 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 \u2014\u00a0\u043e\u0431\u044a\u0435\u043a\u0442, \u0441\u043e\u0434\u0435\u0440\u0436\u0430\u0449\u0438\u0439 \u0440\u0430\u0437\u043c\u0435\u0442\u043a\u0443 \u0441\u0442\u0440\u0435\u043b\u043a\u0438 \u0438 \u0435\u0451 \u043e\u0442\u0441\u0442\u0443\u043f \u043e\u0442 \u043a\u0440\u0430\u044f.<\/p>\n<\/li>\n<li>\n<p>\u0412\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u0435 \u043f\u043e-\u0432\u0435\u0440\u0442\u0438\u043a\u0430\u043b\u0438 \u0438 \u043f\u0440\u043e\u0447\u0438\u0435 \u043e\u0431\u0449\u0438\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u043c\u044b \u0443\u043a\u0430\u0437\u0430\u043b\u0438 \u0432 <code>arrowStyles<\/code><\/p>\n<\/li>\n<li>\n<p>\u0417\u0430\u043f\u043e\u043c\u0438\u043d\u0430\u0435\u043c \u0438\u043d\u0434\u0435\u043a\u0441 \u0442\u0435\u043a\u0443\u0449\u0435\u0433\u043e \u0441\u043b\u0430\u0439\u0434\u0430 \u0432 \u0440\u0435\u0444\u0435 <code>activeSlideIndex<\/code>. \u042f \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u044e ref, \u0430 \u043d\u0435 state, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043d\u0435 \u0445\u043e\u0447\u0443, \u0447\u0442\u043e\u0431\u044b \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f \u044d\u0442\u043e\u0433\u043e \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u043f\u0440\u0438\u0432\u043e\u0434\u0438\u043b\u0438 \u043a \u043d\u0435\u043d\u0443\u0436\u043d\u043e\u0439 \u043f\u0435\u0440\u0435\u0440\u0438\u0441\u043e\u0432\u043a\u0435.<\/p>\n<\/li>\n<li>\n<p>\u041f\u0438\u0448\u0435\u043c \u0444\u0443\u043d\u043a\u0446\u0438\u044e <code>scrollTo<\/code>, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442 \u0433\u0430\u043b\u0435\u0440\u0435\u044e \u043a \u0443\u043a\u0430\u0437\u0430\u043d\u043d\u043e\u043c\u0443 \u0432 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u0435 \u0438\u043d\u0434\u0435\u043a\u0441\u0443. \u0412\u043d\u0443\u0442\u0440\u0438 \u043c\u043e\u0436\u043d\u043e \u0443\u0432\u0438\u0434\u0435\u0442\u044c \u043f\u0435\u0440\u0435\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0443\u043f\u043e\u043c\u044f\u043d\u0443\u0442\u043e\u0433\u043e \u0432\u044b\u0448\u0435 \u043c\u0435\u0442\u043e\u0434\u0430 <code>scrollIntoView<\/code>, \u043d\u043e \u0443\u0436\u0435 \u0441 \u043f\u0435\u0440\u0435\u0434\u0430\u043d\u043d\u044b\u043c \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u043c <code>{ behavior: \u201csmooth\u201d }<\/code> \u0434\u043b\u044f \u043f\u043b\u0430\u0432\u043d\u043e\u0433\u043e \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u0435 \u043f\u0440\u0438 \u043a\u043b\u0438\u043a\u0435 \u043f\u043e \u0441\u0442\u0440\u0435\u043b\u043a\u0435.<\/p>\n<\/li>\n<li>\n<p>\u0412 \u043e\u0431\u0449\u0435\u043c-\u0442\u043e \u0432\u0441\u0451 \ud83d\ude0a<\/p>\n<\/li>\n<\/ul>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/679379d2757b29e4c07b7319\" data-style=\"\" id=\"679379d2757b29e4c07b7319\" width=\"\"><\/div>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/67937ba72c30d7f9a2dd9049\" data-style=\"\" id=\"67937ba72c30d7f9a2dd9049\" width=\"\"><\/div>\n<h3>\u0411\u0443\u0434\u0443\u0449\u0435\u0435<\/h3>\n<p>\u0412\u043e \u0432\u0441\u0435\u043c, \u0447\u0442\u043e \u043a\u0430\u0441\u0430\u0435\u0442\u0441\u044f \u0441\u043a\u0440\u043e\u043b\u043b\u0430, \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 \u0433\u043e\u0434\u044b \u043c\u043e\u0436\u043d\u043e \u0443\u0432\u0438\u0434\u0435\u0442\u044c \u0431\u0443\u0440\u043d\u044b\u0439 \u0434\u0432\u0438\u0436. \u0420\u0430\u0437\u0432\u0438\u0432\u0430\u044e\u0442\u0441\u044f \u043a\u0430\u043a CSS, \u0442\u0430\u043a \u0438 JS. \u0420\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043f\u0430\u0440\u0443, \u043d\u0430 \u043c\u043e\u0439 \u0432\u0437\u0433\u043b\u044f\u0434, \u043c\u043e\u0449\u043d\u044b\u0445 \u043d\u043e\u0432\u043e\u0432\u0432\u0435\u0434\u0435\u043d\u0438\u0439, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0432\u0441\u043a\u043e\u0440\u0435 \u043f\u043e\u044f\u0432\u044f\u0442\u0441\u044f \u0432\u043e \u0432\u0441\u0435\u0445 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0430\u0445.<\/p>\n<h4>\u0423\u0434\u043e\u0431\u043d\u044b\u0435 \u0441\u043e\u0431\u044b\u0442\u0438\u044f<\/h4>\n<p>\u041e\u0447\u0435\u0432\u0438\u0434\u043d\u043e, \u0447\u0442\u043e \u0440\u0430\u043d\u043e \u0438\u043b\u0438 \u043f\u043e\u0437\u0434\u043d\u043e \u043d\u0430\u0448\u0435\u043c\u0443 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u0443 \u043f\u0440\u0438\u0434\u0435\u0442\u0441\u044f \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438 \u0441\u043e\u0431\u044b\u0442\u0438\u0439 \u0442\u0438\u043f\u0430 <code>onChange<\/code>. \u0412 \u0446\u0435\u043b\u043e\u043c, \u0437\u0430\u0434\u0430\u0447\u0430 \u043b\u0435\u0433\u043a\u043e \u0440\u0435\u0448\u0430\u0435\u0442\u0441\u044f \u0438 \u0441 \u0442\u0435\u043a\u0443\u0449\u0438\u043c\u0438 \u0430\u0440\u0441\u0435\u043d\u0430\u043b\u043e\u043c \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u043d\u044b\u0445 \u0441\u043e\u0431\u044b\u0442\u0438\u0439. \u0423 \u043d\u0430\u0441 \u043a\u0430\u043a \u043c\u0438\u043d\u0438\u043c\u0443\u043c \u0435\u0441\u0442\u044c <code>onscroll<\/code>. \u041d\u043e \u043a\u0430\u0436\u0434\u044b\u0439, \u043a\u0442\u043e \u0440\u0430\u0431\u043e\u0442\u0430\u043b \u0441 \u044d\u0442\u0438\u043c \u0441\u043e\u0431\u044b\u0442\u0438\u0435\u043c \u0437\u043d\u0430\u0435\u0442, \u0447\u0442\u043e \u043f\u0440\u0438\u0434\u0435\u0442\u0441\u044f \u043f\u0440\u0438\u0431\u0435\u0433\u0430\u0442\u044c \u043a \u0442\u0430\u043a \u043d\u0430\u0437\u044b\u0432\u0430\u0435\u043c\u043e\u043c\u0443 \u0434\u0435\u0431\u0430\u0443\u043d\u0441\u0443, \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u0434\u0435\u0440\u0433\u0430\u0442\u044c \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u043f\u043e \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0434\u0435\u0441\u044f\u0442\u043a\u043e\u0432 \u0440\u0430\u0437 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443.<\/p>\n<p>\u0422\u0430\u043a \u0432\u043e\u0442, \u0432\u0441\u043a\u043e\u0440\u0435 \u0432 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442 HTML \u0432\u0435\u0440\u043e\u044f\u0442\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u044f\u0442 \u0441\u043e\u0431\u044b\u0442\u0438\u044f <code>onscrollsnapchange<\/code> \u0438 <code>onscrollend<\/code>. \u042d\u0442\u0438 \u0441\u043e\u0431\u044b\u0442\u0438\u044f \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0442 \u0440\u0435\u0448\u0438\u0442\u044c \u043d\u0430\u0448\u0443 \u0437\u0430\u0434\u0430\u0447\u0443 \u0432 \u043f\u0430\u0440\u0443 \u0441\u0442\u0440\u043e\u043a. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u0436\u0434\u0435\u043c<\/p>\n<h4>\u0410\u043d\u0438\u043c\u0430\u0446\u0438\u044f \u0441\u043a\u0440\u043e\u043b\u043b\u0430<\/h4>\n<p>\u042d\u0442\u043e \u0432\u043e\u043e\u0431\u0449\u0435 \u0447\u0443\u043c\u0430. \u0421\u0442\u0430\u043d\u0434\u0430\u0440\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0443\u0436\u0435 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0430\u043d \u0432 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0430\u0445 \u043d\u0430 \u0434\u0432\u0438\u0436\u043a\u0435 Chromium, \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u043f\u0440\u044f\u043c \u0432 CSS \u043e\u043f\u0438\u0441\u044b\u0432\u0430\u0442\u044c \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f \u0441\u0432\u043e\u0439\u0441\u0442\u0432 \u043e\u0442\u043d\u043e\u0441\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u043f\u043e\u0437\u0438\u0446\u0438\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0430. \u0422\u043e \u0435\u0441\u0442\u044c \u043a\u043e\u0433\u0434\u0430 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442 \u0437\u0430\u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0432\u043e \u0432\u0441\u0435\u0445 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0430\u0445, \u043c\u043e\u0436\u043d\u043e \u0431\u0443\u0434\u0435\u0442 \u043d\u0430\u043c\u0443\u0442\u0438\u0442\u044c \u0431\u0435\u0441\u043a\u043e\u043d\u0435\u0447\u043d\u043e\u0435 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0439 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u044f \u0431\u0435\u0437 \u0435\u0434\u0438\u043d\u043e\u0439 \u0441\u0442\u0440\u043e\u0447\u043a\u0438 JS-\u043a\u043e\u0434\u0430.<\/p>\n<p>\u041a\u0440\u0430\u0439\u043d\u0435 \u0440\u0435\u043a\u043e\u043c\u0435\u043d\u0434\u0443\u044e \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u043d\u0430 <a href=\"https:\/\/scroll-driven-animations.style\/\" rel=\"noopener noreferrer nofollow\">\u043f\u0440\u0438\u043c\u0435\u0440\u044b<\/a> \u0438 <a href=\"https:\/\/developer.mozilla.org\/en-US\/docs\/Web\/CSS\/CSS_scroll-driven_animations\" rel=\"noopener noreferrer nofollow\">\u0434\u043e\u043a\u0443<\/a>. \u0422\u0430\u043c \u0432\u0441\u0451 \u043e\u0447\u0435\u043d\u044c \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e.<\/p>\n<h3>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435<\/h3>\n<p>\u0412 \u0437\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0438 \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u0441\u043a\u0430\u0437\u0430\u0442\u044c, \u0447\u0442\u043e \u044f \u0431\u0435\u0437\u0443\u043c\u043d\u043e \u0440\u0430\u0434 \u0442\u043e\u043c\u0443, \u043a\u0430\u043a \u0441\u0442\u0440\u0435\u043c\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0438 \u043c\u0430\u0441\u0448\u0442\u0430\u0431\u043d\u043e \u0440\u0430\u0437\u0432\u0438\u0432\u0430\u044e\u0442\u0441\u044f \u0432\u0435\u0431 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u044b. \u0420\u0435\u043a\u043e\u043c\u0435\u043d\u0434\u0443\u044e \u0441\u043b\u0435\u0434\u0438\u0442\u044c \u0437\u0430 \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u044f\u043c\u0438 \u0438 \u043f\u0440\u0438 \u0441\u0442\u043e\u043b\u043a\u043d\u043e\u0432\u0435\u043d\u0438\u0438 \u0441 \u043e\u0447\u0435\u0440\u0435\u0434\u043d\u043e\u0439 \u0437\u0430\u0434\u0430\u0447\u0435\u0439 \u043f\u0440\u0438\u043c\u0435\u043d\u044f\u0442\u044c \u043d\u0430\u0442\u0438\u0432\u043d\u044b\u0435 \u0442\u0435\u0445\u043d\u043e\u043b\u043e\u0433\u0438\u0438, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0437\u0430\u0447\u0430\u0441\u0442\u0443\u044e \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u0441\u043e\u0437\u0434\u0430\u0442\u044c \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u043e\u0435 \u043a\u0440\u0430\u0441\u0438\u0432\u043e\u0435, \u043a\u043e\u043c\u043f\u0430\u043a\u0442\u043d\u043e\u0435, \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0435 \u0440\u0435\u0448\u0435\u043d\u0438\u0435.<\/p>\n<p>\u0420\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0435\u043d\u043d\u043e\u0435 \u0432 \u0441\u0442\u0430\u0442\u044c\u0435 \u0440\u0435\u0448\u0435\u043d\u0438\u0435 \u043e\u0431\u043e\u0448\u043b\u043e\u0441\u044c \u0431\u0435\u0437 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439 \u0438 \u0437\u0430\u043d\u044f\u043b\u043e \u0447\u0443\u0442\u044c \u0431\u043e\u043b\u044c\u0448\u0435 100 \u0441\u0442\u0440\u043e\u043a \u043a\u043e\u0434\u0430.<\/p>\n<\/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\/876202\/\"> https:\/\/habr.com\/ru\/articles\/876202\/<\/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<h3>\u041f\u0430\u0440\u0430 \u0441\u043b\u043e\u0432 \u043e \u0441\u0435\u0431\u0435<\/h3>\n<p>\u041f\u0440\u0438\u0432\u0435\u0442, \u043c\u0435\u043d\u044f \u0437\u043e\u0432\u0443\u0442 \u0410\u0440\u0442\u0443\u0440, \u0438 \u044f \u043b\u044e\u0431\u043b\u044e \u043f\u043b\u0430\u0432\u043d\u044b\u0435 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u044b, CSS \u0438 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u043f\u043e\u0434\u0445\u043e\u0434\u044b. \u0421\u0435\u0433\u043e\u0434\u043d\u044f \u0445\u043e\u0447\u0443 \u043f\u043e\u0433\u043e\u0432\u043e\u0440\u0438\u0442\u044c \u0441 \u0432\u0430\u043c\u0438 \u043e \u0437\u0430\u0434\u0430\u0447\u0435, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u043d\u0430\u0441\u0442\u0438\u0433\u0430\u0435\u0442, \u043f\u043e\u0436\u0430\u043b\u0443\u0439, \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0444\u0440\u043e\u043d\u0442\u0435\u043d\u0434 \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0430: \u043e \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u0438 \u0433\u0430\u043b\u0435\u0440\u0435\u0438.<\/p>\n<h3>\u041f\u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u0437\u0430\u0434\u0430\u0447\u0438<\/h3>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0442\u0430\u043a, \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e \u044d\u0442\u043e \u0438 \u043d\u0435 \u0433\u0430\u043b\u0435\u0440\u0435\u044f \u0432\u043e\u0432\u0441\u0435. \u041d\u0430\u0437\u0432\u0430\u043d\u0438\u0435 \u043d\u0435 \u0442\u0430\u043a \u0432\u0430\u0436\u043d\u043e, \u0432\u0430\u0436\u043d\u0430 \u0437\u0430\u0434\u0430\u0447\u0430. \u041e\u043f\u0438\u0441\u0430\u0442\u044c \u0435\u0451 \u043c\u043e\u0436\u043d\u043e \u0442\u0430\u043a: \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u0430 mobile-first \u0441\u0432\u0430\u0439\u043f\u0430\u043b\u043a\u0430 \u0441\u043b\u0430\u0439\u0434\u043e\u0432. \u0421\u0432\u0430\u0439\u043f\u044b \u0434\u043e\u043b\u0436\u043d\u044b \u0432\u044b\u0433\u043b\u044f\u0434\u0435\u0442\u044c \u043c\u0430\u043a\u0441\u0438\u043c\u0430\u043b\u044c\u043d\u043e \u043d\u0430\u0442\u0438\u0432\u043d\u043e \u0438 \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0431\u0435\u0437 \u0441\u044a\u0435\u0434\u0430\u043d\u0438\u044f \u043a\u0430\u0434\u0440\u043e\u0432. \u0425\u043e\u0442\u044c \u043e\u043d\u0430 \u0438 mobile-first, \u0434\u0435\u0441\u043a\u0442\u043e\u043f\u044b \u0441 \u0438\u0445 \u043c\u044b\u0448\u043a\u0430\u043c\u0438 \u0441\u0442\u043e\u0440\u043e\u043d\u043e\u0439 \u043e\u0431\u0445\u043e\u0434\u0438\u0442\u044c \u043d\u0435\u043b\u044c\u0437\u044f.<\/p>\n<h3>\u041a\u0430\u043a \u044d\u0442\u043e \u0434\u0435\u043b\u0430\u043b\u043e\u0441\u044c \u0432\u0435\u043a\u0430\u043c\u0438<\/h3>\n<p>\u0414\u043e\u043b\u0433\u0438\u0435 \u0433\u043e\u0434\u044b, \u0443\u0436\u0435 \u0434\u0430\u0436\u0435 \u0434\u0435\u0441\u044f\u0442\u0438\u043b\u0435\u0442\u0438\u044f, \u043f\u0440\u0438 \u043f\u0440\u043e\u0438\u0437\u043d\u0435\u0441\u0435\u043d\u0438\u0438 \u0434\u0438\u0437\u0430\u0439\u043d\u0435\u0440\u043e\u043c \u0441\u043b\u043e\u0432\u0430 \u00ab\u0441\u0432\u0430\u0439\u043f\u00bb, \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u0441\u0440\u0430\u0437\u0443 \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u0433\u0440\u0443\u0441\u0442\u043d\u0435\u043b, \u043e\u0441\u043e\u0437\u043d\u0430\u0432\u0430\u044f, \u0447\u0442\u043e \u0441\u0435\u0439\u0447\u0430\u0441 \u043f\u0440\u0438\u0434\u0435\u0442\u0441\u044f \u0438\u043c\u0435\u0442\u044c \u0434\u0435\u043b\u043e \u0441\u043e \u0432\u0441\u044f\u043a\u043e\u0433\u043e \u0440\u043e\u0434\u0430 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u044f\u043c\u0438 \u043d\u0430 JS, \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0441 touch-\u0441\u043e\u0431\u044b\u0442\u0438\u044f\u043c\u0438 \u0438 \u0432\u0441\u043f\u043e\u043c\u0438\u043d\u0430\u0442\u044c \u043a\u0430\u043a \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e position \u0432 CSS (\u0434\u0435\u0441\u044f\u0442\u043a\u0438 \u0441\u043e\u0431\u0435\u0441\u043e\u0432 \u0443\u0431\u0435\u0434\u0438\u043b\u0438 \u043c\u0435\u043d\u044f \u0432 \u0442\u043e\u043c, \u0447\u0442\u043e \u0434\u0430\u043b\u0435\u043a\u043e \u043d\u0435 \u0432\u0441\u0435 \u043f\u043e\u043d\u0438\u043c\u0430\u044e\u0442, \u043a\u0430\u043a \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u044d\u0442\u043e \u043d\u0430 \u043f\u0435\u0440\u0432\u044b\u0439 \u0432\u0437\u0433\u043b\u044f\u0434 \u043f\u0440\u043e\u0441\u0442\u043e\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e).<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0434\u0438\u0437\u0430\u0439\u043d\u0435\u0440 \u043b\u044e\u0442\u043e\u0432\u0430\u043b, \u0442\u043e \u043d\u0443\u0436\u043d\u043e \u0431\u044b\u043b\u043e \u0435\u0449\u0435 \u0438 \u0438\u0437\u0443\u0447\u0430\u0442\u044c \u043d\u0435\u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0437\u0430\u043a\u043e\u043d\u044b \u0444\u0438\u0437\u0438\u043a\u0438, \u043f\u044b\u0442\u0430\u044f\u0441\u044c \u043f\u043e\u043d\u044f\u0442\u044c, \u043a\u0430\u043a \u043d\u0430 JS \u0432\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u0435\u0441\u0442\u0438 \u044d\u0444\u0444\u0435\u043a\u0442 \u043f\u0440\u0443\u0436\u0438\u043d\u044b: \u0438\u043c\u0435\u043d\u043d\u043e \u0442\u0430\u043a \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0434\u043e\u0448\u0435\u0434\u0448\u0438\u0439 \u0434\u043e \u0433\u0440\u0430\u043d\u0438\u0446\u044b \u0441\u0432\u0430\u0439\u043f \u043d\u0430 \u043f\u043e\u0447\u0442\u0438 \u0432\u0441\u0435\u0445 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u0441\u043c\u0430\u0440\u0442\u0444\u043e\u043d\u0430\u0445.<\/p>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e2f79e4767f9b8c8d14c\" data-style=\"\" id=\"6792e2f79e4767f9b8c8d14c\" width=\"\"><\/div>\n<p>Anyway, \u0432\u043e\u043e\u0440\u0443\u0436\u0438\u0432\u0448\u0438\u0441\u044c \u0437\u0430\u043a\u043e\u043d\u0430\u043c\u0438 \u0434\u0438\u043d\u0430\u043c\u0438\u043a\u0438, \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430\u043c\u0438 position\/left\/right\/transform \u0438 \u0441\u043e\u0431\u044b\u0442\u0438\u044f\u043c\u0438 \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430 pointerevents, \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438 \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u044e\u0442\u0441\u044f \u0437\u0430 \u0434\u0435\u043b\u043e. \u0421\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0435\u043c, \u043e\u0434\u0438\u043d \u0438\u0437 \u043d\u0438\u0445 \u043f\u043b\u044e\u043d\u0435\u0442 \u0438 \u0437\u0430\u0433\u0443\u0433\u043b\u0438\u0442 \u00abthe most popular swipeable gallery on JS\u00bb, \u0432\u0442\u043e\u0440\u043e\u0439 \u2014\u00a0\u0434\u043e\u0432\u0435\u0434\u0435\u0442 \u043d\u0430\u0447\u0430\u0442\u043e\u0435 \u0434\u043e \u043a\u043e\u043d\u0446\u0430 \u0438 \u0437\u0430\u043c\u0435\u0442\u0438\u0442, \u0447\u0442\u043e \u043f\u0440\u0438 \u0432\u0441\u0435\u0445 \u0435\u0433\u043e \u0443\u0441\u0438\u043b\u0438\u044f\u0445 \u044d\u0444\u0444\u0435\u043a\u0442\u0430 \u043d\u0430\u0442\u0438\u0432\u043d\u043e\u0441\u0442\u0438 \u0434\u043e\u0431\u0438\u0442\u044c\u0441\u044f \u043d\u0435 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f.<\/p>\n<p>\u0421\u043b\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u0434\u043e\u0431\u043d\u044b\u0445 \u043f\u043e\u0434\u0445\u043e\u0434\u043e\u0432 \u0437\u0430\u043a\u043b\u044e\u0447\u0430\u0435\u0442\u0441\u044f \u0432 \u0442\u043e\u043c, \u0447\u0442\u043e \u043f\u043e\u043c\u0438\u043c\u043e \u0438\u0437\u0443\u0447\u0435\u043d\u0438\u044f \u0432\u0441\u0435\u0433\u043e \u0432\u044b\u0448\u0435\u043f\u0435\u0440\u0435\u0447\u0438\u0441\u043b\u0435\u043d\u043d\u043e\u0433\u043e \u043d\u0443\u0436\u043d\u043e \u0435\u0449\u0435 \u0438 \u0431\u044b\u0442\u044c \u044d\u043a\u0441\u043f\u0435\u0440\u0442\u043e\u043c \u0432 \u0442\u043e\u043c, \u043a\u0430\u043a \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433 \u0432 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0435, \u0432\u0435\u0434\u044c \u043f\u043e\u0432\u0435\u0441\u0438\u0442\u044c \u0432\u0441\u0435 \u0441\u0432\u043e\u0438 \u043f\u0440\u0435\u043a\u0440\u0430\u0441\u043d\u043e \u043e\u043f\u0438\u0441\u0430\u043d\u043d\u044b\u0435 \u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u0438\u044f \u043d\u0430 pointermove \u2014 \u044d\u0442\u043e \u043a\u0430\u043a \u043f\u0440\u0430\u0432\u0438\u043b\u043e \u0444\u0430\u0442\u0430\u043b\u0438\u0442\u0438 \u0441 \u0442\u043e\u0447\u043a\u0438 \u0437\u0440\u0435\u043d\u0438\u044f \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438 \u0440\u0435\u043d\u0434\u0435\u0440\u0438\u043d\u0433\u0430.<\/p>\n<p>\u041a\u043e\u0440\u043e\u0447\u0435, \u0447\u0435\u0433\u043e \u043c\u044b \u043c\u0443\u0447\u0430\u0435\u043c\u0441\u044f? \u041c\u044b \u0432 2025 \u0438\u043b\u0438 \u0433\u0434\u0435?<\/p>\n<h3>\u0421\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0439 CSS \u0431\u0435\u0436\u0438\u0442 \u043d\u0430 \u043f\u043e\u043c\u043e\u0449\u044c<\/h3>\n<p>\u041d\u0435 \u0437\u043d\u0430\u044e \u043a\u0430\u043a \u0432\u044b, \u0430 \u044f, \u043a\u043e\u0433\u0434\u0430 \u0432\u0438\u0436\u0443 \u0433\u0430\u043b\u0435\u0440\u0435\u044e, \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u043b\u044f\u044e \u0441\u0435\u0431\u0435 \u0431\u043b\u043e\u043a \u0441\u043e \u0441\u043a\u0440\u043e\u043b\u043b\u043e\u043c. \u0421\u043a\u0440\u043e\u043b\u043b \u043f\u0440\u0435\u043a\u0440\u0430\u0441\u0435\u043d \u0442\u0435\u043c, \u0447\u0442\u043e \u043e\u043d \u0434\u0435\u043b\u0430\u0435\u0442 \u0437\u0430 \u043d\u0430\u0441 \u043c\u043e\u0440\u0435 \u0440\u0430\u0431\u043e\u0442\u044b:<\/p>\n<ul>\n<li>\n<p>\u041e\u0431\u0435\u0441\u043f\u0435\u0447\u0438\u0432\u0430\u0435\u0442 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432 \u0431\u0435\u0437 JS<\/p>\n<\/li>\n<li>\n<p>\u041e\u0431\u0435\u0441\u043f\u0435\u0447\u0438\u0432\u0430\u0435\u0442 overscroll \u043d\u0430 \u043a\u0440\u0430\u044f\u0445 \u0431\u0435\u0437 JS<\/p>\n<\/li>\n<li>\n<p>\u041f\u0440\u0438 \u0436\u0435\u043b\u0430\u043d\u0438\u0438 \u0434\u0430\u0435\u0442 \u0441\u043b\u0435\u0434\u0438\u0442\u044c \u043d\u0430 \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441\u043e\u043c \u0447\u0435\u0440\u0435\u0437 DOM-\u0441\u043e\u0431\u044b\u0442\u0438\u044f<\/p>\n<\/li>\n<\/ul>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u043f\u043e\u043f\u0440\u043e\u0431\u0443\u0435\u043c \u0438\u043c\u043f\u043b\u0435\u043c\u0435\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430\u0448\u0443 \u0433\u0430\u043b\u0435\u0440\u0435\u044e \u0447\u0435\u0440\u0435\u0437 \u0433\u043e\u0440\u0438\u0437\u043e\u043d\u0442\u0430\u043b\u044c\u043d\u044b\u0439 \u0441\u043a\u0440\u043e\u043b\u043b.<\/p>\n<h3>\u0413\u0430\u043b\u0435\u0440\u0435\u044f \u043a\u0430\u043a \u0431\u043b\u043e\u043a \u0441 \u0433\u043e\u0440\u0438\u0437\u043e\u043d\u0442\u0430\u043b\u044c\u043d\u044b\u043c \u0441\u043a\u0440\u043e\u043b\u043b\u043e\u043c<\/h3>\n<pre><code class=\"typescript\">import { Children, ReactNode } from \"react\";  export const Gallery = ({ children }: { children?: ReactNode; }) =&gt; {   return (     &lt;div&gt;       &lt;div style={{ overflowX: \"auto\", display: \"flex\" }}&gt;         {Children.map(children, (child, i) =&gt; (           &lt;div style={{ width: \"100%\", flexShrink: 0 }} key={i}&gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<p>\u0412\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0438\u0437\u0443\u043c\u0438\u0442\u0435\u043b\u044c\u043d\u043e, \u043a\u0430\u043a \u043f\u043e \u043c\u043d\u0435, \u043d\u043e \u043e\u0447\u0435\u0432\u0438\u0434\u043d\u043e \u043c\u044b \u0442\u0443\u0442 \u0432\u0441\u0442\u0440\u0435\u0442\u0438\u043c\u0441\u044f \u0441 \u0440\u044f\u0434\u043e\u043c \u043f\u0440\u043e\u0431\u043b\u0435\u043c:<\/p>\n<ol>\n<li>\n<p>\u0412\u0438\u0434\u0438\u043c\u043e\u0441\u0442\u044c \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440\u0430.<\/p>\n<\/li>\n<li>\n<p>\u0421\u043a\u0432\u043e\u0437\u043d\u043e\u0435 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u0435: \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e, \u0447\u0442\u043e\u0431\u044b \u043f\u043e\u0441\u043b\u0435 \u0441\u0432\u0430\u0439\u043f\u0430 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u043b\u0430 \u043f\u043b\u0430\u0432\u043d\u0430\u044f \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u043d\u0430 \u0433\u0440\u0430\u043d\u0438\u0446\u0435 \u043c\u0435\u0436\u0434\u0443 \u0441\u043b\u0430\u0439\u0434\u0430\u043c\u0438.<\/p>\n<\/li>\n<\/ol>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e322ca986ae5034d1d08\" data-style=\"\" id=\"6792e322ca986ae5034d1d08\" width=\"\"><\/div>\n<h3>\u0423\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440<\/h3>\n<p>\u0420\u0430\u043d\u044c\u0448\u0435, \u0447\u0442\u043e\u0431\u044b \u0443\u0431\u0440\u0430\u0442\u044c \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440, \u043d\u0443\u0436\u043d\u043e \u0431\u044b\u043b\u043e \u0445\u0438\u043c\u0438\u0447\u0438\u0442\u044c \u0441 \u0434\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u043c\u0438 HTML-\u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430\u043c\u0438, \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e\u043c <code>overflow<\/code> \u0438 \u0445\u0430\u0440\u0434\u043a\u043e\u0434\u043e\u043c \u0433\u0430\u0431\u0430\u0440\u0438\u0442\u043e\u0432 \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440\u0430.<\/p>\n<p>\u041f\u043e\u0442\u043e\u043c \u0443 \u043d\u0430\u0441 \u043f\u043e\u044f\u0432\u0438\u043b\u0441\u044f \u0437\u0430\u043c\u0435\u0447\u0430\u0442\u0435\u043b\u044c\u043d\u044b\u0439 <code>::-webkit-scrollbar-*<\/code>.<\/p>\n<p>\u0421 \u0434\u0435\u043a\u0430\u0431\u0440\u044f 2024-\u0433\u043e \u0432\u0441\u0435\u043c \u044d\u0442\u0438\u043c \u0437\u0430\u043d\u0438\u043c\u0430\u0442\u044c\u0441\u044f \u043d\u0435 \u043d\u0443\u0436\u043d\u043e: \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scrollbar-width<\/code> \u0441\u0442\u0430\u043b\u043e \u0434\u043e\u0441\u0442\u0443\u043f\u043d\u043e \u0432\u043e <a href=\"https:\/\/caniuse.com\/mdn-css_properties_scrollbar-width_none\" rel=\"noopener noreferrer nofollow\">\u0432\u0441\u0435\u0445 \u043f\u043e\u043f\u0443\u043b\u044f\u0440\u043d\u044b\u0445 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0430\u0445<\/a>:<\/p>\n<pre><code class=\"typescript\">import { Children, ReactNode } from \"react\";  export const Gallery = ({ children }: { children?: ReactNode; }) =&gt; {   return (     &lt;div&gt;       &lt;div          style={{            overflowX: \"auto\",            display: \"flex\",           scrollbarWidth: \"none\"         }}       &gt;         {Children.map(children, (child, i) =&gt; (           &lt;div style={{ width: \"100%\", flexShrink: 0 }} key={i}&gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e342999b83f994c8b5e7\" data-style=\"\" id=\"6792e342999b83f994c8b5e7\" width=\"\"><\/div>\n<h3>\u0423\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u043a\u0432\u043e\u0437\u043d\u043e\u0435 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u0435<\/h3>\n<p>\u0421\u0430\u043c\u043e\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0435. 2024-\u0439 \u043f\u043e\u0434\u0430\u0440\u0438\u043b \u043d\u0430\u043c \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e \u043c\u0430\u0441\u0441\u043e\u0432\u0443\u044e \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0443 \u043d\u0435\u0432\u0438\u0434\u0438\u043c\u043e\u0433\u043e \u0441\u043a\u0440\u043e\u043b\u043b\u0431\u0430\u0440\u0430, \u043d\u043e \u0438 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0443 \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430 \u0441\u0432\u043e\u0439\u0441\u0442\u0432 <code>scroll-snap<\/code>. Snap \u0432 \u043f\u0435\u0440\u0435\u0432\u043e\u0434\u0435 \u0441 \u0430\u043d\u0433\u043b\u0438\u0439\u0441\u043a\u043e\u0433\u043e \u043e\u0437\u043d\u0430\u0447\u0430\u0435\u0442 \u00ab\u0449\u0435\u043b\u0447\u043e\u043a\u00bb. \u042d\u0442\u043e\u0442 \u0449\u0435\u043b\u0447\u043e\u043a \u2014 \u043a\u0430\u043a \u0440\u0430\u0437 \u0442\u043e, \u0447\u0442\u043e \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e.<\/p>\n<h4>\u041f\u0430\u0440\u0443 \u0441\u043b\u043e\u0432 \u043e\u0431 \u044d\u0442\u043e\u043c \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0435.<\/h4>\n<p>\u0415\u0441\u043b\u0438 \u0433\u043e\u0432\u043e\u0440\u0438\u0442\u044c \u043e\u0431\u0449\u043e, \u0442\u043e \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u044d\u0442\u043e\u0433\u043e \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u0433\u0438\u0431\u043a\u043e \u043d\u0430\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c \u043f\u043e\u0432\u0435\u0434\u0435\u043d\u0438\u0435 \u0441\u043a\u0440\u043e\u043b\u043b\u0430, \u043e\u0441\u043e\u0431\u0435\u043d\u043d\u043e \u043a\u043e\u0433\u0434\u0430 \u0432 \u0431\u043b\u043e\u043a\u0435 \u043f\u0440\u0438\u0441\u0443\u0442\u0441\u0442\u0432\u0443\u044e\u0442 \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b. \u041a\u0430\u0436\u0434\u044b\u0439 \u044d\u043b\u0435\u043c\u0435\u043d\u0442 \u043c\u043e\u0436\u0435\u0442 \u0441\u0442\u0430\u0442\u044c \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u043e\u0439 \u0442\u043e\u0447\u043a\u043e\u0439 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0430. \u0417\u0430 \u044d\u0442\u043e \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scroll-snap-type<\/code>.<\/p>\n<p>\u041f\u043e\u043c\u0438\u043c\u043e \u0442\u043e\u0447\u0435\u043a \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u043c\u043e\u0436\u043d\u043e \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0442\u0430\u043a\u0438\u0435 \u0432\u0435\u0449\u0438 \u043a\u0430\u043a \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435 (<code>scroll-snap-padding<\/code>, <code>scroll-snap-margin<\/code>) \u043c\u0435\u0436\u0434\u0443 \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u044b\u043c\u0438 \u0442\u043e\u0447\u043a\u0430\u043c\u0438, \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u0435 \u044d\u0442\u0438\u0445 \u0442\u043e\u0447\u0435\u043a (<code>scroll-snap-align<\/code>) \u0438 \u0442\u043e, \u043c\u043e\u0436\u043d\u043e \u043b\u0438 \u043f\u0440\u043e\u0441\u043a\u0430\u043a\u0438\u0432\u0430\u0442\u044c \u0442\u043e\u0447\u043a\u0438 \u0432 \u0441\u043b\u0443\u0447\u0430\u0435, \u0435\u0441\u043b\u0438 \u0438\u043d\u0442\u0435\u043d\u0441\u0438\u0432\u043d\u043e\u0441\u0442\u044c \u0436\u0435\u0441\u0442\u0430 \u044d\u0442\u043e \u043f\u043e\u0434\u0440\u0430\u0437\u0443\u043c\u0435\u0432\u0430\u0435\u0442 (<code>scroll-snap-stop<\/code>).<\/p>\n<p>\u041c\u044b \u043d\u0435 \u0431\u0443\u0434\u0435\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0432\u0441\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u0441\u0435\u043c\u0435\u0439\u0441\u0442\u0432\u0430, \u043d\u043e \u0435\u0441\u043b\u0438 \u0432\u0430\u043c \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u0438\u0437\u0443\u0447\u0438\u0442\u044c \u043a\u0430\u043a\u043e\u0435-\u0442\u043e \u0438\u0437 \u043d\u0438\u0445, \u0440\u0435\u043a\u043e\u043c\u0435\u043d\u0434\u0443\u044e <a href=\"https:\/\/developer.mozilla.org\/en-US\/docs\/Web\/CSS\/CSS_scroll_snap\" rel=\"noopener noreferrer nofollow\">\u0441\u0442\u0430\u0442\u044c\u044e \u0438\u0437 MDN<\/a>. \u0422\u0430\u043c \u0438 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0435 \u043f\u0440\u0438\u043c\u0435\u0440\u044b, \u0438 \u043f\u043e\u0434\u0440\u043e\u0431\u043d\u043e\u0435 \u043e\u043f\u0438\u0441\u0430\u043d\u0438\u0435, \u0438 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044f \u043e \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u043d\u043e\u0439 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0435.<\/p>\n<p>\u041d\u0430\u0448\u0430 \u0437\u0430\u0434\u0430\u0447\u0430 \u0441\u0435\u0439\u0447\u0430\u0441 \u2014\u00a0\u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u044b\u0435 \u0442\u043e\u0447\u043a\u0438 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0430. \u0414\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u0432\u043e\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c\u0441\u044f \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430\u043c\u0438 <code>scroll-snap-type<\/code> \u0438 <code>scroll-snap-stop<\/code>.<\/p>\n<h4>\u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u044b\u0435 \u0442\u043e\u0447\u043a\u0438 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0430<\/h4>\n<pre><code class=\"typescript\">&lt;div    style={{      overflowX: \"auto\",      display: \"flex\",     scrollbarWidth: \"none\",     scrollSnapType: \"x mandatory\",     scrollSnapStop: \"always\"   }} &gt;   {Children.map(children, (child, i) =&gt; (     &lt;div style={{ width: \"100%\", flexShrink: 0 }} key={i}&gt;       {child}     &lt;\/div&gt;   ))} &lt;\/div&gt;<\/code><\/pre>\n<p>\u041a\u0430\u043a \u0432\u0438\u0434\u0438\u0442\u0435, \u043d\u0435\u0434\u043e\u0441\u0442\u0430\u044e\u0449\u0430\u044f \u043c\u0435\u0445\u0430\u043d\u0438\u043a\u0430 \u043f\u0440\u043e\u043b\u0438\u0441\u0442\u044b\u0432\u0430\u043d\u0438\u044f \u0433\u0430\u043b\u0435\u0440\u0435\u0438 \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u0442\u0441\u044f \u043f\u0443\u0442\u0435\u043c \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u0431\u0443\u043a\u0432\u0430\u043b\u044c\u043d\u043e \u0434\u0432\u0443\u0445 CSS-\u0441\u0432\u043e\u0439\u0441\u0442\u0432.<\/p>\n<p>\u0417\u043d\u0430\u0447\u0435\u043d\u0438\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 <code>scroll-snap-type: x mandatory<\/code> \u043e\u0437\u043d\u0430\u0447\u0430\u0435\u0442, \u0447\u0442\u043e \u043f\u043e \u043e\u0441\u0438 X \u0441\u043a\u0440\u043e\u043b\u043b \u0434\u043e\u043b\u0436\u0435\u043d \u043e\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043d\u0430 \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u043e\u0439 \u0442\u043e\u0447\u043a\u0435. \u041f\u043e\u043c\u0438\u043c\u043e mandatory, \u0435\u0441\u0442\u044c \u0435\u0449\u0435 proximity, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0434\u0435\u043b\u0435\u0433\u0438\u0440\u0443\u0435\u0442 \u0440\u0435\u0448\u0435\u043d\u0438\u0435 \u043e\u0431 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0435 \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u043e\u043c\u0443 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0443. \u0417\u0434\u0435\u0441\u044c \u043d\u0430\u043c \u044d\u0442\u043e \u043d\u0438 \u043a \u0447\u0435\u043c\u0443. \u041f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u0447\u043d\u043e\u0439 \u0442\u043e\u0447\u043a\u043e\u0439 \u0441\u0435\u0439\u0447\u0430\u0441 \u044f\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u043b\u0435\u0432\u044b\u0439 \u043a\u0440\u0430\u0439 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0434\u043e\u0447\u0435\u0440\u043d\u0435\u0433\u043e \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430 (\u0441\u043b\u0430\u0439\u0434\u0430).<\/p>\n<p>\u0417\u043d\u0430\u0447\u0435\u043d\u0438\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430 <code>scroll-snap-stop: always<\/code> \u0433\u043e\u0432\u043e\u0440\u0438\u0442 \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0443 \u043d\u0435 \u043f\u0440\u043e\u0433\u043b\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u0441\u043b\u0430\u0439\u0434, \u0434\u0430\u0436\u0435 \u0435\u0441\u043b\u0438 \u0441\u0432\u0430\u0439\u043f \u0431\u044b\u043b \u043e\u0447\u0435\u043d\u044c \u0441\u0438\u043b\u044c\u043d\u044b\u043c.<\/p>\n<div class=\"tm-iframe_temp\" data-src=\"https:\/\/embedd.srv.habr.com\/iframe\/6792e35bf97ec4e4dc644e25\" data-style=\"\" id=\"6792e35bf97ec4e4dc644e25\" width=\"\"><\/div>\n<h4>\u041a\u0430\u0441\u0442\u043e\u043c\u043d\u0430\u044f \u0448\u0438\u0440\u0438\u043d\u0430 \u0441\u043b\u0430\u0439\u0434\u0430, \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u0435 \u0438 \u0437\u0430\u0437\u043e\u0440 \u043c\u0435\u0436\u0434\u0443 \u0441\u043b\u0430\u0439\u0434\u0430\u043c\u0438<\/h4>\n<p>\u0421\u043b\u0430\u0439\u0434\u044b \u043d\u0435 \u0432\u0441\u0435\u0433\u0434\u0430 \u0434\u043e\u043b\u0436\u043d\u044b \u0431\u044b\u0442\u044c \u0448\u0438\u0440\u0438\u043d\u043e\u0439 \u0441 \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044f, \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u043f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u044e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e \u0434\u043b\u044f \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0448\u0438\u0440\u0438\u043d\u044b. \u0410 \u0440\u0430\u0437 \u0434\u0430\u0435\u043c \u043f\u043e\u0438\u0433\u0440\u0430\u0442\u044c\u0441\u044f \u0441 \u0448\u0438\u0440\u0438\u043d\u043e\u0439, \u0442\u043e \u0438 \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u0435 \u043e\u0442\u043d\u043e\u0441\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044f \u0442\u043e\u0436\u0435 \u043f\u0440\u0438\u0433\u043e\u0434\u0438\u0442\u0441\u044f. \u0414\u043b\u044f \u0432\u044b\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u043d\u0438\u044f \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0443\u043f\u043e\u043c\u044f\u043d\u0443\u0442\u043e\u0435 \u0432\u044b\u0448\u0435 CSS-\u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scroll-snap-align<\/code>. \u041d\u0443 \u0438 \u0434\u0430\u0434\u0438\u043c \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u043c\u0435\u0436\u0434\u0443 \u0441\u043b\u0430\u0439\u0434\u0430\u043c\u0438. \u0422\u0443\u0442 \u043d\u0435 \u0431\u0443\u0434\u0435\u043c \u043d\u0438\u0447\u0435\u0433\u043e \u0432\u044b\u0434\u0443\u043c\u044b\u0432\u0430\u0442\u044c \u0438 \u043f\u0440\u0438\u043c\u0435\u043d\u0438\u043c \u0441\u0442\u0430\u0440\u044b\u0439 \u0441\u043e\u0432\u0435\u0441\u0442\u043a\u0438\u0439 <code>gap<\/code>:<\/p>\n<pre><code class=\"typescript\">import { Children, ReactNode } from \"react\";  export const Gallery = ({   children,    slideWidth = \"100%\",   slideAlign = \"center\",   slidesGap = 0 }: {    children?: ReactNode;   slideWidth?: string | number;   slideAlign?: \"start\" | \"center\" | \"end\";   slidesGap?: string | number; }) =&gt; {   return (     &lt;div&gt;       &lt;div          style={{            overflowX: \"auto\",            display: \"flex\",           scrollbarWidth: \"none\",           scrollSnapType: \"x mandatory\",           scrollSnapStop: \"always\",           gap: slidesGap         }}       &gt;         {Children.map(children, (child, i) =&gt; (           &lt;div              style={{               width: slideWidth,               scrollSnapAlign: slideAlign,               flexShrink: 0,             }}              key={i}           &gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<blockquote>\n<p>\u041f\u0440\u043e\u0448\u0443 \u043e\u0431\u0440\u0430\u0442\u0438\u0442\u044c \u0432\u043d\u0438\u043c\u0430\u043d\u0438\u0435 \u043d\u0430 \u0442\u043e, \u0447\u0442\u043e \u0441\u0432\u0435\u0436\u0435\u0438\u0441\u043f\u0435\u0447\u0435\u043d\u043d\u043e\u0435 \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e <code>scroll-snap-align<\/code> \u0432\u043c\u0435\u0441\u0442\u043e top, left, bottom \u0438 right \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u0435\u0442 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f start \u0438 end. \u042d\u0442\u043e \u0442\u0430\u043a \u043d\u0430\u0437\u044b\u0432\u0430\u0435\u043c\u044b\u0435 logical values: \u043e\u043d\u0438 \u043a\u043e\u043c\u043f\u0430\u043a\u0442\u043d\u0435\u0435 \u0438 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u043d\u0435 \u0434\u0443\u043c\u0430\u0442\u044c \u043e \u043f\u0435\u0440\u0435\u0432\u0435\u0440\u043d\u0443\u0442\u044b\u0445 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430\u0445 (\u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430\u0445 \u043d\u0430 \u0430\u0440\u0430\u0431\u0441\u043a\u043e\u043c \u044f\u0437\u044b\u043a\u0435).<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e, \u0447\u0442\u043e \u0437\u0430 logical values \u0442\u0430\u043a\u0438\u0435, \u0442\u043e \u043c\u043e\u0436\u0435\u0442\u0435 <a href=\"https:\/\/css-tricks.com\/css-logical-properties-and-values\/\" rel=\"noopener noreferrer nofollow\">\u043f\u043e\u0447\u0438\u0442\u0430\u0442\u044c \u0441\u0442\u0430\u0442\u044c\u044e \u043d\u0430 CSS Tricks<\/a> \u0438\u043b\u0438 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c <a href=\"https:\/\/www.youtube.com\/watch?v=UNqH9rT6nqQ\" rel=\"noopener noreferrer nofollow\">\u043c\u043e\u0439 \u0432\u0435\u0431\u0438\u043d\u0430\u0440<\/a>.<\/p>\n<\/blockquote>\n<h3>\u0414\u0430\u0435\u043c \u043c\u0435\u043d\u044f\u0442\u044c \u043d\u0430\u0447\u0430\u043b\u044c\u043d\u044b\u0439 \u0441\u043b\u0430\u0439\u0434<\/h3>\n<p>\u041b\u0430\u0434\u043d\u043e, \u043f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u044e \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u043f\u043e\u043a\u043e\u0434\u0438\u0442\u044c. \u0420\u0430\u0437\u0440\u0430\u0431\u044b \u043f\u043e-\u043b\u044e\u0431\u043e\u043c\u0443 \u0437\u0430\u0445\u043e\u0442\u044f\u0442 \u0440\u0438\u0441\u043e\u0432\u0430\u0442\u044c \u0433\u0430\u043b\u0435\u0440\u0435\u044e \u0441 \u043f\u0435\u0440\u0432\u043e\u043d\u0430\u0447\u0430\u043b\u044c\u043d\u044b\u043c \u0441\u043b\u0430\u0439\u0434\u043e\u043c, \u043e\u0442\u043b\u0438\u0447\u043d\u044b\u043c \u043e\u0442 \u043f\u0435\u0440\u0432\u043e\u0433\u043e. \u0422\u0443\u0442 \u0443\u0436 \u0431\u0435\u0437 TS \u043d\u0435 \u043e\u0431\u043e\u0439\u0442\u0438\u0441\u044c:<\/p>\n<pre><code class=\"typescript\">import { Children, ReactNode, useEffect, useRef } from \"react\";  export const Gallery = ({   children,    slideWidth = \"100%\",    slideAlign = \"center\",   slidesGap = 0,   initialSlideIndex = 0, }: {    children?: ReactNode;   slideWidth?: string | number;   slideAlign?: \"start\" | \"center\" | \"end\";   slidesGap?: string | number;   initialSlideIndex?: number; }) =&gt; {   const scrollContainer = useRef&lt;HTMLDivElement&gt;(null);      useEffect(() =&gt; {     if (initialSlideIndex &gt; 0) {       scrollContainer.current?.children[initialSlideIndex].scrollIntoView({         behavior: \"instant\",         block: \"nearest\",       });     }   }, []);     return (     &lt;div&gt;       &lt;div          style={{            overflowX: \"auto\",            display: \"flex\",           scrollbarWidth: \"none\",           scrollSnapType: \"x mandatory\",           scrollSnapStop: \"always\",           gap: slidesGap         }}         ref={scrollContainer}       &gt;         {Children.map(children, (child, i) =&gt; (           &lt;div              style={{               width: slideWidth,               scrollSnapAlign: slideAlign,               flexShrink: 0,             }}              key={i}           &gt;             {child}           &lt;\/div&gt;         ))}       &lt;\/div&gt;     &lt;\/div&gt;   ); };<\/code><\/pre>\n<h4>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u043f\u043e-\u043f\u043e\u0440\u044f\u0434\u043a\u0443<\/h4>\n<p><code>useRef<\/code> \u0438 <code>useEffect<\/code> \u2014\u00a0\u0441\u0443\u0449\u043d\u043e\u0441\u0442\u0438 \u0438\u0437 \u043c\u0438\u0440\u0430 React, \u043e\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043d\u0430\u0434\u043e\u043b\u0433\u043e \u043d\u0430 \u043d\u0438\u0445 \u043d\u0435 \u0445\u043e\u0447\u0435\u0442\u0441\u044f. \u0427\u0435\u0440\u0435\u0437 <code>useRef<\/code> \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0441\u0441\u044b\u043b\u043a\u0443 \u043d\u0430 DOM-\u044d\u043b\u0435\u043c\u0435\u043d\u0442, \u0447\u0435\u0440\u0435\u0437 <code>useEffect<\/code> \u2014\u00a0\u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c \u0441\u043d\u0438\u043f\u043f\u0435\u0442 \u0432 \u043a\u043e\u043b\u043b\u0431\u044d\u043a\u0435 \u043f\u0440\u0438 \u043f\u0435\u0440\u0432\u043e\u043c \u0440\u0435\u043d\u0434\u0435\u0440\u0435 \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u0430.<\/p>\n<p>\u0418\u0437 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0433\u043e \u0442\u0443\u0442 \u043c\u0435\u0442\u043e\u0434 DOM-\u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430 <code>scrollIntoView<\/code>. \u042d\u0442\u043e \u043c\u0435\u0442\u043e\u0434, \u043f\u0440\u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0435 \u043a\u043e\u0442\u043e\u0440\u043e\u0433\u043e \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044c \u043f\u0440\u043e\u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442 \u0441\u0435\u0431\u044f \u0434\u043e \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430.<\/p>\n<blockquote>\n<p><strong>\u0412\u0430\u0436\u043d\u043e:<\/strong> \u043c\u0435\u0442\u043e\u0434 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0442\u044c \u043d\u0430 \u0434\u043e\u0447\u0435\u0440\u043d\u0435\u043c \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0435, \u0430 \u043d\u0435 \u043d\u0430 \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u0435.<\/p>\n<\/blockquote>\n<p>\u0421\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u0443 \u043c\u0435\u0442\u043e\u0434\u0430 \u043d\u0435 \u043c\u0435\u043d\u0435\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u044b\u0435.<\/p>\n<p><code>behavior<\/code> \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0442\u0438\u043f \u0441\u043a\u0440\u043e\u043b\u043b\u0430: \u043e\u043d \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c \u0438\u043b\u0438 \u043c\u0433\u043d\u043e\u0432\u0435\u043d\u043d\u044b\u043c, \u0438\u043b\u0438 \u043f\u043b\u0430\u0432\u043d\u044b\u043c. \u0414\u043b\u044f \u0440\u0435\u0448\u0435\u043d\u0438\u044f \u043d\u0430\u0448\u0435\u0439 \u0437\u0430\u0434\u0430\u0447\u0438 \u043d\u0443\u0436\u0435\u043d \u0438\u043c\u0435\u043d\u043d\u043e \u043c\u0433\u043d\u043e\u0432\u0435\u043d\u043d\u044b\u0439.<\/p>\n<p><code>block<\/code> \u0440\u0435\u0433\u0443\u043b\u0438\u0440\u0443\u0435\u0442 \u0442\u043e, \u043a \u043a\u0430\u043a\u0438\u043c \u0440\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u0441\u043a\u0438\u043c \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430\u043c \u0431\u0443\u0434\u0435\u0442 \u043f\u0440\u0438\u043c\u0435\u043d\u0435\u043d \u0432\u044b\u0437\u043e\u0432. \u0415\u0441\u043b\u0438 \u043d\u0435 \u0443\u043a\u0430\u0437\u0430\u0442\u044c \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u0435 <strong>nearest<\/strong>, \u043c\u044b \u0443\u0432\u0438\u0434\u0438\u043c, \u0447\u0442\u043e \u043f\u0440\u0438 \u0432\u044b\u0437\u043e\u0432\u0435 \u043e\u0442\u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e \u0441\u043a\u0440\u043e\u043b\u043b \u0433\u0430\u043b\u0435\u0440\u0435\u0438, \u043d\u043e \u0438 \u0441\u043a\u0440\u043e\u043b\u043b \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430. \u0410 \u044d\u0442\u043e \u043d\u0430\u043c \u043d\u0435 \u043d\u0443\u0436\u043d\u043e.<\/p>\n<p>\u0421\u043e \u0432\u0441\u0435\u043c\u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044f\u043c\u0438 \u0438 \u043d\u044e\u0430\u043d\u0441\u0430\u043c\u0438 \u044d\u0442\u043e\u0433\u043e \u043c\u0435\u0442\u043e\u0434\u0430 \u043c\u043e\u0436\u043d\u043e \u043e\u0437\u043d\u0430\u043a\u043e\u043c\u0438\u0442\u044c\u0441\u044f \u0432 <a href=\"https:\/\/developer.mozilla.org\/en-US\/docs\/Web\/API\/Element\/scrollIntoView\" rel=\"noopener noreferrer nofollow\">\u0441\u0442\u0430\u0442\u044c\u0435 \u043e\u0442 MDN<\/a>.<\/p>\n<h3>\u041f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u0430 \u0434\u0435\u0441\u043a\u0442\u043e\u043f\u0430<\/h3>\n<p>\u0412 \u043d\u0430\u0447\u0430\u043b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 \u044f \u043f\u0438\u0441\u0430\u043b \u043e \u0442\u043e\u043c, \u0447\u0442\u043e \u0445\u043e\u0442\u044c \u043c\u044b \u0438 mobile-first, \u0434\u0435\u0441\u043a\u0442\u043e\u043f\u044b \u043d\u0435\u043b\u044c\u0437\u044f \u0441\u043f\u0438\u0441\u044b\u0432\u0430\u0442\u044c \u0441\u043e \u0441\u0447\u0435\u0442\u043e\u0432.<\/p>\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-446016","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/446016","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=446016"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/446016\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=446016"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=446016"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=446016"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}