{"id":327331,"date":"2022-01-10T08:51:11","date_gmt":"2022-01-10T08:51:11","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=327331"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=327331","title":{"rendered":"<span>Bottom Sheet, \u043f\u0435\u0440\u0435\u0439\u0434\u0451\u043c \u043d\u0430 \u00ab\u0442\u044b\u00bb?<\/span>"},"content":{"rendered":"<div><\/div>\n<div id=\"post-content-body\" class=\"article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>Bottom Sheet \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u043b\u044f\u043b\u0441\u044f \u043c\u043d\u0435 \u0441\u043b\u043e\u0436\u043d\u044b\u043c \u0438 \u043d\u0435\u0434\u043e\u0441\u044f\u0433\u0430\u0435\u043c\u044b\u043c. \u042d\u0442\u043e \u0431\u044b\u043b \u0432\u044b\u0437\u043e\u0432! \u042f \u043d\u0435 \u043f\u043e\u043d\u0438\u043c\u0430\u043b, \u0441 \u0447\u0435\u0433\u043e \u043d\u0430\u0447\u0430\u0442\u044c. \u0412\u043e\u0437\u043d\u0438\u043a\u0430\u043b\u043e \u043c\u043d\u043e\u0433\u043e \u0432\u043e\u043f\u0440\u043e\u0441\u043e\u0432: \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c view \u0438\u043b\u0438 view controller? Auto \u0438\u043b\u0438 manual layout? \u041a\u0430\u043a \u0430\u043d\u0438\u043c\u0438\u0440\u043e\u0432\u0430\u0442\u044c? \u041a\u0430\u043a \u0441\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e?<\/p>\n<p>\u041d\u043e \u0432\u0441\u0451 \u0438\u0437\u043c\u0435\u043d\u0438\u043b\u043e\u0441\u044c \u043f\u043e\u0441\u043b\u0435 \u0440\u0430\u0431\u043e\u0442\u044b \u043d\u0430\u0434 Bottom Sheet \u0434\u043b\u044f \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f Joom, \u0433\u0434\u0435 \u043e\u043d \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u043f\u043e\u0432\u0441\u0435\u043c\u0435\u0441\u0442\u043d\u043e. \u0412 \u0442\u043e\u043c \u0447\u0438\u0441\u043b\u0435 \u0438 \u0432 \u0442\u0430\u043a\u0438\u0445 \u043a\u0440\u0438\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u0445 \u0441\u0446\u0435\u043d\u0430\u0440\u0438\u044f\u0445, \u043a\u0430\u043a \u043e\u043f\u043b\u0430\u0442\u0430. \u0422\u0430\u043a \u0447\u0442\u043e \u043c\u043e\u0433\u0443 \u0442\u043e\u0447\u043d\u043e \u0441\u043a\u0430\u0437\u0430\u0442\u044c, \u0447\u0442\u043e \u0432 \u044d\u0442\u043e\u043c \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u0435 \u043c\u044b \u0443\u0432\u0435\u0440\u0435\u043d\u044b. \u041d\u0430\u0441\u0442\u043e\u043b\u044c\u043a\u043e \u0443\u0432\u0435\u0440\u0435\u043d\u044b, \u0447\u0442\u043e \u044f \u0434\u0430\u0436\u0435 <a href=\"https:\/\/youtu.be\/4VLYzbrKe7c\">\u0440\u0430\u0441\u0441\u043a\u0430\u0437\u044b\u0432\u0430\u043b<\/a> \u043e \u043d\u0451\u043c \u043d\u0430 Podlodka iOS crew #7. \u0412 \u0440\u0430\u043c\u043a\u0430\u0445 \u0432\u043e\u0440\u043a\u0448\u043e\u043f\u0430 \u044f \u043f\u043e\u043a\u0430\u0437\u0430\u043b, \u043a\u0430\u043a \u0441\u0434\u0435\u043b\u0430\u0442\u044c Bottom Sheet, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0443\u043c\u0435\u0435\u0442 \u043f\u043e\u0434\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430, \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u0438 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 <em>UINavigationController.<\/em><\/p>\n<p>\u0421\u0442\u043e\u043f, \u043d\u043e Apple \u0436\u0435 <a href=\"https:\/\/developer.apple.com\/videos\/play\/wwdc2021\/10063\/\">\u043f\u0440\u0435\u0434\u043e\u0441\u0442\u0430\u0432\u0438\u043b\u0430<\/a> \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 <a href=\"https:\/\/developer.apple.com\/design\/human-interface-guidelines\/ios\/views\/sheets\/\">Bottom Sheet<\/a>. \u0417\u0430\u0447\u0435\u043c \u043f\u0438\u0441\u0430\u0442\u044c \u0441\u0432\u043e\u0439? \u0414\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e, \u044d\u0442\u043e \u0442\u0430\u043a, \u043d\u043e \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e \u0441 iOS 15. \u0410 \u044d\u0442\u043e \u0437\u043d\u0430\u0447\u0438\u0442, \u0447\u0442\u043e \u043f\u043e\u043b\u043d\u043e\u0446\u0435\u043d\u043d\u043e \u0435\u0433\u043e \u043c\u043e\u0436\u043d\u043e \u0431\u0443\u0434\u0435\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u0447\u0435\u0440\u0435\u0437 2-3 \u0433\u043e\u0434\u0430. \u041a \u0442\u043e\u043c\u0443 \u0436\u0435 \u0447\u0430\u0441\u0442\u043e \u0442\u0440\u0435\u0431\u043e\u0432\u0430\u043d\u0438\u044f \u0434\u0438\u0437\u0430\u0439\u043d\u0435\u0440\u043e\u0432 \u0432\u044b\u0445\u043e\u0434\u044f\u0442 \u0437\u0430 \u0440\u0430\u043c\u043a\u0438 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0445 iOS-\u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432.<\/p>\n<p>\u0412 \u0440\u0430\u043c\u043a\u0430\u0445 \u0441\u0442\u0430\u0442\u044c\u0438 \u0445\u043e\u0447\u0443 \u0440\u0430\u0437\u0432\u0435\u044f\u0442\u044c \u0442\u0443\u043c\u0430\u043d \u043d\u0430\u0434 Bottom Sheet, \u043e\u0442\u0432\u0435\u0442\u0438\u0442\u044c \u043d\u0430 \u0432\u043e\u043f\u0440\u043e\u0441\u044b, \u043a\u043e\u0442\u043e\u0440\u044b\u043c\u0438 \u0437\u0430\u0434\u0430\u0432\u0430\u043b\u0441\u044f \u044f \u0441\u0430\u043c \u0438 \u043f\u0440\u0435\u0434\u043b\u043e\u0436\u0438\u0442\u044c \u043e\u0434\u0438\u043d \u0438\u0437 \u0432\u0430\u0440\u0438\u0430\u043d\u0442\u043e\u0432 \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438. \u0427\u0442\u043e\u0431\u044b \u0432 \u043a\u043e\u043d\u0446\u0435 \u0432\u044b \u043c\u043e\u0433\u043b\u0438 \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432 \u0440\u0435\u0437\u044e\u043c\u0435 \u0441\u0442\u0440\u043e\u0447\u043a\u0443 \u00ab\u041f\u0440\u043e\u0444\u0435\u0441\u0441\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e \u0434\u0435\u043b\u0430\u044e Bottom Sheet&#8217;\u044b\u00bb.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/a4c\/7b8\/7b9\/a4c7b87b98a6cc3b8817488241648b43.png\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0415\u0441\u043b\u0438 \u0437\u0430\u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043e\u0432\u0430\u043b, \u0442\u043e \u043d\u0430\u0447\u043d\u0451\u043c! \u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u043f\u0440\u043e\u0441\u0442\u043e\u0439 Bottom Sheet \u0438 \u0448\u0430\u0433 \u0437\u0430 \u0448\u0430\u0433\u043e\u043c \u0435\u0433\u043e \u043f\u0440\u043e\u043a\u0430\u0447\u0430\u0435\u043c. <\/p>\n<ol>\n<li>\n<p>\u041d\u0430\u0443\u0447\u0438\u043c\u0441\u044f \u043f\u043e\u0434\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0438 \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet.<\/p>\n<\/li>\n<li>\n<p>\u0414\u043e\u0431\u0430\u0432\u0438\u043c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435, \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u044f \u043a\u043e\u043d\u0442\u0435\u043d\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442\u0441\u044f.<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u043c <em>UINavigationController<\/em> \u0441 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u0435\u0439 \u0432\u043d\u0443\u0442\u0440\u0438 Bottom Sheet.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0427\u0430\u0441\u0442\u044c 1. \u0410\u0434\u0430\u043f\u0442\u0438\u0440\u0443\u0435\u043c\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u0417\u0430\u043a\u0440\u044b\u0432\u0430\u0435\u043c Bottom Sheet. \u0411\u0430\u0437\u043e\u0432\u044b\u0439 \u0434\u0438\u0437\u0430\u0439\u043d<\/h3>\n<p>\u0421\u043f\u0435\u0440\u0432\u0430 \u0443\u0431\u0435\u0434\u0438\u043c\u0441\u044f, \u0447\u0442\u043e \u043c\u044b \u043f\u043e\u043d\u0438\u043c\u0430\u0435\u043c \u043f\u043e\u0434 &#171;Bottom Sheet&#187; \u043e\u0434\u043d\u043e \u0438 \u0442\u043e \u0436\u0435. Bottom Sheet &#8212; \u044d\u0442\u043e \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0440\u0430\u0441\u043f\u043e\u043b\u0430\u0433\u0430\u0435\u0442\u0441\u044f \u0441\u043d\u0438\u0437\u0443 \u0438 \u0430\u0434\u0430\u043f\u0442\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u041f\u0440\u0438\u043c\u0435\u0440\u044b \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f \u0435\u0441\u0442\u044c \u0432 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0445 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f\u0445: <a href=\"https:\/\/apps.apple.com\/us\/app\/apple-maps\/id915056765\">Apple Maps<\/a> (\u043f\u043e\u0438\u0441\u043a), <a href=\"https:\/\/apps.apple.com\/us\/app\/stocks\/id1069512882\">Stocks<\/a> (\u043d\u043e\u0432\u043e\u0441\u0442\u0438), <a href=\"https:\/\/apps.apple.com\/us\/app\/voice-memos\/id1069512134\">Voice Memos<\/a> (\u0437\u0430\u043f\u0438\u0441\u044c \u0433\u043e\u043b\u043e\u0441\u0430) \u0438 \u0442.\u0434.<\/p>\n<figure class=\"full-width\"><img loading=\"lazy\" decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w780q1\/getpro\/habr\/upload_files\/2fc\/183\/91c\/2fc18391c3b552ede620b9fa66d96aee.jpg\" alt=\"Apple Maps, Stocks, Voice Memos\" title=\"Apple Maps, Stocks, Voice Memos\" width=\"4717\" height=\"2623\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/2fc\/183\/91c\/2fc18391c3b552ede620b9fa66d96aee.jpg\" data-blurred=\"true\"\/><figcaption>Apple Maps, Stocks, Voice Memos<\/figcaption><\/figure>\n<h3>\u0421\u0442\u0430\u0440\u0442\u043e\u0432\u044b\u0439 \u043f\u0440\u043e\u0435\u043a\u0442<\/h3>\n<p>\u0412\u043e\u0442 \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u0441\u0442\u0430\u0440\u0442\u043e\u0432\u044b\u0439 \u043f\u0440\u043e\u0435\u043a\u0442 \u043d\u0430 <a href=\"https:\/\/github.com\/joomcode\/BottomSheet\">Github<\/a>. \u0412 \u043f\u0440\u043e\u0435\u043a\u0442\u0435 \u0435\u0441\u0442\u044c \u0434\u0432\u0430 \u0442\u0430\u0440\u0433\u0435\u0442\u0430: <em>BottomSheetDemo<\/em> \u0438 <em>BottomSheet<\/em> \u2014 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430 \u0441 Bottom Sheet.<\/p>\n<details class=\"spoiler\">\n<summary>\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"yaml\">BottomSheetDemo     Sources\/User Interface         Screens             Resize                 ResizeViewController.swift             Root                 RootViewController.swift BottomSheet     Core         BottomSheetModalDismissalHandler.swift         BottomSheetPresentationController.swift         BottomSheetPresentationController+PullBar.swift         BottomSheetTransitioningDelegate.swift     Helpers         ...<\/code><\/pre>\n<\/div>\n<\/details>\n<p><em>RootViewController<\/em> \u2014 \u044d\u0442\u043e \u043f\u0435\u0440\u0432\u044b\u0439 \u044d\u043a\u0440\u0430\u043d \u0432 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0438. \u0412 \u043d\u0451\u043c \u0435\u0441\u0442\u044c \u0432\u0441\u0435\u0433\u043e \u043e\u0434\u043d\u0430 \u043a\u043d\u043e\u043f\u043a\u0430 Show Bottom Sheet. \u041f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e \u043f\u043e\u043a\u0430\u0436\u0435\u0442\u0441\u044f <em>ResizeViewController.<\/em><\/p>\n<pre><code class=\"swift\">@objc private func handleShowBottomSheet() {     let viewController = ResizeViewController(initialHeight: 300)     present(viewController, animated: true, completion: nil) }<\/code><\/pre>\n<p>\u0412 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0442\u043e\u0440\u0435 <em>ResizeViewController<\/em> \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u0435\u0442 \u0432\u044b\u0441\u043e\u0442\u0443 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u0422\u0430\u043a\u0436\u0435 \u0435\u0441\u0442\u044c \u0447\u0435\u0442\u044b\u0440\u0435 \u043a\u043d\u043e\u043f\u043a\u0438, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0438\u0437\u043c\u0435\u043d\u044f\u044e\u0442 \u0432\u044b\u0441\u043e\u0442\u0443 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430: \u043d\u0430 +100 \u0438 -100, \u0432 2 \u0438 0.5 \u0440\u0430\u0437.<\/p>\n<p>\u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/692\/942\/ba8\/692942ba8bcebdd2ad230201777db148.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<h3>\u0422\u0435\u043e\u0440\u0438\u044f. \u041a\u0430\u043a \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c Bottom Sheet?<\/h3>\n<p>\u041d\u0430\u043c \u043d\u0443\u0436\u043d\u0430 \u0441\u0443\u0449\u043d\u043e\u0441\u0442\u044c, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0431\u0443\u0434\u0435\u0442 \u0443\u043f\u0440\u0430\u0432\u043b\u044f\u0442\u044c \u043f\u043e\u043a\u0430\u0437\u043e\u043c. \u041e\u043d\u0430 \u0434\u043e\u0431\u0430\u0432\u0438\u0442 Bottom Sheet \u0432 UI-\u0438\u0435\u0440\u0430\u0440\u0445\u0438\u044e, \u0440\u0430\u0441\u043f\u043e\u043b\u043e\u0436\u0438\u0442 \u0435\u0433\u043e \u043d\u0430 \u044d\u043a\u0440\u0430\u043d\u0435, \u0443\u0447\u0442\u0451\u0442 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430, \u0431\u0443\u0434\u0435\u0442 \u0440\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430 \u0435\u0433\u043e \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f, \u043f\u043e\u0437\u0430\u0431\u043e\u0442\u0438\u0442\u0441\u044f \u043e\u0431 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0438, \u0438 \u043c\u043e\u0436\u043d\u043e \u0431\u0443\u0434\u0435\u0442 \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435.<\/p>\n<p>\u042d\u0442\u043e \u043f\u043e\u0445\u043e\u0436\u0435 \u043d\u0430 \u0437\u043e\u043d\u0443 \u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u043e\u0441\u0442\u0438 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uipresentationcontroller\">UIPresentationController<\/a>. \u0421 \u043c\u043e\u043c\u0435\u043d\u0442\u0430 \u043f\u043e\u044f\u0432\u043b\u0435\u043d\u0438\u044f view controller&#8217;\u0430 \u0438 \u0434\u043e \u043c\u043e\u043c\u0435\u043d\u0442\u0430 \u0441\u043a\u0440\u044b\u0442\u0438\u044f, UIKit \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 presentation controller \u0434\u043b\u044f \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u043e\u043c \u043f\u043e\u043a\u0430\u0437\u0430.<\/p>\n<p>\u0414\u043b\u044f \u0435\u0433\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f \u043d\u0430\u0434\u043e \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontroller\/1621355-modalpresentationstyle\">modalPresentationStyle<\/a> \u0438 \u043f\u0435\u0440\u0435\u0434\u0430\u0442\u044c presentation controller \u0447\u0435\u0440\u0435\u0437 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontroller\/1621421-transitioningdelegate\">transitioningDelegate<\/a>.<\/p>\n<p>\u0412\u043e\u043e\u0440\u0443\u0436\u0438\u0432\u0448\u0438\u0441\u044c \u044d\u0442\u0438\u043c \u0437\u043d\u0430\u043d\u0438\u0435\u043c, \u043d\u0430\u0447\u043d\u0451\u043c \u0434\u0435\u043b\u0430\u0442\u044c Bottom Sheet!<\/p>\n<h3>\u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c presentation controller<\/h3>\n<p>\u0414\u043b\u044f \u043f\u043e\u043a\u0430\u0437\u0430 Bottom Sheet \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043c <em>modalPresentationStyle<\/em> \u0438 <em>transitioningDelegate<\/em>. \u041d\u0435 \u0437\u0430\u0431\u044b\u0432\u0430\u0435\u043c, \u0447\u0442\u043e <em>transitioningDelegate<\/em> \u2014 \u044d\u0442\u043e weak \u0441\u0441\u044b\u043b\u043a\u0430, \u0438 \u043d\u0430\u043c \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f strong \u0441\u0441\u044b\u043b\u043a\u0430, \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u043f\u043e\u0442\u0435\u0440\u044f\u0442\u044c \u043e\u0431\u044a\u0435\u043a\u0442.<\/p>\n<pre><code class=\"swift\">private var bottomSheetTransitioningDelegate: UIViewControllerTransitioningDelegate?  @objc private func handleShowBottomSheet() {     let viewController = ResizeViewController(initialHeight: 300)     \/\/ TODO: bottomSheetTransitioningDelegate = ...     viewController.modalPresentationStyle = .custom     viewController.transitioningDelegate = bottomSheetTransitioningDelegate     present(viewController, animated: true, completion: nil) }<\/code><\/pre>\n<p>\u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c BottomSheetTransitioningDelegate \u2014 \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e <em>transitioningDelegate.<\/em><\/p>\n<pre><code class=\"swift\">public final class BottomSheetTransitioningDelegate: NSObject, UIViewControllerTransitioningDelegate {     private func _presentationController(         forPresented presented: UIViewController,         presenting: UIViewController?,         source: UIViewController     ) -> BottomSheetPresentationController {         BottomSheetPresentationController(             presentedViewController: presented,             presenting: presenting         )     } }<\/code><\/pre>\n<p>\u0418 presentation controller.<\/p>\n<pre><code class=\"swift\">public final class BottomSheetPresentationController: UIPresentationController {}<\/code><\/pre>\n<p>\u041d\u0430\u043a\u043e\u043d\u0435\u0446 \u0432\u0435\u0440\u043d\u0451\u043c\u0441\u044f \u0432 <em>RootViewController<\/em> \u0438 \u0437\u0430\u043a\u0440\u043e\u0435\u043c TODO.<\/p>\n<pre><code class=\"swift\">\/\/ TODO: bottomSheetTransitioningDelegate = ... bottomSheetTransitioningDelegate = BottomSheetTransitioningDelegate()<\/code><\/pre>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0437\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/ed4\/5dc\/fe3\/ed45dcfe303e1c738c75175b22ea1d01.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u041a\u0430\u043a \u0431\u0443\u0434\u0442\u043e \u0441\u0442\u0430\u043b\u043e \u0442\u043e\u043b\u044c\u043a\u043e \u0445\u0443\u0436\u0435. View controller \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u0432\u043e \u0432\u0435\u0441\u044c \u044d\u043a\u0440\u0430\u043d \u0438 \u0441\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u0437\u0430 status bar. \u041c\u044b \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043b\u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 presentation controller, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u043b view controller \u043a\u0440\u0430\u0441\u0438\u0432\u043e, \u043f\u043e\u0437\u0438\u0446\u0438\u043e\u043d\u0438\u0440\u043e\u0432\u0430\u043b \u0435\u0433\u043e \u0441 \u0443\u0447\u0435\u0442\u043e\u043c safeArea. \u0412 \u043d\u0430\u0448\u0435\u043c presentation controller \u043d\u0438\u0447\u0435\u0433\u043e \u043f\u043e\u0434\u043e\u0431\u043d\u043e\u0433\u043e \u043d\u0435\u0442, \u043c\u044b \u043d\u0438\u043a\u0430\u043a \u043d\u0435 \u0443\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u0435 view controller, \u0434\u0430\u0432\u0430\u0439\u0442\u0435 \u0438\u0441\u043f\u0440\u0430\u0432\u0438\u043c\u0441\u044f.<\/p>\n<h3>\u0423\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u043c \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430<\/h3>\n<p>\u0412\u0435\u0440\u043d\u0451\u043c\u0441\u044f \u043a <em>ResizeViewController.<\/em> \u041f\u043e\u043b\u0435 <em>currentHeight<\/em> \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 \u0442\u0435\u043a\u0443\u0449\u0443\u044e \u0432\u044b\u0441\u043e\u0442\u0443. \u0427\u0442\u043e\u0431\u044b \u043d\u0435 \u0441\u043e\u0437\u0434\u0430\u0432\u0430\u0442\u044c \u043b\u0438\u0448\u043d\u0438\u0445 \u043f\u0440\u043e\u0442\u043e\u043a\u043e\u043b\u043e\u0432, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontroller\/1621476-preferredcontentsize\">preferredContentSize<\/a>. \u041e\u043d \u0431\u0443\u0434\u0435\u0442 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u0442\u0435\u043a\u0443\u0449\u0438\u0439 \u0436\u0435\u043b\u0430\u0435\u043c\u044b\u0439 \u0440\u0430\u0437\u043c\u0435\u0440 \u0434\u043b\u044f Bottom Sheet.<\/p>\n<p>\u0412 presentation controller \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043c <em>frameOfPresentedViewInContainerView<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u0435 <em>presentedView<\/em>. \u0412 \u043d\u0430\u0448\u0435\u043c \u0441\u043b\u0443\u0447\u0430\u0435 <em>presentedView<\/em> \u2014 \u044d\u0442\u043e view <em>ResizeViewController<\/em>. <em>containerView<\/em> \u2014 \u044d\u0442\u043e view, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u0442 <em>presentedView<\/em> \u0438 \u043a\u0443\u0434\u0430 \u043c\u043e\u0436\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0442\u0435\u043d\u044c.<\/p>\n<pre><code class=\"swift\">public override var frameOfPresentedViewInContainerView: CGRect {     targetFrameForPresentedView() }  private func targetFrameForPresentedView() -> CGRect {     guard let containerView = containerView else {         return .zero     }      let windowInsets = presentedView?.window?.safeAreaInsets ?? .zero      let preferredHeight = presentedViewController.preferredContentSize.height + windowInsets.bottom     let maxHeight = containerView.bounds.height - windowInsets.top     let height = min(preferredHeight, maxHeight)      return .init(         x: 0,         y: (containerView.bounds.height - height).pixelCeiled,         width: containerView.bounds.width,         height: height.pixelCeiled     ) }<\/code><\/pre>\n<p>\u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0443\u043a\u0430\u0436\u0435\u043c <em>shouldPresentInFullscreen<\/em> \u0432 <em>false<\/em>, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e Bottom Sheet \u043f\u043e\u043a\u0440\u044b\u0432\u0430\u0435\u0442 \u043d\u0435 \u0432\u0435\u0441\u044c \u044d\u043a\u0440\u0430\u043d.<\/p>\n<pre><code>public override var shouldPresentInFullscreen: Bool {     false }<\/code><\/pre>\n<p>\u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/1c0\/130\/649\/1c013064911b45053a70d70f217c0230.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0418\u0437\u043d\u0430\u0447\u0430\u043b\u044c\u043d\u044b\u0439 \u0440\u0430\u0437\u043c\u0435\u0440 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u0442\u0441\u044f, \u043d\u043e \u043d\u0435\u0442 \u0440\u0435\u0430\u043a\u0446\u0438\u0438 \u043d\u0430 \u0435\u0433\u043e \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f.<\/p>\n<h3>\u0420\u0435\u0430\u0433\u0438\u0440\u0443\u0435\u043c \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0435 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430<\/h3>\n<p>\u0420\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c <em>UIPresentationController<\/em>. \u041e\u043d \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u0442 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uicontentcontainer\">UIContentContainer<\/a>, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u043d\u0430\u043c \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u0435\u043d <em>preferredContentSizeDidChange(forChildContentContainer:)<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f\u0445 <em>preferredContentSize<\/em> \u0432 \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0445 view controller&#8217;\u0430\u0445.<\/p>\n<pre><code class=\"swift\">public override func preferredContentSizeDidChange(forChildContentContainer container: UIContentContainer) {     updatePresentedViewSize() }  private func updatePresentedViewSize() {     guard let presentedView = presentedView else {         return     }      let oldFrame = presentedView.frame     let targetFrame = targetFrameForPresentedView()     if !oldFrame.isAlmostEqual(to: targetFrame) {         presentedView.frame = targetFrame     } }<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u0442\u0435\u043a\u0443\u0449\u0438\u0439 frame \u0438 \u0442\u043e\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043c\u044b \u0441\u0447\u0438\u0442\u0430\u0435\u043c \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u044b\u043c. \u0415\u0441\u043b\u0438 \u043e\u043d\u0438 \u0440\u0430\u0437\u043d\u044b\u0435, \u0442\u043e \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0435\u043c <em>presentedView.frame.<\/em> \u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/f17\/828\/120\/f1782812012dc6a909d2d3f9f1273959.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0420\u0430\u0437\u043c\u0435\u0440 \u0438\u0437\u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f \u043d\u0435\u0440\u0430\u0432\u043d\u043e\u043c\u0435\u0440\u043d\u043e \u0431\u0435\u0437 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0438. \u041f\u043e\u0447\u0435\u043c\u0443? \u041f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043c\u044b \u043d\u0438\u043a\u0430\u043a \u043d\u0435 \u0443\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c \u044d\u0442\u0443 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044e. \u0414\u043e\u0431\u0430\u0432\u0438\u043c \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044e \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f <em>preferredContentSize<\/em> \u0432 <em>ResizeViewController.<\/em><\/p>\n<pre><code class=\"swift\">UIView.animate(     withDuration: 0.25,     animations: { [self] in         preferredContentSize = CGSize(             width: UIScreen.main.bounds.width,             height: newValue         )     } )<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/45d\/066\/d8f\/45d066d8f1c03475ec38e7b479ef7226.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0420\u0430\u0431\u043e\u0442\u0430\u0435\u0442! \u041d\u043e \u043c\u044b \u043d\u0438\u043a\u0430\u043a \u043d\u0435 \u043c\u043e\u0436\u0435\u043c \u0443\u0439\u0442\u0438 \u0441 Bottom Sheet.<\/p>\n<h3>\u0417\u0430\u043a\u0440\u044b\u0432\u0430\u0435\u043c Bottom Sheet<\/h3>\n<p>\u0414\u043b\u044f \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f \u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0442\u0435\u043d\u044c \u0438 \u043f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e \u0431\u0443\u0434\u0435\u043c \u0441\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet. \u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f, \u0447\u0435\u0440\u0435\u0437 \u043a\u043e\u0442\u043e\u0440\u044b\u0439 presentation controller \u0431\u0443\u0434\u0435\u0442 \u0441\u043e\u043e\u0431\u0449\u0430\u0442\u044c, \u0447\u0442\u043e \u0433\u043e\u0442\u043e\u0432, \u0447\u0442\u043e\u0431\u044b \u0435\u0433\u043e \u0437\u0430\u043a\u0440\u044b\u043b\u0438.<\/p>\n<pre><code class=\"swift\">public protocol BottomSheetModalDismissalHandler {     func performDismissal(animated: Bool) }<\/code><\/pre>\n<p>\u041f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u0435\u0433\u043e \u0432 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0442\u043e\u0440 presentation controller.<\/p>\n<pre><code class=\"swift\">private let dismissalHandler: BottomSheetModalDismissalHandler  public init(     presentedViewController: UIViewController,     presentingViewController: UIViewController?,     dismissalHandler: BottomSheetModalDismissalHandler ) {     self.dismissalHandler = dismissalHandler     super.init(presentedViewController: presentedViewController, presenting: presentingViewController) }<\/code><\/pre>\n<p>\u0414\u043b\u044f \u0443\u0434\u043e\u0431\u0441\u0442\u0432\u0430 \u0441\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u0444\u0430\u0431\u0440\u0438\u043a\u0443 presentation controller&#8217;\u0430.<\/p>\n<pre><code class=\"swift\">public protocol BottomSheetPresentationControllerFactory {     func makeBottomSheetPresentationController(         presentedViewController: UIViewController,         presentingViewController: UIViewController?     ) -> BottomSheetPresentationController }<\/code><\/pre>\n<p>\u041a\u043e\u0442\u043e\u0440\u0430\u044f \u0431\u0443\u0434\u0435\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0432\u043d\u0443\u0442\u0440\u0438 <em>BottomSheetTransitioningDelegate.<\/em> <\/p>\n<pre><code class=\"swift\">private let factory: BottomSheetPresentationControllerFactory  public init(factory: BottomSheetPresentationControllerFactory) {     self.factory = factory }  public func presentationController(     forPresented presented: UIViewController,     presenting: UIViewController?,     source: UIViewController ) -> UIPresentationController? {     factory.makeBottomSheetPresentationController(         presentedViewController: presented,         presentingViewController: presenting     ) }<\/code><\/pre>\n<p>\u0412 <em>RootViewController<\/em> \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c \u0444\u0430\u0431\u0440\u0438\u043a\u0443 \u0438 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f. \u0421\u043a\u0440\u044b\u0432\u0430\u0435\u043c <em>presentedViewController<\/em>, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u044d\u0442\u043e \u0438 \u0435\u0441\u0442\u044c Bottom Sheet.<\/p>\n<pre><code class=\"swift\">extension RootViewController: BottomSheetPresentationControllerFactory {     func makeBottomSheetPresentationController(         presentedViewController: UIViewController,         presentingViewController: UIViewController?     ) -> BottomSheetPresentationController {         .init(             presentedViewController: presentedViewController,             presentingViewController: presentingViewController,             dismissalHandler: self         )     } }  extension RootViewController: BottomSheetModalDismissalHandler {     func performDismissal(animated: Bool) {         presentedViewController?.dismiss(animated: animated, completion: nil)     } }<\/code><\/pre>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u0432 presentation controller&#8217;\u0435 \u0441\u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0438\u0440\u0443\u0435\u043c \u0442\u0435\u043d\u044c \u0441 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u043e\u043c \u0441\u043a\u0440\u044b\u0442\u0438\u044f. \u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u0442\u0435\u043d\u044c \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u0447\u0430\u043b\u043e\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430 \u0438 \u0443\u0431\u0438\u0440\u0430\u0435\u043c \u043f\u043e\u0441\u043b\u0435 \u043e\u043a\u043e\u043d\u0447\u0430\u043d\u0438\u044f.<\/p>\n<p>\u0414\u043b\u044f \u043d\u0430\u0447\u0430\u043b\u0430 \u043d\u0430\u043c \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f \u043e\u0442\u0441\u043b\u0435\u0436\u0438\u0432\u0430\u0442\u044c \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 presentation controller. \u0412\u0432\u0435\u0434\u0451\u043c \u043f\u043e\u043b\u0435 state, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u0431\u0443\u0434\u0435\u0442 \u043e\u0442\u0432\u0435\u0447\u0430\u0442\u044c \u0437\u0430 \u0442\u0435\u043a\u0443\u0449\u0435\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 Bottom Sheet. \u0414\u043b\u044f \u043e\u0442\u0441\u043b\u0435\u0436\u0438\u0432\u0430\u043d\u0438\u044f \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043c \u043c\u0435\u0442\u043e\u0434\u044b \u0436\u0438\u0437\u043d\u0435\u043d\u043d\u043e\u0433\u043e \u0446\u0438\u043a\u043b\u0430 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430.<\/p>\n<details class=\"spoiler\">\n<summary>\u0416\u0438\u0437\u043d\u0435\u043d\u043d\u044b\u0439 \u0446\u0438\u043a\u043b presentation controller<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"swift\">\/\/ MARK: - Nested types  private enum State {     case dismissed     case presenting     case presented     case dismissing }  \/\/ MARK: - Private properties  private var state: State = .dismissed  \/\/ MARK: - UIPresentationController  public override func presentationTransitionWillBegin() {     state = .presenting }  public override func presentationTransitionDidEnd(_ completed: Bool) {     if completed {         state = .presented     } else {         state = .dismissed     } }  public override func dismissalTransitionWillBegin() {     state = .dismissing }  public override func dismissalTransitionDidEnd(_ completed: Bool) {     if completed {         state = .dismissed     } else {         state = .presented     } }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u0414\u0430\u043b\u0435\u0435 \u0432\u043e\u0437\u043d\u0438\u043a\u0430\u0435\u0442 \u0432\u043e\u043f\u0440\u043e\u0441, \u0432 \u043a\u0430\u043a\u043e\u0439 \u043c\u043e\u043c\u0435\u043d\u0442 \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0442\u044c \u0438 \u0443\u0434\u0430\u043b\u044f\u0442\u044c \u0442\u0435\u043d\u044c. \u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0442\u044c \u0442\u0435\u043d\u044c \u0431\u0443\u0434\u0435\u043c \u043f\u0435\u0440\u0435\u0434 \u043f\u043e\u043a\u0430\u0437\u043e\u043c Bottom Sheet, \u0430 \u0443\u0434\u0430\u043b\u044f\u0442\u044c \u2014 \u0441\u0440\u0430\u0437\u0443 \u043f\u043e\u0441\u043b\u0435 \u0441\u043a\u0440\u044b\u0442\u0438\u044f Bottom Sheet.<\/p>\n<pre><code class=\"swift\">public override func presentationTransitionWillBegin() {     state = .presenting      addSubviews() }  public override func dismissalTransitionDidEnd(_ completed: Bool) {     if completed {         removeSubviews()          state = .dismissed     } else {         state = .presented     } }<\/code><\/pre>\n<p>\u041e\u0441\u0442\u0430\u0451\u0442\u0441\u044f \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c <em>addSubviews()<\/em> \u0438 <em>removeSubviews().<\/em><\/p>\n<details class=\"spoiler\">\n<summary>\u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u0438 \u0443\u0434\u0430\u043b\u044f\u0435\u043c \u0442\u0435\u043d\u044c \u2014 addSubviews() \u0438 removeSubviews()<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"swift\">private func addSubviews() {     guard let containerView = containerView else {         assertionFailure()         return     }          setupShadingView(containerView: containerView) }  private func setupShadingView(containerView: UIView) {     let shadingView = UIView()     containerView.addSubview(shadingView)     shadingView.backgroundColor = UIColor.black.withAlphaComponent(0.6)     shadingView.frame = containerView.bounds      let tapGesture = UITapGestureRecognizer()     shadingView.addGestureRecognizer(tapGesture)      tapGesture.addTarget(self, action: #selector(handleShadingViewTapGesture))     self.shadingView = shadingView }  @objc private func handleShadingViewTapGesture() {     dismissIfPossible() }  private func removeSubviews() {     shadingView?.removeFromSuperview()     shadingView = nil }  private func dismissIfPossible() {     let canBeDismissed = state == .presented      if canBeDismissed {         dismissalHandler.performDismissal(animated: true)     } }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/9e1\/f08\/5c5\/9e1f085c5488c410f1eba9aaf446f0b8.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u041e\u0442\u043b\u0438\u0447\u043d\u043e, \u0442\u0435\u043f\u0435\u0440\u044c \u0443 \u043d\u0430\u0441 \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0442\u0435\u043d\u044c, \u0438 \u043f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e Bottom Sheet \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f! \u041d\u043e \u0442\u0435\u043d\u044c \u043f\u043e\u044f\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0438 \u0438\u0441\u0447\u0435\u0437\u0430\u0435\u0442 \u0431\u0435\u0437 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0438.<\/p>\n<h3>\u0410\u043d\u0438\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d<\/h3>\n<p>\u041a\u0430\u043a \u0431\u044b\u0442\u044c? \u0422\u0435\u043d\u044c \u043e\u0442\u043d\u043e\u0441\u0438\u0442\u0441\u044f \u043a \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0443 \u0438 \u0434\u043e\u043b\u0436\u043d\u0430 \u0430\u043d\u0438\u043c\u0438\u0440\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0432\u043c\u0435\u0441\u0442\u0435 \u0441 \u043d\u0438\u043c. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e \u0432\u0441\u0442\u0440\u043e\u0438\u0442\u044c\u0441\u044f \u0432 transitioning delegate.<\/p>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c \u043f\u0440\u043e\u0442\u043e\u043a\u043e\u043b\u00a0<a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontrolleranimatedtransitioning\">UIViewControllerAnimatedTransitioning<\/a>, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u0431\u0443\u0434\u0435\u0442 \u043b\u043e\u0433\u0438\u043a\u0430 \u0434\u043b\u044f \u0430\u043d\u0438\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0433\u043e \u043f\u043e\u0434\u043d\u0438\u043c\u0430\u043d\u0438\u044f \u0438 \u043e\u043f\u0443\u0441\u043a\u0430\u043d\u0438\u044f \u0448\u0442\u043e\u0440\u043a\u0438. \u0420\u043e\u0432\u043d\u043e \u0442\u0430\u043a\u0430\u044f \u0436\u0435, \u043a\u0430\u043a \u0438 \u0443 \u0441\u0438\u0441\u0442\u0435\u043c\u044b, \u043d\u043e \u0434\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u043c fade-\u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044e \u0434\u043b\u044f \u0442\u0435\u043d\u0438.<\/p>\n<details class=\"spoiler\">\n<summary>\u041f\u043e\u0434\u043d\u0438\u043c\u0430\u0435\u043c \u0438 \u043e\u043f\u0443\u0441\u043a\u0430\u0435\u043c \u0448\u0442\u043e\u0440\u043a\u0443 \u0447\u0435\u0440\u0435\u0437 animated transitioning<\/summary>\n<div class=\"spoiler__content\">\n<p>\u0414\u043b\u044f \u043f\u0440\u043e\u0441\u0442\u043e\u0442\u044b \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c \u043f\u0440\u043e\u0442\u043e\u043a\u043e\u043b \u0432\u043d\u0443\u0442\u0440\u0438 presentation controller, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u0443 \u043d\u0435\u0433\u043e \u0443\u0436\u0435 \u0435\u0441\u0442\u044c \u0434\u043e\u0441\u0442\u0443\u043f \u043a \u043d\u0443\u0436\u043d\u044b\u043c UI-\u044d\u043b\u0435\u043c\u0435\u043d\u0442\u0430\u043c.<\/p>\n<pre><code class=\"swift\">extension BottomSheetPresentationController: UIViewControllerAnimatedTransitioning {     public func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval {         0.3     }      public func animateTransition(using transitionContext: UIViewControllerContextTransitioning) {         guard             let sourceViewController = transitionContext.viewController(forKey: .from),             let destinationViewController = transitionContext.viewController(forKey: .to),             let sourceView = sourceViewController.view,             let destinationView = destinationViewController.view         else {             return         }          let isPresenting = destinationViewController.isBeingPresented         let presentedView = isPresenting ? destinationView : sourceView         let containerView = transitionContext.containerView         if isPresenting {             containerView.addSubview(destinationView)              destinationView.frame = containerView.bounds         }          sourceView.layoutIfNeeded()         destinationView.layoutIfNeeded()          let frameInContainer = frameOfPresentedViewInContainerView         let offscreenFrame = CGRect(             origin: CGPoint(                 x: 0,                 y: containerView.bounds.height             ),             size: sourceView.frame.size         )          presentedView.frame = isPresenting ? offscreenFrame : frameInContainer         pullBar?.frame.origin.y = presentedView.frame.minY - Style.pullBarHeight + pixelSize         shadingView?.alpha = isPresenting ? 0 : 1          let animations = {             presentedView.frame = isPresenting ? frameInContainer : offscreenFrame             self.pullBar?.frame.origin.y = presentedView.frame.minY - Style.pullBarHeight + pixelSize             self.shadingView?.alpha = isPresenting ? 1 : 0         }          let completion = { (completed: Bool) in             transitionContext.completeTransition(completed &amp;&amp; !transitionContext.transitionWasCancelled)         }          let options: UIView.AnimationOptions = transitionContext.isInteractive ? .curveLinear : .curveEaseInOut         let transitionDurationValue = transitionDuration(using: transitionContext)         UIView.animate(withDuration: transitionDurationValue, delay: 0, options: options, animations: animations, completion: completion)     } }<\/code><\/pre>\n<p>\u0418 \u043d\u0435 \u0437\u0430\u0431\u0443\u0434\u0435\u043c \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u044e\u0449\u0438\u0435 \u043c\u0435\u0442\u043e\u0434\u044b \u0432 <em>BottomSheetTransitioningDelegate.<\/em><\/p>\n<pre><code class=\"swift\">\/\/ MARK: - UIViewControllerTransitioningDelegate  public func animationController(     forPresented presented: UIViewController,     presenting: UIViewController,     source: UIViewController ) -> UIViewControllerAnimatedTransitioning? {     presentationController }  public func animationController(     forDismissed dismissed: UIViewController ) -> UIViewControllerAnimatedTransitioning? {     presentationController }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u0423\u0431\u0435\u0434\u0438\u043c\u0441\u044f, \u0447\u0442\u043e \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044f \u043f\u043e\u044f\u0432\u0438\u043b\u0430\u0441\u044c.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/250\/46c\/c75\/25046cc75fddf6467800c31785eeccaf.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u041e\u0441\u0442\u0430\u0451\u0442\u0441\u044f \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0437\u0430\u043a\u0440\u0443\u0433\u043b\u0435\u043d\u043d\u044b\u0435 \u043a\u0440\u0430\u044f, \u0438 \u043d\u0430\u0448 Bottom Sheet \u0433\u043e\u0442\u043e\u0432!<\/p>\n<h3>\u0417\u0430\u043a\u0440\u0443\u0433\u043b\u044f\u0435\u043c \u043a\u0440\u0430\u044f<\/h3>\n<p>\u0427\u0435\u0440\u0435\u0437 <em>cornerRadius<\/em> \u0443 <em>presentedViewController<\/em> \u0432 presentation controller. \u041d\u0430\u043c \u043d\u0443\u0436\u043d\u043e \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u044d\u0442\u043e \u043f\u0435\u0440\u0435\u0434 \u043d\u0430\u0447\u0430\u043b\u043e\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430 \u0432 <em>presentationTransitionWillBegin().<\/em><\/p>\n<pre><code class=\"swift\">private func applyStyle() {     guard presentedViewController.isViewLoaded else { return }      presentedViewController.view.clipsToBounds = true     presentedViewController.view.layer.cornerRadius = cornerRadius }<\/code><\/pre>\n<p>\u0421\u043b\u0435\u0434\u0438\u043c \u0437\u0430 \u0443\u0433\u043b\u0430\u043c\u0438.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/ac0\/581\/2df\/ac05812df175705ac599032538841a81.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0417\u0430\u043a\u0440\u0443\u0433\u043b\u0435\u043d\u043d\u044b\u0435! \u0422\u0435\u043f\u0435\u0440\u044c Bottom Sheet \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u0435\u0442 \u0434\u0438\u0437\u0430\u0439\u043d\u0443!<\/p>\n<h3>\u0427\u0442\u043e \u043c\u044b \u0441\u0434\u0435\u043b\u0430\u043b\u0438 \u0432 \u043f\u0435\u0440\u0432\u043e\u0439 \u0447\u0430\u0441\u0442\u0438?<\/h3>\n<ol>\n<li>\n<p>\u041f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043b\u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 transitioning delegate.<\/p>\n<\/li>\n<li>\n<p>\u0421\u043e\u0437\u0434\u0430\u043b\u0438 presentation controller.<\/p>\n<\/li>\n<li>\n<p>\u0414\u043e\u0431\u0430\u0432\u0438\u043b\u0438 \u0442\u0435\u043d\u044c \u0434\u043b\u044f \u0441\u043a\u0440\u044b\u0442\u0438\u044f Bottom Sheet \u0447\u0435\u0440\u0435\u0437 dismiss handler.<\/p>\n<\/li>\n<li>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043b\u0438 \u0430\u043d\u0438\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d \u0447\u0435\u0440\u0435\u0437 transitioning delegate.<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0430\u043b\u0438 \u0431\u0430\u0437\u043e\u0432\u044b\u0439 \u0434\u0438\u0437\u0430\u0439\u043d.<\/p>\n<\/li>\n<\/ol>\n<h2>\u0427\u0430\u0441\u0442\u044c 2. \u0418\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435 Bottom Sheet<\/h2>\n<h3>\u0421\u0442\u0430\u0440\u0442\u043e\u0432\u044b\u0439 \u043f\u0440\u043e\u0435\u043a\u0442<\/h3>\n<p>\u041a\u0430\u043a \u0438 \u0432 \u043f\u0435\u0440\u0432\u043e\u0439 \u0447\u0430\u0441\u0442\u0438 \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u043c \u0441\u043e <a href=\"https:\/\/github.com\/joomcode\/BottomSheet\/tree\/feature\/part-2\">\u0441\u0442\u0430\u0440\u0442\u043e\u0432\u043e\u0433\u043e \u043f\u0440\u043e\u0435\u043a\u0442\u0430<\/a>. \u041a \u043f\u0440\u043e\u0435\u043a\u0442\u0443 \u0434\u043e\u0431\u0430\u0432\u0438\u043b\u0441\u044f pull bar, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u043e\u0434\u0441\u043a\u0430\u0436\u0435\u0442 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e, \u0447\u0442\u043e Bottom Sheet \u043c\u043e\u0436\u043d\u043e \u0441\u043a\u0440\u044b\u0442\u044c \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e \u0432 \u043f\u0443\u0441\u0442\u043e\u0435 \u043f\u0440\u043e\u0441\u0442\u0440\u0430\u043d\u0441\u0442\u0432\u043e, \u043d\u043e \u0435\u0449\u0451 \u0438 \u043f\u043e swipe-\u0436\u0435\u0441\u0442\u0443. \u0422\u0430\u043a \u0436\u0435 \u0432 ResizeViewController \u043f\u043e\u044f\u0432\u0438\u043b\u0441\u044f scrollView \u0432\u043e \u0432\u0435\u0441\u044c \u044d\u043a\u0440\u0430\u043d. \u041e\u043d \u043d\u0430\u043c \u043f\u0440\u0438\u0433\u043e\u0434\u0438\u0442\u0441\u044f \u0434\u043b\u044f \u0441\u043f\u0438\u0441\u043e\u0447\u043d\u044b\u0445 \u044d\u043a\u0440\u0430\u043d\u043e\u0432. \u041e\u0441\u0442\u0430\u043b\u044c\u043d\u043e\u0435 \u0438\u0437 \u043f\u0435\u0440\u0432\u043e\u0439 \u0447\u0430\u0441\u0442\u0438.<\/p>\n<p>\u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/ddb\/317\/ace\/ddb317acefeabfee26d68dcb4e23b885.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<h3>\u0422\u0435\u043e\u0440\u0438\u044f. \u041e\u0441\u043e\u0431\u0435\u043d\u043d\u043e\u0441\u0442\u0438 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0433\u043e \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f<\/h3>\n<p>\u0418\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c <em>UISwipeGestureRecognizer<\/em> \u0434\u043b\u044f \u0440\u0430\u0441\u043f\u043e\u0437\u043d\u0430\u043d\u0438\u044f swipe-\u0436\u0435\u0441\u0442\u0430. \u041f\u043e \u043d\u0435\u043c\u0443 \u0431\u0443\u0434\u0435\u043c \u043d\u0430\u0447\u0438\u043d\u0430\u0442\u044c \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435 Bottom Sheet.<\/p>\n<p>\u041d\u043e \u0447\u0442\u043e, \u0435\u0441\u043b\u0438 \u0443 presented controller \u0443\u0436\u0435 \u0435\u0441\u0442\u044c \u0442\u0430\u043a\u043e\u0439 \u0436\u0435\u0441\u0442? \u0422\u043e\u0433\u0434\u0430 \u044d\u0442\u043e \u043c\u043e\u0436\u0435\u0442 \u043f\u0440\u0438\u0432\u043e\u0434\u0438\u0442\u044c \u043a \u043a\u043e\u043d\u0444\u043b\u0438\u043a\u0442\u0443 \u0436\u0435\u0441\u0442\u043e\u0432, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043d\u0435\u043f\u043e\u043d\u044f\u0442\u043d\u043e, \u043a\u0430\u043a\u043e\u0439 \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u043f\u0435\u0440\u0432\u044b\u043c.<\/p>\n<p>\u041d\u043e \u0442\u0430\u043a \u043b\u0438 \u0447\u0430\u0441\u0442\u043e \u0443 presented controller \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c \u0442\u0430\u043a\u043e\u0439 \u0436\u0435\u0441\u0442? \u041d\u0430 \u0441\u0430\u043c\u043e\u043c \u0434\u0435\u043b\u0435 \u043f\u043e\u0441\u0442\u043e\u044f\u043d\u043d\u043e. \u0412 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u043e\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0438 99 % \u044d\u043a\u0440\u0430\u043d\u043e\u0432 \u0441\u043f\u0438\u0441\u043e\u0447\u043d\u044b\u0435. \u042d\u0442\u043e \u043e\u0437\u043d\u0430\u0447\u0430\u0435\u0442, \u0447\u0442\u043e \u0432 \u043a\u0430\u0436\u0434\u043e\u043c \u0435\u0441\u0442\u044c <em>UIScrollView<\/em> \u0438\u043b\u0438 \u0435\u0433\u043e \u043d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a\u0438: <em>UITableView<\/em> \u0438\u043b\u0438 <em>UICollectionView<\/em>, \u0432 \u043a\u043e\u0442\u043e\u0440\u044b\u0445 \u0435\u0441\u0442\u044c \u0442\u043e\u0442 \u0441\u0430\u043c\u044b\u0439 \u0436\u0435\u0441\u0442. \u041a\u0430\u043a \u0436\u0435 \u0431\u044b\u0442\u044c?<\/p>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0440\u0430\u0437\u0431\u0435\u0440\u0451\u043c \u0434\u0432\u0430 \u0441\u043b\u0443\u0447\u0430\u044f, \u043a\u043e\u0433\u0434\u0430 <em>UIScrollView<\/em> \u043d\u0435\u0442 \u0438 \u043e\u043d \u0435\u0441\u0442\u044c.<\/p>\n<ol>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u043d\u0435\u0442, \u0442\u043e \u0432\u0441\u0451 \u043f\u0440\u043e\u0441\u0442\u043e \u2014 \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c swipe-\u0436\u0435\u0441\u0442.<\/p>\n<\/li>\n<li>\n<p>\u0415\u0441\u043b\u0438 \u0435\u0441\u0442\u044c, \u0442\u043e <strong>\u043a\u043e\u043d\u0442\u0435\u043d\u0442 \u043c\u043e\u0436\u0435\u0442 \u043f\u043e\u043c\u0435\u0449\u0430\u0442\u044c\u0441\u044f<\/strong>:<\/p>\n<ol>\n<li>\n<p><strong>\u041f\u043e\u043b\u043d\u043e\u0441\u0442\u044c\u044e<\/strong>. \u0422\u043e\u0433\u0434\u0430 \u0440\u0430\u0437\u043c\u0435\u0440 Bottom Sheet \u043c\u0435\u043d\u044c\u0448\u0435 \u044d\u043a\u0440\u0430\u043d\u0430, \u0438 \u0431\u0443\u0434\u0435\u043c \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet \u0441\u0440\u0430\u0437\u0443 \u043f\u043e swipe-\u0436\u0435\u0441\u0442\u0443.<\/p>\n<\/li>\n<li>\n<p><strong>\u0427\u0430\u0441\u0442\u0438\u0447\u043d\u043e<\/strong>. \u0422\u043e\u0433\u0434\u0430 swipe \u043c\u043e\u0436\u0435\u0442 \u043e\u0437\u043d\u0430\u0447\u0430\u0442\u044c \u0442\u0430\u043a \u0436\u0435 \u0438 \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u043d\u0433. \u0411\u0443\u0434\u0435\u043c \u0441\u0447\u0438\u0442\u0430\u0442\u044c, \u0447\u0442\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0445\u043e\u0447\u0435\u0442 \u0437\u0430\u043a\u0440\u044b\u0442\u044c Bottom Sheet \u043f\u043e swipe&#8217;\u0443 \u0432\u043d\u0438\u0437 \u0438, \u043a\u043e\u0433\u0434\u0430 \u043a\u043e\u043d\u0442\u0435\u043d\u0442 \u0437\u0430\u043a\u043e\u043d\u0447\u0438\u043b\u0441\u044f \u0441\u0432\u0435\u0440\u0445\u0443 (\u043d\u0443\u043b\u0435\u0432\u043e\u0439 <em>contentOffset<\/em>).<\/p>\n<\/li>\n<\/ol>\n<\/li>\n<\/ol>\n<p>\u0412 \u0441\u043b\u0443\u0447\u0430\u0435 \u043d\u0430\u043b\u0438\u0447\u0438\u044f <em>UIScrollView<\/em> \u043f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f <em>contentOffset,<\/em> \u0438 \u043f\u043e \u043d\u0438\u043c \u043f\u043e\u043d\u0438\u043c\u0430\u0435\u043c, \u0432 \u043a\u0430\u043a\u043e\u0439 \u043c\u043e\u043c\u0435\u043d\u0442 \u043c\u043e\u0436\u043d\u043e \u043d\u0430\u0447\u0438\u043d\u0430\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435.<\/p>\n<p>\u041f\u043e \u043c\u0435\u0445\u0430\u043d\u0438\u043a\u0435 \u0434\u043e\u0433\u043e\u0432\u043e\u0440\u0438\u043b\u0438\u0441\u044c, \u0434\u0430\u0432\u0430\u0439\u0442\u0435 \u0435\u0451 \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c.<\/p>\n<h3>\u0415\u0441\u043b\u0438 UIScrollView \u043d\u0435\u0442<\/h3>\n<p>\u0422\u043e \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c pan gesture \u043a <em>presentedView<\/em>. \u0412 \u043a\u0430\u043a\u043e\u0439 \u043c\u043e\u043c\u0435\u043d\u0442 \u044d\u0442\u043e \u0434\u0435\u043b\u0430\u0442\u044c? \u0416\u0435\u0441\u0442 \u0438\u043d\u0438\u0446\u0438\u0438\u0440\u0443\u0435\u0442 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435. \u0410 Bottom Sheet \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c \u0437\u0430\u043a\u0440\u044b\u0442, \u0442\u043e\u043b\u044c\u043a\u043e \u0435\u0441\u043b\u0438 \u043e\u043d \u043f\u043e\u043b\u043d\u043e\u0441\u0442\u044c\u044e \u043d\u0430 \u044d\u043a\u0440\u0430\u043d\u0435. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u0440\u0430\u0437\u0443\u043c\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0442\u044c \u0436\u0435\u0441\u0442 \u043d\u0430 \u043e\u043a\u043e\u043d\u0447\u0430\u043d\u0438\u0435 \u043f\u043e\u043a\u0430\u0437\u0430 \u0432 <em>presentationTransitionDidEnd(_:).<\/em><\/p>\n<pre><code class=\"swift\">public override func presentationTransitionDidEnd(_ completed: Bool) {     if completed {         setupGesturesForPresentedView()          state = .presented     } else {         state = .dismissed     } }  private func setupGesturesForPresentedView() {     setupPanGesture(for: presentedView) }<\/code><\/pre>\n<p>\u0418 \u043d\u0430\u043f\u0438\u0448\u0435\u043c \u0444\u0443\u043d\u043a\u0446\u0438\u044e, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u0442 pan gesture \u0437\u0430\u0434\u0430\u043d\u043d\u043e\u0439 view.<\/p>\n<pre><code class=\"swift\">private func setupPanGesture(for view: UIView?) {     guard let view = view else { return }      let panRecognizer = UIPanGestureRecognizer(target: self, action: #selector(handlePanGesture(_:)))     view.addGestureRecognizer(panRecognizer) }  @objc private func handlePanGesture(_ panGesture: UIPanGestureRecognizer) {     switch panGesture.state {     case .began:         processPanGestureBegan(panGesture)     case .changed:         processPanGestureChanged(panGesture)     case .ended:         processPanGestureEnded(panGesture)     case .cancelled:         processPanGestureCancelled(panGesture)     default:         break     } }<\/code><\/pre>\n<p>\u0420\u0430\u0437\u0431\u0435\u0440\u0451\u043c \u043a\u0430\u0436\u0434\u043e\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u0436\u0435\u0441\u0442\u0430.<\/p>\n<p><strong>began<\/strong> \u2014 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0442\u043e\u043b\u044c\u043a\u043e-\u0442\u043e\u043b\u044c\u043a\u043e \u043d\u0430\u0447\u0430\u043b \u0434\u0432\u0438\u0436\u0435\u043d\u0438\u0435 \u043f\u0430\u043b\u044c\u0446\u0430 \u0438 \u0436\u0435\u0441\u0442 \u0431\u044b\u043b \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d, \u043a\u0430\u043a pan gesture. \u0418\u043d\u0438\u0446\u0438\u0438\u0440\u0443\u0435\u043c \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435 Bottom Sheet.<\/p>\n<p><strong>changed<\/strong> \u2014 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043d\u0435\u043f\u0440\u0435\u0440\u044b\u0432\u043d\u043e \u0432\u0435\u0434\u0451\u0442 \u043f\u0430\u043b\u044c\u0446\u0435\u043c \u043f\u043e \u044d\u043a\u0440\u0430\u043d\u0443. \u0421\u043a\u0440\u044b\u0432\u0430\u0435\u043c Bottom Sheet \u043f\u0440\u043e\u043f\u043e\u0440\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u044e, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043f\u0440\u043e\u0448\u0451\u043b \u043f\u0430\u043b\u0435\u0446 \u043f\u043e \u044d\u043a\u0440\u0430\u043d\u0443.<\/p>\n<p><strong>ended<\/strong> \u2014 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043e\u0442\u043f\u0443\u0441\u0442\u0438\u043b \u043f\u0430\u043b\u0435\u0446 \u0441 \u044d\u043a\u0440\u0430\u043d\u0430. \u041f\u0440\u0438\u043d\u0438\u043c\u0430\u0435\u043c \u0440\u0435\u0448\u0435\u043d\u0438\u0435, \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet \u0438\u043b\u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0442\u044c \u0435\u0433\u043e \u0432 \u0438\u0441\u0445\u043e\u0434\u043d\u043e\u0435 \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<p><strong>cancelled<\/strong> \u2014 \u0436\u0435\u0441\u0442 \u0431\u044b\u043b \u043e\u0442\u043c\u0435\u043d\u0451\u043d. \u0412\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c Bottom Sheet \u0432 \u0438\u0441\u0445\u043e\u0434\u043d\u043e\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435.<\/p>\n<p>\u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0431\u0443\u0434\u0435\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c <em>UIPercentDrivenInteractiveTransition<\/em> \u0434\u043b\u044f \u043f\u0435\u0440\u0435\u0434\u0430\u0447\u0438 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430 transitioning delegate&#8217;\u0443.<\/p>\n<pre><code class=\"swift\">\/\/ BottomSheetPresentationController.swift  private var interactionController: UIPercentDrivenInteractiveTransition?<\/code><\/pre>\n<p>\u041d\u0430\u0447\u043d\u0451\u043c \u0441 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f <strong>began<\/strong>. \u042d\u0442\u043e \u043f\u043e\u0434\u0445\u043e\u0434\u044f\u0449\u0438\u0439 \u043c\u043e\u043c\u0435\u043d\u0442 \u0434\u043b\u044f \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0433\u043e \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u044d\u0442\u043e \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u0432\u0441\u0435\u0433\u043e \u043e\u0434\u0438\u043d \u0440\u0430\u0437. \u0422\u0430\u043a \u0436\u0435 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c dismiss \u0443 <em>presentingViewController<\/em> \u0434\u043b\u044f \u0443\u0432\u0435\u0434\u043e\u043c\u043b\u0435\u043d\u0438\u044f UIKit \u043e \u043d\u0430\u043c\u0435\u0440\u0435\u043d\u0438\u0438 \u0437\u0430\u043a\u0440\u044b\u0442\u044c Bottom Sheet.<\/p>\n<pre><code class=\"swift\">private func processPanGestureBegan(_ panGesture: UIPanGestureRecognizer) {     startInteractiveTransition() }  private func startInteractiveTransition() {     interactionController = UIPercentDrivenInteractiveTransition()      presentingViewController.dismiss(animated: true) { [weak self] in         guard let self = self else { return }          if self.presentingViewController.presentedViewController !== self.presentedViewController {             self.dismissalHandler.performDismissal(animated: true)         }     } }<\/code><\/pre>\n<p>\u0414\u0430\u043b\u0435\u0435 \u2014 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 <strong>changed<\/strong>. \u0418\u0437\u043c\u0435\u043d\u044f\u0435\u043c \u043f\u043e\u0437\u0438\u0446\u0438\u044e <em>presentedView<\/em> \u043f\u0440\u043e\u043f\u043e\u0440\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e \u0434\u0432\u0438\u0436\u0435\u043d\u0438\u044e \u043f\u0430\u043b\u044c\u0446\u0430 \u043f\u043e \u044d\u043a\u0440\u0430\u043d\u0443. \u0420\u0430\u0441\u0441\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u043c \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u043e\u0442 \u043d\u0430\u0447\u0430\u043b\u044c\u043d\u043e\u0439 \u0442\u043e\u0447\u043a\u0438, \u0433\u0434\u0435 \u0436\u0435\u0441\u0442 \u043d\u0430\u0447\u0430\u043b\u0441\u044f, \u0434\u043e \u0442\u0435\u043a\u0443\u0449\u0435\u0439. \u0414\u0430\u043b\u0435\u0435 \u0432\u044b\u0447\u0438\u0441\u043b\u044f\u0435\u043c \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430 \u043e\u0442\u043d\u043e\u0441\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0432\u044b\u0441\u043e\u0442\u044b content view controller, \u0442.\u0435. <em>presentedView.<\/em><\/p>\n<pre><code class=\"swift\">private func processPanGestureChanged(_ panGesture: UIPanGestureRecognizer) {     let translation = panGesture.translation(in: nil)     updateInteractionControllerProgress(verticalTranslation: translation.y) }  private func updateInteractionControllerProgress(verticalTranslation: CGFloat) {     guard let presentedView = presentedView else { return }      let progress = verticalTranslation \/ presentedView.bounds.height     interactionController?.update(progress) }<\/code><\/pre>\n<p>\u041a\u043e\u0433\u0434\u0430 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043e\u0442\u043f\u0443\u0441\u0442\u0438\u043b \u043f\u0430\u043b\u0435\u0446, \u0442\u043e \u0436\u0435\u0441\u0442 \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0438\u0442 \u0432 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 <strong>ended<\/strong>. \u041d\u0443\u0436\u043d\u043e \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c \u043d\u0430\u043c\u0435\u0440\u0435\u043d\u0438\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f: \u0441\u043a\u0440\u043e\u043b\u043b \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0438\u043b\u0438 \u0436\u0435\u043b\u0430\u043d\u0438\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u044c Bottom Sheet. \u0415\u0441\u043b\u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0440\u0435\u0437\u043a\u043e \u043e\u043f\u0443\u0441\u0442\u0438\u043b \u043f\u0430\u043b\u0435\u0446 \u0432\u043d\u0438\u0437, \u043f\u0440\u043e\u0439\u0434\u044f \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u0435 \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435, \u0442\u043e \u0441\u043a\u043e\u0440\u0435\u0435 \u0432\u0441\u0435\u0433\u043e \u043e\u043d \u0445\u043e\u0442\u0435\u043b \u0437\u0430\u043a\u0440\u044b\u0442\u044c Bottom Sheet. \u0412\u043e\u0437\u043c\u043e\u0436\u043d\u0430 \u0438 \u0434\u0440\u0443\u0433\u0430\u044f \u0441\u0438\u0442\u0443\u0430\u0446\u0438\u044f, \u0435\u0441\u043b\u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043f\u0440\u043e\u0448\u0435\u043b \u0431\u043e\u043b\u044c\u0448\u043e\u0435 \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u043f\u043e \u044d\u043a\u0440\u0430\u043d\u0443 \u0438 \u0432 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0439 \u043c\u043e\u043c\u0435\u043d\u0442 \u0441 \u0443\u0441\u043a\u043e\u0440\u0435\u043d\u0438\u0435\u043c \u043e\u0442\u043f\u0443\u0441\u0442\u0438\u043b \u043f\u0430\u043b\u0435\u0446 \u0441 \u044d\u043a\u0440\u0430\u043d\u0430 \u0432\u0432\u0435\u0440\u0445. \u0412 \u0442\u0430\u043a\u043e\u0439 \u0441\u0438\u0442\u0443\u0430\u0446\u0438\u0438 Bottom Sheet \u0434\u043e\u043b\u0436\u0435\u043d \u0432\u0435\u0440\u043d\u0443\u0442\u044c\u0441\u044f \u0432 \u0438\u0441\u0445\u043e\u0434\u043d\u0443\u044e \u043f\u043e\u0437\u0438\u0446\u0438\u044e. \u042d\u0442\u043e \u043d\u0430\u0432\u043e\u0434\u0438\u0442 \u043d\u0430 \u043c\u044b\u0441\u043b\u0438 \u043e \u0440\u0430\u0441\u0447\u0451\u0442\u0435 \u043d\u0435\u043a\u043e\u0433\u043e \u0438\u043c\u043f\u0443\u043b\u044c\u0441\u0430 \u0434\u0432\u0438\u0436\u0435\u043d\u0438\u044f, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u0442 \u0443\u0441\u043a\u043e\u0440\u0435\u043d\u0438\u0435 \u0438 \u043d\u0430\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435.<\/p>\n<p>\u041d\u0435\u043c\u043d\u043e\u0433\u043e \u0444\u0438\u0437\u0438\u043a\u0438. \u041f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u0438\u043c, \u0447\u0442\u043e \u0435\u0441\u0442\u044c \u043d\u0435\u043a\u043e\u0435 \u0442\u0435\u043b\u043e, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u0434\u0432\u0438\u0433\u0430\u0435\u0442\u0441\u044f \u0441 \u043f\u043e\u0441\u0442\u043e\u044f\u043d\u043d\u043e\u0439 \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u044c\u044e <img decoding=\"async\" class=\"formula inline\" source=\"v_0\" alt=\"v_0\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/6cc\/a71\/50c\/6cca7150c95aad84cbe5b38eebc6eca3.svg\"\/>. \u0414\u0430\u043b\u044c\u0448\u0435 \u043d\u0430 \u043d\u0435\u0433\u043e \u043f\u043e\u0434\u0435\u0439\u0441\u0442\u0432\u043e\u0432\u0430\u043b\u043e \u0437\u0430\u043c\u0435\u0434\u043b\u0435\u043d\u0438\u0435 <img decoding=\"async\" class=\"formula inline\" source=\"a\" alt=\"a\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/660\/a10\/4eb\/660a104eb7c8dafba826dfe93d2138a9.svg\"\/> \u043d\u0430 \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0438 <img decoding=\"async\" class=\"formula inline\" source=\"x_0\" alt=\"x_0\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/9a8\/58c\/73f\/9a858c73fdcdf74d730edf5da30566b7.svg\"\/>. \u0412\u043e\u043f\u0440\u043e\u0441 \u2014 \u0433\u0434\u0435 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u0441\u044f \u0442\u0435\u043b\u043e?<\/p>\n<details class=\"spoiler\">\n<summary>\u0412\u044b\u0432\u043e\u0434 \u0444\u043e\u0440\u043c\u0443\u043b\u044b \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u044f \u0434\u043b\u044f \u0442\u0435\u043b\u0430 \u0441 \u0437\u0430\u043c\u0435\u0434\u043b\u0435\u043d\u0438\u0435\u043c<\/summary>\n<div class=\"spoiler__content\">\n<p>\u041d\u0430\u043f\u0438\u0448\u0435\u043c \u0444\u043e\u0440\u043c\u0443\u043b\u0443 \u0434\u043b\u044f \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u0438 <img decoding=\"async\" class=\"formula inline\" source=\"v(t) = v_0 + a * t\" alt=\"v(t) = v_0 + a * t\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/d17\/0f8\/b14\/d170f8b1479d54e64ceab3e75cdd23f4.svg\"\/><\/p>\n<p>\u0421 \u043e\u0442\u0440\u0438\u0446\u0430\u0442\u0435\u043b\u044c\u043d\u044b\u043c \u0443\u0441\u043a\u043e\u0440\u0435\u043d\u0438\u0435\u043c \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u044c \u0441\u0442\u0430\u043d\u0435\u0442 \u043d\u0443\u043b\u0435\u0432\u043e\u0439, \u043e\u0431\u043e\u0437\u043d\u0430\u0447\u0438\u043c \u044d\u0442\u043e\u0442 \u043c\u043e\u043c\u0435\u043d\u0442 \u0432\u0440\u0435\u043c\u0435\u043d\u0438 <img decoding=\"async\" class=\"formula inline\" source=\"t_1\" alt=\"t_1\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/a2b\/cf5\/bae\/a2bcf5bae509ba198e2c38064d8d2afc.svg\"\/>. \u041f\u043e\u0434\u0441\u0442\u0430\u0432\u0438\u043c <img decoding=\"async\" class=\"formula inline\" source=\"t_1\" alt=\"t_1\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/f97\/077\/26c\/f9707726cbc7bab66073022fd2de3fd6.svg\"\/>\u0432 \u0444\u043e\u0440\u043c\u0443\u043b\u0443 \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u0438:<\/p>\n<p><img loading=\"lazy\" decoding=\"async\" class=\"formula\" source=\"v(t_1) = v_0 + a * t_1 = 0 \\\\  t_1 = -v_0 \/ a\" alt=\"v(t_1) = v_0 + a * t_1 = 0 \\\\  t_1 = -v_0 \/ a\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/457\/13c\/354\/45713c354a93b2371fa78248309b4f14.svg\" width=\"583\" height=\"50\"\/><\/p>\n<p>\u0414\u0430\u043b\u0435\u0435 \u043f\u043e\u0434\u0441\u0442\u0430\u0432\u0438\u043c <img decoding=\"async\" class=\"formula inline\" source=\"t_1\" alt=\"t_1\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/fd7\/721\/b4d\/fd7721b4dc73cf3a63f6e757506e0bcf.svg\"\/>\u0432 \u0444\u043e\u0440\u043c\u0443\u043b\u0443 \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u044f <img decoding=\"async\" class=\"formula inline\" source=\"x(t) = x_0 + v_0 * t + a * t * t \/ 2\" alt=\"x(t) = x_0 + v_0 * t + a * t * t \/ 2\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/00e\/c77\/106\/00ec77106c1717bfc12c31fa889a6396.svg\"\/><\/p>\n<p><img loading=\"lazy\" decoding=\"async\" class=\"formula\" source=\"x(t_1) = x_0 + v_0 * t_1 + a * t_1^2 \/ 2 \\\\ = x_0 - v_0^2 \/ a + a * v_0^2 \/ (2 * a^2) \\\\ = x_0 - v_0^2 \/ a + v_0^2 \/ (2 * a) \\\\ = x_0 - 0.5 * v_0^2 \/ a\" alt=\"x(t_1) = x_0 + v_0 * t_1 + a * t_1^2 \/ 2 \\\\ = x_0 - v_0^2 \/ a + a * v_0^2 \/ (2 * a^2) \\\\ = x_0 - v_0^2 \/ a + v_0^2 \/ (2 * a) \\\\ = x_0 - 0.5 * v_0^2 \/ a\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/31f\/adb\/f24\/31fadbf24dbdd8f19efae0e2bb44472b.svg\" width=\"583\" height=\"118\"\/><\/p>\n<p>\u041e\u043a\u043e\u043d\u0447\u0430\u0442\u0435\u043b\u044c\u043d\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043c \u0444\u043e\u0440\u043c\u0443\u043b\u0443 \u0434\u043b\u044f \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u044f <img decoding=\"async\" class=\"formula inline\" source=\"x_0 - 0.5 * v_0^2 \/ a\" alt=\"x_0 - 0.5 * v_0^2 \/ a\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/b71\/176\/218\/b711762185cf5f67e68eb5ca99b90c65.svg\"\/>, \u0433\u0434\u0435 <img decoding=\"async\" class=\"formula inline\" source=\"x_0\" alt=\"x_0\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/c9b\/754\/079\/c9b7540798dd3a5a4da265f59aabde89.svg\"\/>\u2014 \u043d\u0430\u0447\u0430\u043b\u044c\u043d\u043e\u0435 \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0442\u0435\u043b\u0430, <img decoding=\"async\" class=\"formula inline\" source=\"v_0\" alt=\"v_0\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/80c\/96a\/e27\/80c96ae270310b6abf3be52b9f11fac7.svg\"\/>\u2014 \u043d\u0430\u0447\u0430\u043b\u044c\u043d\u0430\u044f \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u044c \u0438 <img decoding=\"async\" class=\"formula inline\" source=\"a\" alt=\"a\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/960\/6b6\/d92\/9606b6d92a23805914afc4533da3844c.svg\"\/>\u2014 \u0437\u0430\u043c\u0435\u0434\u043b\u0435\u043d\u0438\u0435.<\/p>\n<\/div>\n<\/details>\n<p>\u041f\u0440\u0438\u043c\u0435\u043c Bottom Sheet \u0437\u0430 \u0442\u0435\u043b\u043e \u0438\u0437 \u0437\u0430\u0434\u0430\u0447\u0438 \u0432\u044b\u0448\u0435, \u043a\u043e\u0433\u0434\u0430 \u0436\u0435\u0441\u0442 \u0437\u0430\u043a\u043e\u043d\u0447\u0438\u043b\u0441\u044f. \u0427\u0435\u0440\u0435\u0437 pan gesture \u0443\u0437\u043d\u0430\u0435\u043c \u0442\u0435\u043a\u0443\u0449\u0443\u044e \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u044c. \u041f\u0440\u043e\u0439\u0434\u0435\u043d\u043d\u043e\u0435 \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u0443 \u043d\u0430\u0441 \u0435\u0441\u0442\u044c. \u0417\u0430\u043c\u0435\u0434\u043b\u0435\u043d\u0438\u0435 \u043f\u043e\u0434\u0431\u0438\u0440\u0430\u0435\u043c \u0438 \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u0435\u043c \u0437\u0430 \u043a\u043e\u043d\u0441\u0442\u0430\u043d\u0442\u0443 <img decoding=\"async\" class=\"formula inline\" source=\"800\" alt=\"800\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/cd9\/be0\/2f0\/cd9be02f0e59fb04deddd4026e48e7ce.svg\"\/>. \u0417\u043d\u0430\u044f \u0444\u043e\u0440\u043c\u0443\u043b\u0443 \u0434\u043b\u044f \u0440\u0430\u0441\u0441\u0442\u043e\u044f\u043d\u0438\u044f <img decoding=\"async\" class=\"formula inline\" source=\"x_0 - 0.5 * v_0^2\/ a\" alt=\"x_0 - 0.5 * v_0^2\/ a\" src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/7d4\/7a9\/a7c\/7d47a9a7c9b3f76cb30e2295397f167c.svg\"\/>, \u0440\u0430\u0441\u0441\u0447\u0438\u0442\u0430\u0435\u043c, \u0433\u0434\u0435 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u0441\u044f Bottom Sheet \u043f\u043e\u0434 \u0437\u0430\u043c\u0435\u0434\u043b\u0435\u043d\u0438\u0435\u043c. \u0415\u0441\u043b\u0438 \u0442\u043e\u0447\u043a\u0430 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u0431\u043b\u0438\u0436\u0435 \u043a \u043d\u0430\u0447\u0430\u043b\u044c\u043d\u043e\u043c\u0443 \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u044e, \u0442\u043e \u043e\u0442\u043c\u0435\u043d\u044f\u0435\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d, \u0438\u043d\u0430\u0447\u0435 \u0437\u0430\u0432\u0435\u0440\u0448\u0430\u0435\u043c \u0435\u0433\u043e.<\/p>\n<pre><code class=\"swift\">private func processPanGestureEnded(_ panGesture: UIPanGestureRecognizer) {     let velocity = panGesture.velocity(in: presentedView)     let translation = panGesture.translation(in: presentedView)     endInteractiveTransition(verticalVelocity: velocity.y, verticalTranslation: translation.y) }  private func endInteractiveTransition(verticalVelocity: CGFloat, verticalTranslation: CGFloat) {     guard let presentedView = presentedView else { return }      let deceleration = 800.0 * (verticalVelocity > 0 ? -1.0 : 1.0)     let finalProgress = (verticalTranslation - 0.5 * verticalVelocity * verticalVelocity \/ CGFloat(deceleration))         \/ presentedView.bounds.height     let isThresholdPassed = finalProgress &lt; 0.5      endInteractiveTransition(isCancelled: isThresholdPassed) }  private func endInteractiveTransition(isCancelled: Bool) {     if isCancelled {         interactionController?.cancel()     } else {         interactionController?.finish()     }     interactionController = nil }<\/code><\/pre>\n<p>\u0418 \u0435\u0441\u043b\u0438 \u0436\u0435\u0441\u0442 \u043e\u0442\u043c\u0435\u043d\u0438\u043b\u0441\u044f <strong>cancelled<\/strong>, \u0442\u043e \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c\u0441\u044f \u0432 \u0438\u0441\u0445\u043e\u0434\u043d\u043e\u0435 \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<pre><code class=\"swift\">private func processPanGestureCancelled(_ panGesture: UIPanGestureRecognizer) {     endInteractiveTransition(isCancelled: true) }<\/code><\/pre>\n<p>\u0422\u0430\u043a\u0436\u0435 \u043d\u0435 \u0437\u0430\u0431\u0443\u0434\u0435\u043c \u0432\u0435\u0440\u043d\u0443\u0442\u044c <em>interactiveTransitioning<\/em> \u0432 transitioning delegate.<\/p>\n<details class=\"spoiler\">\n<summary>\u0412\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c interactiveTransitioning \u0432 transitioning delegate<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"swift\">\/\/ BottomSheetPresentationController.swift  var interactiveTransitioning: UIViewControllerInteractiveTransitioning? {     interactionController }  \/\/ BottomSheetTransitioningDelegate.swift  public func interactionControllerForPresentation(using animator: UIViewControllerAnimatedTransitioning) -> UIViewControllerInteractiveTransitioning? {     presentationController?.interactiveTransitioning }      public func interactionControllerForDismissal(using animator: UIViewControllerAnimatedTransitioning) -> UIViewControllerInteractiveTransitioning? {     presentationController?.interactiveTransitioning }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0438 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/be9\/28e\/702\/be928e702489084406ad8c02dec121cc.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0412\u0436\u0443\u0445-\u0432\u0436\u0443\u0445, \u0438 \u043c\u044b \u043d\u0430\u0443\u0447\u0438\u043b\u0438 Bottom Sheet \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0442\u044c\u0441\u044f \u043f\u043e \u0441\u0432\u0430\u0439\u043f\u0443 \u0432\u043d\u0438\u0437! \u041e\u0434\u043d\u0430\u043a\u043e, \u0435\u0441\u043b\u0438 \u043a\u043e\u043d\u0442\u0435\u043d\u0442 \u0431\u043e\u043b\u044c\u0448\u0435 \u044d\u043a\u0440\u0430\u043d\u0430, \u0442\u043e scrollView \u043f\u0435\u0440\u0435\u0445\u0432\u0430\u0442\u044b\u0432\u0430\u0435\u0442 \u0436\u0435\u0441\u0442\u044b.<\/p>\n<h3>\u0420\u0430\u0437\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u043f\u0438\u0441\u043e\u0447\u043d\u044b\u0435 \u044d\u043a\u0440\u0430\u043d\u044b<\/h3>\n<p>\u041d\u0435\u0441\u043c\u043e\u0442\u0440\u044f \u043d\u0430 \u0442\u043e, \u0447\u0442\u043e \u0434\u043e \u044d\u0442\u043e\u0433\u043e <em>ResizeViewController<\/em> \u0438 \u0442\u0430\u043a \u0431\u044b\u043b \u0441\u043f\u0438\u0441\u043e\u0447\u043d\u044b\u043c, \u044d\u0442\u043e \u043d\u0435 \u043f\u043e\u043c\u0435\u0448\u0430\u043b\u043e \u043d\u0430\u043c \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c pan gesture. \u0412\u0441\u0451 \u043f\u043e\u0442\u043e\u043c\u0443, \u0447\u0442\u043e \u0443 scrollView \u043d\u0435\u0442 \u0441\u043a\u0440\u043e\u043b\u043b\u0430, \u043a\u043e\u0433\u0434\u0430 <em>contentSize<\/em> \u0440\u0430\u0432\u0435\u043d \u0435\u0433\u043e \u0440\u0430\u0437\u043c\u0435\u0440\u0443.<\/p>\n<p>\u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u0441\u043b\u0443\u0447\u0430\u0439, \u043a\u043e\u0433\u0434\u0430 <em>contentSize<\/em> \u0431\u043e\u043b\u044c\u0448\u0435 \u0440\u0430\u0437\u043c\u0435\u0440\u0430 Bottom Sheet, \u0438 \u0441\u043a\u0440\u043e\u043b\u043b \u0435\u0441\u0442\u044c. \u041f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c\u0441\u044f \u043d\u0430 <em>contentOffset<\/em>. \u0415\u0441\u043b\u0438 <em>contentOffset<\/em> \u043d\u0443\u043b\u0435\u0432\u043e\u0439, \u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442 \u0432\u043d\u0438\u0437, \u0442\u043e \u0438\u043d\u0438\u0446\u0438\u0438\u0440\u0443\u0435\u043c \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435. \u041a\u043e\u0433\u0434\u0430 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043e\u0442\u043f\u0443\u0441\u043a\u0430\u0435\u0442 \u043f\u0430\u043b\u0435\u0446 \u0441 \u044d\u043a\u0440\u0430\u043d\u0430, \u0442\u043e \u0440\u0435\u0448\u0430\u0435\u043c \u043e\u0442\u043c\u0435\u043d\u0438\u0442\u044c \u0438\u043b\u0438 \u0437\u0430\u043a\u043e\u043d\u0447\u0438\u0442\u044c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d, \u043a\u0430\u043a \u0438 \u0440\u0430\u043d\u044c\u0448\u0435. \u0415\u0441\u043b\u0438 <em>contentOffset<\/em> \u0438\u0437\u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f \u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043d\u0435 \u043a\u0430\u0441\u0430\u0435\u0442\u0441\u044f \u044d\u043a\u0440\u0430\u043d\u0430, \u0442\u043e \u043d\u0438\u0447\u0435\u0433\u043e \u043d\u0435 \u0434\u0435\u043b\u0430\u0435\u043c. \u042d\u0442\u043e \u0437\u043d\u0430\u0447\u0438\u0442, \u0447\u0442\u043e \u0441\u043a\u0440\u043e\u043b\u043b \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u043f\u043e \u0438\u043d\u0435\u0440\u0446\u0438\u0438.<\/p>\n<p>\u0414\u043b\u044f \u043d\u0430\u0447\u0430\u043b\u0430 \u043d\u0430\u043c \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f \u043f\u0440\u0438\u0437\u043d\u0430\u043a, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u043e\u0434\u0441\u043a\u0430\u0436\u0435\u0442, \u0447\u0442\u043e \u0443 view controller \u0435\u0441\u0442\u044c scrollView. \u0412\u0432\u0435\u0434\u0451\u043c \u0434\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u043f\u0440\u043e\u0442\u043e\u043a\u043e\u043b.<\/p>\n<pre><code class=\"swift\">public protocol ScrollableBottomSheetPresentedController: AnyObject {     var scrollView: UIScrollView? { get } }<\/code><\/pre>\n<p>\u0414\u043b\u044f \u043e\u0442\u0441\u043b\u0435\u0436\u0438\u0432\u0430\u043d\u0438\u044f \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0439 <em>contentOffset<\/em> \u043f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c\u0441\u044f \u043d\u0430 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiscrollviewdelegate\">UIScrollViewDelegate<\/a>. \u041d\u043e \u0447\u0442\u043e, \u0435\u0441\u043b\u0438 \u043a\u0442\u043e-\u0442\u043e \u0443\u0436\u0435 \u043f\u043e\u0434\u043f\u0438\u0441\u0430\u043b\u0441\u044f \u043d\u0430 delegate <em>UIScrollView<\/em>? \u0422\u043e\u0433\u0434\u0430 \u043c\u044b \u0437\u0430\u0442\u0440\u0451\u043c \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0439 <em>delegate.<\/em><\/p>\n<p>\u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u0431\u0443\u0434\u0435\u043c \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c <a href=\"https:\/\/github.com\/joomcode\/BottomSheet\/blob\/main\/Sources\/BottomSheet\/Helpers\/Utils\/UIScrollView%2BMulticastDelegate.swift\">\u043f\u0440\u043e\u043a\u0441\u0438 \u043d\u0430 UIScrollViewDelegate<\/a>. \u0418\u0434\u0435\u0439\u043d\u043e <em>MulticastDelegate<\/em> \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u0442 <em>UIScrollViewDelegate<\/em> \u0438 \u043f\u0440\u043e\u043a\u0441\u0438\u0440\u0443\u0435\u0442 \u043c\u0435\u0442\u043e\u0434\u044b delegate \u0437\u0430\u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043e\u0432\u0430\u043d\u043d\u044b\u043c \u0441\u0442\u043e\u0440\u043e\u043d\u0430\u043c. \u041f\u0440\u0438 \u044d\u0442\u043e\u043c \u0437\u0430\u0431\u043e\u0442\u0438\u0442\u0441\u044f, \u0447\u0442\u043e\u0431\u044b \u043f\u043e\u043b\u0435 <em>delegate<\/em> \u043d\u0435 \u0437\u0430\u0442\u0438\u0440\u0430\u043b\u043e\u0441\u044c. \u0412 Swift \u043d\u0430\u043c \u043f\u0440\u0438\u0445\u043e\u0434\u0438\u0442\u0441\u044f \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0442\u044c \u043a\u0430\u0436\u0434\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 delegate. \u0412 Objective-C \u043c\u043e\u0436\u043d\u043e \u0434\u043e\u0431\u0438\u0442\u044c\u0441\u044f \u0430\u043d\u0430\u043b\u043e\u0433\u0438\u0447\u043d\u043e\u0433\u043e \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u0430 \u0431\u0435\u0437 \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438 \u0432\u0441\u0435\u0445 \u043c\u0435\u0442\u043e\u0434\u043e\u0432 delegate \u0447\u0435\u0440\u0435\u0437 runtime.<\/p>\n<p>\u041f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c\u0441\u044f \u043d\u0430 <em>delegate<\/em> \u043f\u043e\u0441\u043b\u0435 \u043e\u043a\u043e\u043d\u0447\u0430\u043d\u0438\u044f \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430 \u0432 <em>presentationTransitionDidEnd(_:)<\/em>, \u043a\u0430\u043a \u0438 \u0441 pan gesture.<\/p>\n<pre><code class=\"swift\">private var trackedScrollView: UIScrollView?      private func setupScrollTrackingIfNeeded() {     trackScrollView(inside: presentedViewController) }  private func trackScrollView(inside viewController: UIViewController) {     guard         let scrollableViewController = viewController as? ScrollableBottomSheetPresentedController,         let scrollView = scrollableViewController.scrollView     else {         return     }          trackedScrollView?.multicastingDelegate.removeDelegate(self)     scrollView.multicastingDelegate.addDelegate(self)     self.trackedScrollView = scrollView }  private func removeScrollTrackingIfNeeded() {     trackedScrollView?.multicastingDelegate.removeDelegate(self)     trackedScrollView = nil }<\/code><\/pre>\n<p>\u0414\u0430\u043b\u044c\u0448\u0435 \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c <em>UIScrollViewDelegate.<\/em><\/p>\n<details class=\"spoiler\">\n<summary>\u0412\u0441\u043f\u043e\u043c\u043e\u0433\u0430\u0442\u0435\u043b\u044c\u043d\u044b\u0435 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u0438 \u0445\u0435\u043b\u043f\u0435\u0440\u044b UIScrollView<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"swift\">private var isInteractiveTransitionCanBeHandled: Bool {     isDragging }  private var isDragging = false private var overlayTranslation: CGFloat = 0 private var scrollViewTranslation: CGFloat = 0 private var lastContentOffsetBeforeDragging: CGPoint = .zero private var didStartDragging = false<\/code><\/pre>\n<p>\u0418 \u0445\u0435\u043b\u043f\u0435\u0440\u044b <em>UIScrollView.<\/em><\/p>\n<pre><code class=\"swift\">private extension UIScrollView {     var scrollsUp: Bool {         panGestureRecognizer.velocity(in: nil).y &lt; 0     }          var scrollsDown: Bool {         !scrollsUp     }          var isContentOriginInBounds: Bool {         contentOffset.y &lt;= -adjustedContentInset.top     } }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u041d\u0430\u0447\u043d\u0451\u043c \u0441 \u043a\u0430\u0441\u0430\u043d\u0438\u044f \u044d\u043a\u0440\u0430\u043d\u0430 <em>scrollViewWillBeginDragging(:_)<\/em>. \u0412 \u044d\u0442\u043e\u0442 \u043c\u043e\u043c\u0435\u043d\u0442 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0442\u043e\u043b\u044c\u043a\u043e-\u0442\u043e\u043b\u044c\u043a\u043e \u043d\u0430\u0447\u0430\u043b swipe-\u0436\u0435\u0441\u0442. \u0417\u0430\u043f\u043e\u043c\u0438\u043d\u0430\u0435\u043c \u044d\u0442\u043e \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u0447\u0435\u0440\u0435\u0437 \u0444\u043b\u0430\u0433 <em>isDragging.<\/em><\/p>\n<pre><code class=\"swift\">public func scrollViewWillBeginDragging(_ scrollView: UIScrollView) {     isDragging = true }<\/code><\/pre>\n<p>\u0414\u0430\u043b\u0435\u0435 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u0432\u0441\u043f\u043e\u043c\u043e\u0433\u0430\u0442\u0435\u043b\u044c\u043d\u0443\u044e \u0444\u0443\u043d\u043a\u0446\u0438\u044e <em>shouldDragOverlay(following:)<\/em>, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u0439 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u043c, \u043d\u0443\u0436\u043d\u043e \u043b\u0438 \u043e\u0431\u043d\u043e\u0432\u0438\u0442\u044c \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430.<\/p>\n<pre><code class=\"swift\">private func shouldDragOverlay(following scrollView: UIScrollView) -> Bool {     guard scrollView.isTracking, isInteractiveTransitionCanBeHandled else {         return false     }          if let percentComplete = interactionController?.percentComplete {         if percentComplete.isAlmostEqual(to: 0) {             return scrollView.isContentOriginInBounds &amp;&amp; scrollView.scrollsDown         }                  return true     } else {         return scrollView.isContentOriginInBounds &amp;&amp; scrollView.scrollsDown     } }<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0441\u0435\u0439\u0447\u0430\u0441 \u0432\u0435\u0434\u0451\u0442 \u043f\u0430\u043b\u044c\u0446\u0435\u043c \u043f\u043e \u044d\u043a\u0440\u0430\u043d\u0443. \u0415\u0441\u043b\u0438 \u0442\u0430\u043a, \u0442\u043e \u0440\u0430\u0437\u0431\u0438\u0440\u0430\u0435\u043c\u0441\u044f, \u0431\u044b\u043b\u043e \u043b\u0438 \u0438\u043d\u0438\u0446\u0438\u0438\u0440\u043e\u0432\u0430\u043d\u043e \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435.<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435 \u0438\u043d\u0438\u0446\u0438\u0438\u0440\u043e\u0432\u0430\u043d\u043e \u0438 \u043c\u044b \u0433\u0434\u0435-\u0442\u043e \u043f\u043e\u0441\u0435\u0440\u0435\u0434\u0438\u043d\u0435 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430, \u0442\u043e \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0430\u0435\u043c \u0434\u0430\u043b\u044c\u0448\u0435. \u0415\u0441\u043b\u0438 \u043c\u044b \u0442\u043e\u043b\u044c\u043a\u043e-\u0442\u043e\u043b\u044c\u043a\u043e \u0438\u043d\u0438\u0446\u0438\u0438\u0440\u043e\u0432\u0430\u043b\u0438 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435, \u0442\u043e \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0441\u043a\u0440\u043e\u043b\u043b \u043d\u0430\u043f\u0440\u0430\u0432\u043b\u0435\u043d \u0432\u043d\u0438\u0437 \u0438 \u043a\u043e\u043d\u0442\u0435\u043d\u0442 \u043d\u0430\u0445\u043e\u0434\u0438\u0442\u0441\u044f \u0441\u0432\u0435\u0440\u0445\u0443 \u0447\u0435\u0440\u0435\u0437 <em>isContentOriginInBounds<\/em>. \u0415\u0441\u043b\u0438 \u043c\u044b \u0432 \u043d\u0430\u0447\u0430\u043b\u0435 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430, \u0442\u043e \u0442\u0430\u043a\u0436\u0435 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0441\u043a\u0440\u043e\u043b\u043b \u043d\u0430\u043f\u0440\u0430\u0432\u043b\u0435\u043d \u0432\u043d\u0438\u0437 \u0438 \u043a\u043e\u043d\u0442\u0435\u043d\u0442 \u0441\u0432\u0435\u0440\u0445\u0443.<\/p>\n<p>\u0414\u0430\u043b\u044c\u0448\u0435 \u0440\u0430\u0437\u0431\u0435\u0440\u0451\u043c, \u043a\u043e\u0433\u0434\u0430 <em>contentOffset<\/em> \u0438\u0437\u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f \u0438 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f <em>scrollViewDidScroll(:_).<\/em><\/p>\n<pre><code class=\"swift\">public func scrollViewDidScroll(_ scrollView: UIScrollView) {     let previousTranslation = scrollViewTranslation     scrollViewTranslation = scrollView.panGestureRecognizer.translation(in: scrollView).y          didStartDragging = shouldDragOverlay(following: scrollView)     if didStartDragging {         startInteractiveTransitionIfNeeded()         overlayTranslation += scrollViewTranslation - previousTranslation                  \/\/ Update scrollView contentInset without invoking scrollViewDidScroll(_:)         scrollView.bounds.origin.y = -scrollView.adjustedContentInset.top                  updateInteractionControllerProgress(verticalTranslation: overlayTranslation)     } else {         lastContentOffsetBeforeDragging = scrollView.panGestureRecognizer.translation(in: scrollView)     } }  private func startInteractiveTransitionIfNeeded() {     guard interactionController == nil else { return }          startInteractiveTransition() }<\/code><\/pre>\n<p><em>startInteractiveTransitionIfNeeded()<\/em> \u2014 \u0438\u043d\u0438\u0446\u0438\u0438\u0440\u0443\u0435\u0442 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d, \u0435\u0441\u043b\u0438 \u043c\u044b \u044d\u0442\u043e \u0435\u0449\u0451 \u043d\u0435 \u0441\u0434\u0435\u043b\u0430\u043b\u0438.<\/p>\n<p>\u0412 <em>scrollViewDidScroll(_:)<\/em> \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u043c\u043e\u0436\u0435\u043c \u043b\u0438 \u043c\u044b \u043f\u0440\u043e\u0434\u043e\u043b\u0436\u0438\u0442\u044c (\u043d\u0430\u0447\u0430\u0442\u044c) \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d. \u0415\u0441\u043b\u0438 \u043c\u043e\u0436\u0435\u043c, \u0442\u043e \u0438\u043d\u0438\u0446\u0438\u0438\u0440\u0443\u0435\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d \u0447\u0435\u0440\u0435\u0437 <em>startInteractiveTransitionIfNeeded(). <\/em>\u0415\u0441\u043b\u0438 \u043d\u0435 \u043c\u043e\u0436\u0435\u043c, \u0442\u043e \u0437\u0430\u043f\u043e\u043c\u0438\u043d\u0430\u0435\u043c \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0439 <em>contentOffset<\/em> \u043f\u0435\u0440\u0435\u0434 \u0430\u043a\u0442\u0438\u0432\u0430\u0446\u0438\u0435\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430. \u041e\u043d \u043f\u0440\u0438\u0433\u043e\u0434\u0438\u0442\u0441\u044f \u0434\u0430\u043b\u044c\u0448\u0435.<\/p>\n<blockquote>\n<p>\u041f\u043e\u043c\u043d\u0438\u0442\u0435, \u043a\u0430\u043a \u043d\u0430 \u0441\u043e\u0431\u0435\u0441\u0435\u0434\u043e\u0432\u0430\u043d\u0438\u044f\u0445 \u0441\u043f\u0440\u0430\u0448\u0438\u0432\u0430\u043b\u0438, \u043a\u043e\u0433\u0434\u0430 <em>bounds.origin<\/em> \u043d\u0435\u043d\u0443\u043b\u0435\u0432\u043e\u0439? \u0412\u0435\u0440\u043e\u044f\u0442\u043d\u043e, \u043e\u0442\u0432\u0435\u0442 \u0431\u044b\u043b \u00ab\u043a\u043e\u0433\u0434\u0430 <em>contentOffset<\/em> \u0443 <em>scrollView <\/em>\u043d\u0435\u043d\u0443\u043b\u0435\u0432\u043e\u0439\u00bb. \u041d\u043e \u0431\u044b\u043b\u043e \u043d\u0435\u043f\u043e\u043d\u044f\u0442\u043d\u043e, \u043a\u0430\u043a \u044d\u0442\u043e \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u043d\u0430 \u043f\u0440\u0430\u043a\u0442\u0438\u043a\u0435, \u043f\u0440\u0430\u0432\u0434\u0430? \u041d\u0438\u0436\u0435 \u0445\u043e\u0440\u043e\u0448\u0430\u044f \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043e\u043f\u0440\u0430\u0432\u0434\u0430\u0442\u044c \u043d\u0430\u043b\u0438\u0447\u0438\u0435 \u044d\u0442\u043e\u0433\u043e \u0437\u043d\u0430\u043d\u0438\u044f!<\/p>\n<\/blockquote>\n<p>\u0414\u0430\u043b\u0435\u0435 \u0443\u0431\u0435\u0434\u0438\u043c\u0441\u044f, \u0447\u0442\u043e \u043a\u043e\u043d\u0442\u0435\u043d\u0442 \u043f\u0440\u0438\u0431\u0438\u0442 \u043a \u0432\u0435\u0440\u0445\u0443, \u0438 \u043f\u0440\u0438\u0440\u0430\u0432\u043d\u044f\u0435\u043c <em>contentInset <\/em>\u043a<em> contentOffset<\/em>. \u0418\u0437\u043c\u0435\u043d\u0435\u043d\u044f\u0435\u043c <em>contentOffset<\/em> \u0447\u0435\u0440\u0435\u0437 <em>bounds.origin<\/em>, \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u0432\u044b\u0437\u044b\u0432\u0430\u0442\u044c <em>scrollViewDidScroll(_:)<\/em>. \u0412 \u043a\u043e\u043d\u0446\u0435 \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0435\u043c \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430.<\/p>\n<p>\u041a\u043e\u0433\u0434\u0430 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043e\u0442\u0440\u044b\u0432\u0430\u0435\u0442 \u043f\u0430\u043b\u0435\u0446 \u043e\u0442 \u044d\u043a\u0440\u0430\u043d\u0430 \u043f\u043e\u0441\u043b\u0435 \u0441\u043a\u0440\u043e\u043b\u043b\u0430, \u0442\u043e <em>UIScrollViewDelegate<\/em> \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442 <em>scrollViewWillEndDragging(_:withVelocity:targetContentOffset:)<\/em>. \u0412 \u043d\u0451\u043c, \u043a\u0430\u043a \u0438 \u0432 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u043a\u0435 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f pan gesture <strong>ended<\/strong>, \u0437\u0430\u0432\u0435\u0440\u0448\u0430\u0435\u043c \u0438\u043b\u0438 \u043e\u0442\u043c\u0435\u043d\u044f\u0435\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d.<\/p>\n<pre><code class=\"swift\">public func scrollViewWillEndDragging(     _ scrollView: UIScrollView,     withVelocity velocity: CGPoint,     targetContentOffset: UnsafeMutablePointer&lt;CGPoint> ) {     if didStartDragging {         let velocity = scrollView.panGestureRecognizer.velocity(in: scrollView)         let translation = scrollView.panGestureRecognizer.translation(in: scrollView)         endInteractiveTransition(             verticalVelocity: velocity.y,             verticalTranslation: translation.y - lastContentOffsetBeforeDragging.y         )     } else {         endInteractiveTransition(isCancelled: true)     }          overlayTranslation = 0     scrollViewTranslation = 0     lastContentOffsetBeforeDragging = .zero     didStartDragging = false     isDragging = false }<\/code><\/pre>\n<p>\u0427\u0435\u0440\u0435\u0437 <em>didStartDragging<\/em> \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c <strong>\u0431\u044b\u043b\u043e \u043b\u0438 \u0430\u043a\u0442\u0438\u0432\u043d\u043e \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435<\/strong> \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435 Bottom Sheet \u043f\u0435\u0440\u0435\u0434 \u043e\u043a\u043e\u043d\u0447\u0430\u043d\u0438\u0435\u043c \u0441\u043a\u0440\u043e\u043b\u043b\u0430.<\/p>\n<p><strong>\u0415\u0441\u043b\u0438 \u0434\u0430<\/strong>, \u0442\u043e, \u043a\u0430\u043a \u0438 \u0441 pan gesture, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u044c \u0441 \u0437\u0430\u043c\u0435\u0434\u043b\u0435\u043d\u0438\u0435\u043c, \u0447\u0442\u043e\u0431\u044b \u0440\u0435\u0448\u0438\u0442\u044c, \u043e\u0442\u043c\u0435\u043d\u0438\u0442\u044c \u0438\u043b\u0438 \u0437\u0430\u043a\u043e\u043d\u0447\u0438\u0442\u044c \u043f\u0435\u0440\u0435\u0445\u043e\u0434.<\/p>\n<p><strong>\u0415\u0441\u043b\u0438 \u043d\u0435\u0442<\/strong>, \u0442\u043e \u043e\u0442\u043c\u0435\u043d\u044f\u0435\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d. \u0412\u043e\u0437\u043c\u043e\u0436\u043d\u0430 \u0441\u0438\u0442\u0443\u0430\u0446\u0438\u044f, \u0447\u0442\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043d\u0430\u0447\u0430\u043b \u0441\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet, \u0430 \u043f\u043e\u0442\u043e\u043c \u0432\u0435\u0440\u043d\u0443\u043b\u0441\u044f \u043a \u0441\u043a\u0440\u043e\u043b\u043b\u0443 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u0412 \u044d\u0442\u043e\u043c \u0441\u043b\u0443\u0447\u0430\u0435 \u0443 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430 \u043d\u0443\u043b\u0435\u0432\u043e\u0439 \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441, \u0438 \u043c\u044b \u0442\u043e\u0447\u043d\u043e \u0445\u043e\u0442\u0438\u043c \u0435\u0433\u043e \u043e\u0442\u043c\u0435\u043d\u0438\u0442\u044c.<\/p>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c!<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/954\/2f6\/601\/9542f66016297d9782e5d892610ecb84.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0418\u0442\u0430\u043a, \u043c\u044b \u043d\u0430\u0443\u0447\u0438\u043b\u0438\u0441\u044c \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u0441\u043e \u0432\u0441\u0435\u043c\u0438 \u0440\u0430\u0437\u043c\u0435\u0440\u0430\u043c\u0438 Bottom Sheet.<\/p>\n<h3>\u0427\u0442\u043e \u043c\u044b \u0441\u0434\u0435\u043b\u0430\u043b\u0438 \u0432\u043e \u0432\u0442\u043e\u0440\u043e\u0439 \u0447\u0430\u0441\u0442\u0438?<\/h3>\n<ol>\n<li>\n<p>\u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0430\u043b\u0438 pan gesture \u0438 \u0440\u0430\u0437\u043e\u0431\u0440\u0430\u043b\u0438 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f: <strong>began<\/strong>, <strong>changed<\/strong>, <strong>ended<\/strong>, <strong>cancelled<\/strong>.<\/p>\n<\/li>\n<li>\n<p>\u0420\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043b\u0438 \u043c\u043d\u043e\u0436\u0435\u0441\u0442\u0432\u0435\u043d\u043d\u0443\u044e \u043f\u043e\u0434\u043f\u0438\u0441\u043a\u0443 \u043d\u0430 <em>delegate<\/em> \u0447\u0435\u0440\u0435\u0437 <em>MulticastDelegate.<\/em><\/p>\n<\/li>\n<li>\n<p>\u041e\u0442\u0441\u043b\u0435\u0436\u0438\u0432\u0430\u0435\u043c <em>UIScrollViewDelegate<\/em> \u0438 \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0435\u043c \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0435 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430.<\/p>\n<\/li>\n<\/ol>\n<h2>\u0427\u0430\u0441\u0442\u044c 3. \u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u043c UINavigationController<\/h2>\n<h3>\u0421\u0442\u0430\u0440\u0442\u043e\u0432\u044b\u0439 \u043f\u0440\u043e\u0435\u043a\u0442<\/h3>\n<p>\u041d\u0430\u0447\u0438\u043d\u0430\u0435\u043c \u0441\u043e <a href=\"https:\/\/github.com\/joomcode\/BottomSheet\/tree\/feature\/part-3\">\u0441\u0442\u0430\u0440\u0442\u043e\u0432\u043e\u0433\u043e \u043f\u0440\u043e\u0435\u043a\u0442\u0430<\/a>. \u0412 <em>ResizeViewController<\/em> \u0434\u043e\u0431\u0430\u0432\u0438\u043b\u043e\u0441\u044c \u0434\u0432\u0435 \u043a\u043d\u043e\u043f\u043a\u0438, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0432\u0438\u0434\u043d\u044b, \u0435\u0441\u043b\u0438 \u0435\u0441\u0442\u044c navigation controller. \u041f\u0435\u0440\u0432\u0430\u044f \u043f\u0443\u0448\u0438\u0442 <em>ResizeViewController<\/em> \u0441 \u0442\u0435\u043a\u0443\u0449\u0435\u0439 \u0432\u044b\u0441\u043e\u0442\u043e\u0439 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u0412\u0442\u043e\u0440\u0430\u044f \u0441\u0432\u043e\u0440\u0430\u0447\u0438\u0432\u0430\u0435\u0442 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0439 \u0441\u0442\u0435\u043a \u043a <em>rootViewController<\/em>. \u041e\u0441\u0442\u0430\u043b\u044c\u043d\u043e\u0435 \u0438\u0437 \u0432\u0442\u043e\u0440\u043e\u0439 \u0447\u0430\u0441\u0442\u0438.<\/p>\n<p>\u0414\u043e\u0431\u0430\u0432\u0438\u043c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c <em>UINavigationController<\/em> c \u043f\u0440\u0438\u0432\u044b\u0447\u043d\u044b\u043c\u0438 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0438 push \u0438 pop. \u0422\u0430\u043a\u0436\u0435 \u043d\u0435 \u0437\u0430\u0431\u044b\u0432\u0430\u0435\u043c \u043f\u0440\u043e \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 pop, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0430\u0442\u044c.<\/p>\n<h3>\u0422\u0435\u043e\u0440\u0438\u044f. \u0410 \u0438\u0437 \u043a\u043e\u0440\u043e\u0431\u043a\u0438 \u0437\u0430\u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442?<\/h3>\n<p>\u041c\u043e\u0436\u043d\u043e \u043b\u0438 \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 <em>UINavigationController<\/em>? \u041a \u0441\u043e\u0436\u0430\u043b\u0435\u043d\u0438\u044e, \u043d\u0435\u0442.<\/p>\n<p>Navigation controller \u043d\u0435 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u0442 <em>preferredContentSize<\/em> \u0432 \u043f\u043e\u043b\u043d\u043e\u0439 \u043c\u0435\u0440\u0435. \u0418\u0437\u043d\u0430\u0447\u0430\u043b\u044c\u043d\u044b\u0439 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0438 \u0435\u0433\u043e \u0443\u0432\u0435\u043b\u0438\u0447\u0435\u043d\u0438\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043e\u0436\u0438\u0434\u0430\u0435\u043c\u043e. \u041e\u0434\u043d\u0430\u043a\u043e \u043d\u0430 \u0443\u043c\u0435\u043d\u044c\u0448\u0435\u043d\u0438\u0435 navigation controller \u043d\u0438\u043a\u0430\u043a \u043d\u0435 \u0440\u0435\u0430\u0433\u0438\u0440\u0443\u0435\u0442. \u041f\u0440\u0438 \u043d\u0430\u0436\u0430\u0442\u0438\u0438 \u043d\u0430 -100 \u0440\u0430\u0437\u043c\u0435\u0440 \u043d\u0435 \u0438\u0437\u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/630\/f50\/28d\/630f5028ddcbce0f5b794cb047afc0db.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u043d\u0430\u043c \u0442\u043e\u0447\u043d\u043e \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f \u043d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a <em>UINavigationController<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0441\u043c\u043e\u0436\u0435\u0442 \u043e\u0442\u0441\u043b\u0435\u0436\u0438\u0432\u0430\u0442\u044c \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0435 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u043e\u0433\u043e \u0441\u0442\u0435\u043a\u0430 \u0438 \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0442\u044c \u0441\u0432\u043e\u0439 \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0439 <em>preferredContentSize<\/em> \u043e\u0440\u0438\u0435\u043d\u0442\u0438\u0440\u0443\u044f\u0441\u044c \u043d\u0430 <em>topViewController.<\/em><\/p>\n<p>\u0412 presentation controller \u043f\u0440\u0438 \u043e\u0442\u0441\u043b\u0435\u0436\u0438\u0432\u0430\u043d\u0438\u0438 scrollView \u043d\u0443\u0436\u043d\u043e \u0443\u0447\u0435\u0441\u0442\u044c, \u0447\u0442\u043e <em>presentedViewController<\/em> \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c <em>UINavigationController<\/em>. \u0418 \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u043e\u0433\u043e \u0441\u0442\u0435\u043a\u0430 \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u043e \u0438\u0437\u0432\u043b\u0435\u0447\u044c <em>scrollView<\/em> \u0438\u0437 \u0442\u0435\u043a\u0443\u0449\u0435\u0433\u043e <em>topViewController<\/em>, \u0435\u0441\u043b\u0438 \u043e\u043d \u0435\u0441\u0442\u044c.<\/p>\n<p>\u041f\u043e\u0441\u043b\u0435\u0434\u043d\u0435\u0435. Navigation controller \u043f\u043e\u0441\u0442\u0430\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0432\u043c\u0435\u0441\u0442\u0435 \u0441 \u0432\u0435\u0440\u0441\u0438\u0435\u0439 iOS SDK. \u041f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f, \u0447\u0442\u043e \u043a\u0430\u0436\u0434\u044b\u0439 \u0440\u0430\u0437 \u043c\u044b \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u043c \u0441 \u043d\u043e\u0432\u044b\u043c \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u043e\u043c \u0441\u043e \u0441\u0432\u043e\u0438\u043c\u0438 \u043e\u0441\u043e\u0431\u0435\u043d\u043d\u043e\u0441\u0442\u044f\u043c\u0438. \u0418, \u043a\u0430\u043a \u043c\u044b \u0443\u0431\u0435\u0434\u0438\u043c\u0441\u044f \u0434\u0430\u043b\u044c\u0448\u0435, \u044d\u0442\u0438 \u043e\u0441\u043e\u0431\u0435\u043d\u043d\u043e\u0441\u0442\u0438 \u0441\u0435\u0431\u044f \u043f\u0440\u043e\u044f\u0432\u044f\u0442 \u0438 \u043d\u0430\u043c \u043f\u043e\u043c\u0435\u0448\u0430\u044e\u0442. \u041a\u0430\u043a \u0438 \u0447\u0442\u043e \u0431\u0443\u0434\u0435\u043c \u0441 \u044d\u0442\u0438\u043c \u0434\u0435\u043b\u0430\u0442\u044c \u2014 \u043e\u0431\u0441\u0443\u0434\u0438\u043c \u0431\u043b\u0438\u0436\u0435 \u043a \u0434\u0435\u043b\u0443.<\/p>\n<h3>\u0410\u0434\u0430\u043f\u0442\u0438\u0440\u0443\u0435\u043c\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430<\/h3>\n<p>\u041c\u044b \u0443\u0436\u0435 \u0434\u0435\u043b\u0430\u043b\u0438 \u0430\u0434\u0430\u043f\u0442\u0430\u0446\u0438\u044e \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0432 \u043f\u0435\u0440\u0432\u043e\u0439 \u0447\u0430\u0441\u0442\u0438, \u043d\u043e \u0441 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u043a\u043e\u0439 navigation controller \u0444\u0438\u0447\u0430 \u0441\u043b\u043e\u043c\u0430\u043b\u0430\u0441\u044c. \u0421\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 navigation controller \u043d\u0435 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u0442 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f <em>preferredContentSize<\/em> \u0432 \u043f\u043e\u043b\u043d\u043e\u0439 \u043c\u0435\u0440\u0435. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u0441\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u043d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a\u0430 <em>UINavigationController<\/em> \u0438 \u0432\u043e\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c\u0441\u044f \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u043e\u043c <em>UIContentContainer.<\/em><\/p>\n<p>\u0412 <em>updatePreferredContentSize<\/em> \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u043c <em>topViewController<\/em> \u0438 <em>additionalSafeAreaInsets.<\/em><\/p>\n<pre><code class=\"swift\">public final class BottomSheetNavigationController: UINavigationController {     private func updatePreferredContentSize() {         preferredContentSize = CGSize(             width: view.bounds.width,             height: topViewController?.preferredContentSize.height ?? 0 + additionalSafeAreaInsets.verticalInsets         )     } }<\/code><\/pre>\n<p>\u0410\u043d\u0430\u043b\u043e\u0433\u0438\u0447\u043d\u043e presentation controller, \u0440\u0435\u0430\u0433\u0438\u0440\u0443\u0435\u043c \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0435 content size \u0447\u0435\u0440\u0435\u0437 <em>preferredContentSizeDidChange(forChildContentContainer:)<\/em>. \u041f\u043e\u043c\u043d\u0438\u043c, \u0447\u0442\u043e \u043d\u0443\u0436\u043d\u043e \u0441\u0430\u043c\u0438\u043c \u043f\u043e\u0437\u0430\u0431\u043e\u0442\u0438\u0442\u044c\u0441\u044f \u043e\u0431 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0438 \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 <em>preferredContentSize<\/em>. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044e \u0432 <em>BottomSheetNavigationController<\/em> \u0438 \u0443\u0431\u0435\u0440\u0451\u043c \u0435\u0451 \u0438\u0437 <em>ResizeViewController.<\/em><\/p>\n<pre><code class=\"swift\">\/\/ BottomSheetNavigationController.swift  private var isUpdatingNavigationStack = false private var canAnimatePreferredContentSizeUpdates = false  \/\/ MARK: - Private methods  public override func preferredContentSizeDidChange(forChildContentContainer container: UIContentContainer) {     guard         let viewController = container as? UIViewController,         viewController === topViewController     else { return }          let updates = { [self] in         updatePreferredContentSize()         view.layoutIfNeeded()     }          if canAnimatePreferredContentSizeUpdates {         UIView.animate(withDuration: 0.25, animations: updates)     } else {         updates()     }          canAnimatePreferredContentSizeUpdates = true }<\/code><\/pre>\n<p>\u0412 presentation controller \u0443\u0447\u0442\u0451\u043c, \u0447\u0442\u043e <em>presentedViewController<\/em> \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c <em>UINavigationController<\/em>. \u0422\u043e\u0433\u0434\u0430 \u043d\u0443\u0436\u043d\u043e \u043f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0442\u044c\u0441\u044f \u043d\u0430 \u0442\u0435\u043a\u0443\u0449\u0438\u0439 <em>topViewController<\/em> \u0438 \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u043e\u0433\u043e \u0441\u0442\u0435\u043a\u0430. \u041e\u0431\u043d\u043e\u0432\u0438\u043c <em>setupScrollTrackingIfNeeded().<\/em><\/p>\n<pre><code class=\"swift\">private func setupScrollTrackingIfNeeded() {     if let navigationController = presentedViewController as? UINavigationController {         navigationController.multicastingDelegate.addDelegate(self)          if let topViewController = navigationController.topViewController {             trackScrollView(inside: topViewController)         }     } else {         trackScrollView(inside: presentedViewController)     } }<\/code><\/pre>\n<p>\u0414\u043b\u044f \u043e\u0442\u0441\u043b\u0435\u0436\u0438\u0432\u0430\u043d\u0438\u044f \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0439 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u043e\u0433\u043e \u0441\u0442\u0435\u043a\u0430 \u043f\u043e\u0434\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c\u0441\u044f \u043d\u0430 <em>delegate<\/em> \u0447\u0435\u0440\u0435\u0437 \u0438\u0437\u0432\u0435\u0441\u0442\u043d\u044b\u0439 \u043d\u0430\u043c \u043f\u0430\u0442\u0442\u0435\u0440\u043d <em>MulticastDelegate<\/em>. \u041f\u0440\u0438\u0441\u043c\u0430\u0442\u0440\u0438\u0432\u0430\u0435\u043c \u0437\u0430 scrollView \u043f\u0440\u0438 \u043f\u043e\u043a\u0430\u0437\u0435 view controller.<\/p>\n<pre><code class=\"swift\">extension BottomSheetPresentationController: UINavigationControllerDelegate {     public func navigationController(         _ navigationController: UINavigationController,         didShow viewController: UIViewController,         animated: Bool     ) {         trackScrollView(inside: viewController)     } }<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/4d7\/4bd\/552\/4d74bd552f013991c2538732b1fa85ea.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>Navigation controller \u0441\u0442\u0430\u043b \u0440\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f \u0440\u0430\u0437\u043c\u0435\u0440\u0430 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430, \u043d\u043e \u043f\u0440\u0438 \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0435 \u043d\u0430\u0437\u0430\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043d\u0435 \u0443\u0447\u0430\u0441\u0442\u0432\u0443\u0435\u0442 \u0432 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0438.<\/p>\n<h3>\u0410\u043d\u0438\u043c\u0438\u0440\u0443\u0435\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d push \u0438 pop<\/h3>\n<p>\u041f\u043e\u0447\u0435\u043c\u0443 \u0442\u0430\u043a \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442? \u041f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u0430\u044f \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044f navigation controller \u043d\u0435 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u0442 <em>preferredContentSize<\/em> \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0438 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u043e\u0433\u043e \u0441\u0442\u0435\u043a\u0430. \u041f\u043e\u044d\u0442\u043e\u043c\u0443 \u043d\u0443\u0436\u043d\u043e \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0442\u044c \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0432\u043c\u0435\u0441\u0442\u0435 \u0441 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f\u043c\u0438 \u0441\u0442\u0435\u043a\u0430 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u0438.<\/p>\n<p>\u0414\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u0432\u0432\u0435\u0434\u0451\u043c \u0432\u0441\u043f\u043e\u043c\u043e\u0433\u0430\u0442\u0435\u043b\u044c\u043d\u0443\u044e \u0444\u0443\u043d\u043a\u0446\u0438\u044e, \u0447\u0435\u0440\u0435\u0437 \u043a\u043e\u0442\u043e\u0440\u0443\u044e \u0431\u0443\u0434\u0443\u0442 \u0438\u0434\u0442\u0438 \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u044f \u0441\u0442\u0435\u043a\u0430 \u0432\u043c\u0435\u0441\u0442\u0435 \u0441 \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u0435\u043c <em>preferredContentSize<\/em>. \u0415\u0441\u043b\u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e, \u0442\u043e \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0435 \u0440\u0430\u0437\u043c\u0435\u0440\u0430 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0434\u0435\u043b\u0430\u0435\u043c \u0430\u043d\u0438\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e \u0447\u0435\u0440\u0435\u0437 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontroller\/1619294-transitioncoordinator\">transitionCoordinator<\/a>. \u0412\u0430\u0436\u043d\u043e \u0441\u043d\u0430\u0447\u0430\u043b\u0430 \u043e\u0431\u043d\u043e\u0432\u0438\u0442\u044c \u0441\u0442\u0435\u043a \u0438 \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u043e\u0442\u043e\u043c \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0442\u044c \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u0418\u043d\u0430\u0447\u0435 <em>topViewController<\/em> \u0431\u0443\u0434\u0435\u0442 \u043d\u0435\u0430\u043a\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u043c.<\/p>\n<pre><code class=\"swift\">\/\/ BottomSheetNavigationController.swift  private var isUpdatingNavigationStack = false  \/\/ MARK: - Private methods  private func updateNavigationStack(animated: Bool, applyChanges: () -> Void) {     isUpdatingNavigationStack = true          applyChanges()          if let transitionCoordinator = transitionCoordinator, animated, transitionCoordinator.isAnimated {         transitionCoordinator.animate(             alongsideTransition: { _ in                 self.updatePreferredContentSize()             },             completion: { context in                 self.isUpdatingNavigationStack = false                 self.updatePreferredContentSize()             }         )     } else {         isUpdatingNavigationStack = false         updatePreferredContentSize()     } }<\/code><\/pre>\n<p>\u041d\u0430\u043f\u043e\u0441\u043b\u0435\u0434\u043e\u043a \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c \u043c\u0435\u0442\u043e\u0434\u044b <em>UINavigationController<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0438\u0437\u043c\u0435\u043d\u044f\u044e\u0442 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0439 \u0441\u0442\u0435\u043a, \u0447\u0435\u0440\u0435\u0437 <em>updateNavigationStack(animated:applyChanges:).<\/em><\/p>\n<details class=\"spoiler\">\n<summary>\u041c\u0435\u0442\u043e\u0434\u044b, \u043c\u0430\u043d\u0438\u043f\u0443\u043b\u0438\u0440\u0443\u044e\u0449\u0438\u0435 \u0441\u0442\u0435\u043a\u043e\u043c UINavigationController<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"swift\">\/\/ MARK: - UINavigationController      public override func setViewControllers(_ viewControllers: [UIViewController], animated: Bool) {     updateNavigationStack(animated: animated) {         super.setViewControllers(viewControllers, animated: animated)     } }  public override func pushViewController(_ viewController: UIViewController, animated: Bool) {     updateNavigationStack(animated: animated) {         super.pushViewController(viewController, animated: animated)     } }  public override func popViewController(animated: Bool) -> UIViewController? {     var viewController: UIViewController?          updateNavigationStack(animated: animated) {         viewController = super.popViewController(animated: animated)     }          return viewController }  public override func popToRootViewController(animated: Bool) -> [UIViewController]? {     var viewControllers: [UIViewController]?          updateNavigationStack(animated: animated) {         viewControllers = super.popToRootViewController(animated: animated)     }          return viewControllers }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u0438 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/32f\/a96\/515\/32fa96515d5cd84961f1a3760d577410.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0421\u0442\u0430\u043b\u043e \u043b\u0443\u0447\u0448\u0435 \u0438 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u0442\u0441\u044f, \u043d\u043e \u0441 \u0430\u0440\u0442\u0435\u0444\u0430\u043a\u0442\u0430\u043c\u0438. \u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043d\u0430 iOS 12.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/86a\/ea4\/8fe\/86aea48fed54e967937466da17efe3aa.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0422\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d \u043e\u0442\u043b\u0438\u0447\u0430\u0435\u0442\u0441\u044f \u0432 \u0445\u0443\u0434\u0448\u0443\u044e \u0441\u0442\u043e\u0440\u043e\u043d\u0443 \u0438 \u043f\u043e\u0441\u043b\u0435 \u043e\u043a\u043e\u043d\u0447\u0430\u043d\u0438\u044f \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u043e\u0441\u0442\u0430\u0451\u0442\u0441\u044f \u0441 \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u0433\u043e view controller.<\/p>\n<p>\u041f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f, \u043c\u044b \u043d\u0435 \u043c\u043e\u0436\u0435\u043c \u0440\u0430\u0441\u0441\u0447\u0438\u0442\u044b\u0432\u0430\u0442\u044c \u043d\u0430 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d <em>UINavigationController,<\/em> \u0438 \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u0435\u0433\u043e \u0441\u0432\u043e\u0438\u043c\u0438 \u0441\u0438\u043b\u0430\u043c\u0438. \u0420\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c <em>UINavigationControllerDelegate<\/em>, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043c \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d \u0434\u043b\u044f push \u0438 pop \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0439.<\/p>\n<details class=\"spoiler\">\n<summary>\u0422\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d \u0434\u043b\u044f push \u0438 pop \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0439<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"swift\">public final class BottomSheetNavigationAnimatedTransitioning: NSObject, UIViewControllerAnimatedTransitioning {     \/\/ MARK: - Private          private let operation: UINavigationController.Operation          \/\/ MARK: - Init          public init(operation: UINavigationController.Operation) {         self.operation = operation     }          \/\/ MARK: - UIViewControllerAnimatedTransitioning          public func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval {         0.25     }          public func animateTransition(using transitionContext: UIViewControllerContextTransitioning) {         let containerView = transitionContext.containerView                  guard             let sourceViewController = transitionContext.viewController(forKey: .from),             let destinationViewController = transitionContext.viewController(forKey: .to),             let destinationView = destinationViewController.view,             let sourceView = sourceViewController.view,             let containerViewWindow = containerView.window         else {             return         }                  let isPushing = operation == .push                  let containerBounds = containerView.bounds                  let topView = isPushing ? destinationView : sourceView         let bottomView = isPushing ? sourceView : destinationView                  let topViewFrame = { bounds, isTopViewVisible -> CGRect in             isTopViewVisible                 ? bounds                 : bounds.offsetBy(dx: bounds.width, dy: 0)         }                  let bottomViewFrame = { bounds, isTopViewVisible -> CGRect in             isTopViewVisible                 ? bounds.offsetBy(dx: -bounds.width, dy: 0)                 : bounds         }                  let originalTopViewAutoresizingMask = topView.autoresizingMask         let originalBottomViewAutoresizingMask = bottomView.autoresizingMask                  topView.autoresizingMask = []         bottomView.autoresizingMask = []                  topView.frame = topViewFrame(containerBounds, !isPushing)         bottomView.frame = bottomViewFrame(containerBounds, !isPushing)                  containerView.addSubview(destinationView)                  destinationView.setNeedsUpdateConstraints()         destinationView.updateConstraintsIfNeeded()         destinationView.setNeedsLayout()         destinationView.layoutIfNeeded()                  let preferredContentSize = CGSize(             width: destinationViewController.preferredContentSize.width,             height: destinationViewController.preferredContentSize.height + destinationView.safeAreaInsets.top + destinationView.safeAreaInsets.bottom         )                  let maxHeight = containerViewWindow.bounds.size.height             - containerViewWindow.safeAreaInsets.top             - BottomSheetPresentationController.pullBarHeight                  let targetSize = CGSize(             width: preferredContentSize.width,             height: min(preferredContentSize.height, maxHeight)         )                  let navBarOffset = topView.safeAreaInsets.top         let separatorFrame = CGRect(             origin: CGPoint(                 x: topView.frame.origin.x,                 y: navBarOffset             ),             size: CGSize(                 width: pixelSize,                 height: containerBounds.size.height - navBarOffset             )         )         let separatorView = UIView(frame: separatorFrame)         if #available(iOS 13.0, *) {             separatorView.backgroundColor = .separator         } else {             separatorView.backgroundColor = .lightGray         }         containerView.addSubview(separatorView)                  let animations = {             let frame = CGRect(origin: .zero, size: targetSize)                          topView.frame = topViewFrame(isPushing ? frame : containerBounds, isPushing)             topView.layoutIfNeeded()                          bottomView.frame = bottomViewFrame(frame, isPushing)             bottomView.layoutIfNeeded()                          separatorView.frame = CGRect(                 origin: CGPoint(                     x: topView.frame.origin.x,                     y: navBarOffset                 ),                 size: separatorView.bounds.size             )         }                  let completion = { (isCompleted: Bool) in             containerView.addSubview(bottomView)             containerView.addSubview(topView)                          separatorView.removeFromSuperview()                          topView.autoresizingMask = originalTopViewAutoresizingMask             bottomView.autoresizingMask = originalBottomViewAutoresizingMask                          transitionContext.completeTransition(isCompleted &amp;&amp; !transitionContext.transitionWasCancelled)         }                  let duration = transitionDuration(using: transitionContext)         UIView.animate(withDuration: duration, delay: 0, options: .curveLinear, animations: animations, completion: completion)     } }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u0414\u0430\u043b\u0435\u0435 \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c <em>UINavigationControllerDelegate<\/em> \u0438 \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043c \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044e \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u0430.<\/p>\n<pre><code class=\"swift\">extension BottomSheetNavigationController: UINavigationControllerDelegate {     public func navigationController(         _ navigationController: UINavigationController,         animationControllerFor operation: UINavigationController.Operation,         from fromVC: UIViewController,         to toVC: UIViewController     ) -> UIViewControllerAnimatedTransitioning? {         BottomSheetNavigationAnimatedTransitioning(operation: operation)     } }<\/code><\/pre>\n<p>\u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0438 \u043f\u043e\u0441\u043b\u0435\u0434\u0438\u043c \u0437\u0430 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044f\u043c\u0438.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/0c5\/f51\/d8c\/0c5f51d8c273df9a9fc86ee18b7b41e4.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0412\u0441\u0451 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043f\u043b\u0430\u0432\u043d\u043e, \u043a\u0430\u043a \u043e\u0436\u0438\u0434\u0430\u0435\u0442\u0441\u044f! <\/p>\n<h3>\u0418\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 pop<\/h3>\n<p>\u041a\u043e\u0433\u0434\u0430 \u043c\u044b \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043b\u0438 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d \u0434\u043b\u044f push \u0438 pop \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0439, \u0442\u043e \u043c\u044b \u043f\u043e\u0442\u0435\u0440\u044f\u043b\u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 \u043f\u0435\u0440\u0435\u0445\u043e\u0434, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0434\u043e \u044d\u0442\u043e\u0433\u043e \u0440\u0430\u0431\u043e\u0442\u0430\u043b \u0438\u0437 \u043a\u043e\u0440\u043e\u0431\u043a\u0438. \u041f\u0440\u0438\u0434\u0435\u0442\u0441\u044f \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u0442\u044c \u0435\u0433\u043e \u0441\u0430\u043c\u0438\u043c. \u0418\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c <em>navigationController(_:interactionControllerFor:)<\/em> \u0438\u0437 <em>UINavigationControllerDelegate<\/em>. \u0427\u0435\u0440\u0435\u0437 \u043d\u0435\u0433\u043e \u043c\u043e\u0436\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d \u043f\u0440\u0438 \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0435 \u043c\u0435\u0436\u0434\u0443 view controller&#8217;\u0430\u043c\u0438.<\/p>\n<p>\u041d\u043e \u043a\u0430\u043a \u043f\u043e\u0432\u0442\u043e\u0440\u0438\u0442\u044c \u0436\u0435\u0441\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0441\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0435\u0442 \u043f\u0440\u0438 \u0441\u0432\u0430\u0439\u043f\u0435 \u043e\u0442 \u043a\u0440\u0430\u044f \u044d\u043a\u0440\u0430\u043d\u0430? \u0412\u043e\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c\u0441\u044f <em>UIScreenEdgePanGestureRecognizer<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0434\u0435\u043b\u0430\u0435\u0442 \u0440\u043e\u0432\u043d\u043e \u0442\u043e, \u0447\u0442\u043e \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e.<\/p>\n<p>\u0411\u0443\u0434\u0435\u043c \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0442\u044c \u044d\u0442\u043e\u0442 \u0436\u0435\u0441\u0442 \u0434\u043b\u044f view controller&#8217;\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0437\u0430\u043f\u0443\u0448\u0438\u043b\u0441\u044f \u0432 \u0438\u0435\u0440\u0430\u0440\u0445\u0438\u044e navigation controller&#8217;\u0430. \u041f\u0440\u0438\u0432\u043e\u0436\u0443 \u0442\u043e\u043b\u044c\u043a\u043e \u043a\u0440\u0438\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u0435 \u0443\u0447\u0430\u0441\u0442\u043a\u0438 \u043a\u043e\u0434\u0430.<\/p>\n<pre><code class=\"swift\">public extension UIViewController {     private(set) var customInteractivePopGestureRecognizer: UIGestureRecognizer? {         get { objc_getAssociatedObject(self, &amp;Self.gestureRecognizerKey) as? UIGestureRecognizer }         set { objc_setAssociatedObject(self, &amp;Self.gestureRecognizerKey, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC) }     }      func setupCustomInteractivePopTransition() {         let gestureRecognizer = UIScreenEdgePanGestureRecognizer()         view.addGestureRecognizer(gestureRecognizer)          gestureRecognizer.edges = .left         gestureRecognizer.addTarget(self, action: #selector(handleGestureRecognizer))          customInteractivePopGestureRecognizer = gestureRecognizer         ...     }      ... }<\/code><\/pre>\n<p>\u0412 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0435 \u0436\u0435\u0441\u0442\u0430 \u0434\u0435\u043b\u0430\u0435\u043c \u0442\u043e \u0436\u0435 \u0441\u0430\u043c\u043e\u0435, \u0447\u0442\u043e \u0438 \u0434\u043b\u044f pan gesture. \u0418\u043d\u0438\u0446\u0438\u0438\u0440\u0443\u0435\u043c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d, \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0435\u043c \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441 \u043f\u0440\u043e\u043f\u043e\u0440\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e \u0434\u0432\u0438\u0436\u0435\u043d\u0438\u044e \u043f\u0430\u043b\u044c\u0446\u0430 \u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0442\u043e\u0442 \u0436\u0435 \u043a\u0440\u0438\u0442\u0435\u0440\u0438\u0439 \u0434\u043b\u044f \u043e\u0442\u043c\u0435\u043d\u044b \u0438 \u0437\u0430\u0432\u0435\u0440\u0448\u0435\u043d\u0438\u044f \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0430.<\/p>\n<pre><code class=\"swift\">@objc private func handleGestureRecognizer(_ recognizer: UIScreenEdgePanGestureRecognizer) {     switch recognizer.state {     case .began:         processPanGestureBegan(recognizer)     case .changed:         processPanGestureChanged(recognizer)     case .ended:         processPanGestureEnded(recognizer)     case .cancelled, .failed:         processPanGestureCancelled(recognizer)     @unknown default:         break     } }<\/code><\/pre>\n<details class=\"spoiler\">\n<summary>\u041f\u043e\u043b\u043d\u0430\u044f \u0432\u0435\u0440\u0441\u0438\u044f<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"swift\">public extension UIViewController {     \/\/ MARK: - Public properties      private(set) var customInteractivePopGestureRecognizer: UIGestureRecognizer? {         get { objc_getAssociatedObject(self, &amp;Self.gestureRecognizerKey) as? UIGestureRecognizer }         set { objc_setAssociatedObject(self, &amp;Self.gestureRecognizerKey, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC) }     }          var customInteractivePopTransitioning: UIViewControllerInteractiveTransitioning? { transition }          \/\/ MARK: - Private properties          private static var gestureRecognizerKey: UInt8 = 0     private static var gestureRecognizerDelegateKey: UInt8 = 0     private static var transitionKey: UInt8 = 0          private var transition: UIPercentDrivenInteractiveTransition? {         get { objc_getAssociatedObject(self, &amp;Self.transitionKey) as? UIPercentDrivenInteractiveTransition }         set { objc_setAssociatedObject(self, &amp;Self.transitionKey, newValue, .OBJC_ASSOCIATION_RETAIN_NONATOMIC) }     }          \/\/ MARK: - Public methods          func setupCustomInteractivePopTransition() {         let gestureRecognizer = UIScreenEdgePanGestureRecognizer()         gestureRecognizer.edges = .left         let gestureRecognizerDelegate = GestureRecognizerDelegate(navigationItem: navigationItem)         gestureRecognizer.delegate = gestureRecognizerDelegate                  if let view = viewIfLoaded {             view.addGestureRecognizer(gestureRecognizer)         } else {             subscribe(onEvent: .viewDidLoad) { [unowned self] in                 view.addGestureRecognizer(gestureRecognizer)             }         }                  gestureRecognizer.addTarget(self, action: #selector(handleGestureRecognizer))         customInteractivePopGestureRecognizer = gestureRecognizer         objc_setAssociatedObject(self, &amp;Self.gestureRecognizerDelegateKey, gestureRecognizerDelegate, .OBJC_ASSOCIATION_RETAIN_NONATOMIC)     }          \/\/ MARK: - Private methods          @objc     private func handleGestureRecognizer(_ recognizer: UIScreenEdgePanGestureRecognizer) {         switch recognizer.state {         case .began:             processPanGestureBegan(recognizer)         case .changed:             processPanGestureChanged(recognizer)         case .ended:             processPanGestureEnded(recognizer)         case .cancelled, .failed:             processPanGestureCancelled(recognizer)         @unknown default:             break         }     }          private func processPanGestureBegan(_ recognizer: UIScreenEdgePanGestureRecognizer) {         transition = UIPercentDrivenInteractiveTransition()         navigationController?.popViewController(animated: true)     }          private func processPanGestureChanged(_ recognizer: UIScreenEdgePanGestureRecognizer) {         let translation = recognizer.translation(in: view)         let progress: CGFloat = translation.x \/ view.bounds.width         transition?.update(progress)     }          private func processPanGestureEnded(_ recognizer: UIScreenEdgePanGestureRecognizer) {         let velocity = recognizer.velocity(in: view)         let translation = recognizer.translation(in: view)          let deceleration = 800.0 * (velocity.x > 0 ? -1.0 : 1.0)         let finalProgress = (translation.x - 0.5 * velocity.x * velocity.x \/ CGFloat(deceleration)) \/ view.bounds.width                  let isThresholdPassed = finalProgress &lt; 0.5                  endTransition(isCancelled: isThresholdPassed)     }          private func processPanGestureCancelled(_ recognizer: UIScreenEdgePanGestureRecognizer) {         endTransition(isCancelled: true)     }          private func endTransition(isCancelled: Bool) {         if isCancelled {             transition?.cancel()         } else {             transition?.finish()         }                  transition = nil     } }  private final class GestureRecognizerDelegate: NSObject, UIGestureRecognizerDelegate {     private let navigationItem: UINavigationItem      init(navigationItem: UINavigationItem) {         self.navigationItem = navigationItem     }          func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool {         !navigationItem.hidesBackButton     }          func gestureRecognizer(         _ gestureRecognizer: UIGestureRecognizer,         shouldBeRequiredToFailBy otherGestureRecognizer: UIGestureRecognizer     ) -> Bool {         true     } }<\/code><\/pre>\n<\/div>\n<\/details>\n<p>\u041e\u0441\u0442\u0430\u0451\u0442\u0441\u044f \u043f\u0440\u043e\u043a\u0430\u0447\u0430\u0442\u044c <em>UINavigationControllerDelegate<\/em> \u0438 \u0443\u0447\u0435\u0441\u0442\u044c \u0432 \u043d\u0451\u043c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d. \u041d\u0430 push \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u043a view controller&#8217;\u0443 \u0436\u0435\u0441\u0442 \u0434\u043b\u044f \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0433\u043e pop&#8217;\u0430.<\/p>\n<pre><code class=\"swift\">extension BottomSheetNavigationController: UINavigationControllerDelegate {     public func navigationController(         _ navigationController: UINavigationController,         animationControllerFor operation: UINavigationController.Operation,         from fromVC: UIViewController,         to toVC: UIViewController     ) -> UIViewControllerAnimatedTransitioning? {         if operation == .push {             toVC.setupCustomInteractivePopTransition()         }                  lastTransitionViewController = fromVC         return BottomSheetNavigationAnimatedTransitioning(operation: operation)     }          public func navigationController(         _ navigationController: UINavigationController,         interactionControllerFor animationController: UIViewControllerAnimatedTransitioning     ) -> UIViewControllerInteractiveTransitioning? {         lastTransitionViewController?.customInteractivePopTransitioning     } }<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c.<\/p>\n<figure class=\"full-width\"><img decoding=\"async\" src=\"\/img\/image-loader.svg\" height=\"1080\" data-src=\"https:\/\/habrastorage.org\/getpro\/habr\/upload_files\/610\/252\/381\/61025238178472ff9f2a3478088a94b6.gif\" data-width=\"1920\"\/><figcaption><\/figcaption><\/figure>\n<p>\u0418 \u0443\u0440\u0430, \u043c\u044b \u0441\u043f\u0440\u0430\u0432\u0438\u043b\u0438\u0441\u044c \u0441 \u043d\u0430\u0448\u0438\u043c \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u043c \u0432\u044b\u0437\u043e\u0432\u043e\u043c!<\/p>\n<h3>\u0427\u0442\u043e \u043c\u044b \u0441\u0434\u0435\u043b\u0430\u043b\u0438 \u0432 \u0442\u0440\u0435\u0442\u044c\u0435\u0439 \u0447\u0430\u0441\u0442\u0438?<\/h3>\n<ol>\n<li>\n<p>\u0410\u0434\u0430\u043f\u0442\u0438\u0440\u043e\u0432\u0430\u043b\u0438 <em>UINavigationController<\/em> \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430.<\/p>\n<\/li>\n<li>\n<p>\u0414\u043e\u0431\u0438\u043b\u0438\u0441\u044c \u043a\u043e\u0440\u0440\u0435\u043a\u0442\u043d\u044b\u0445 push \u0438 pop \u0442\u0440\u0430\u043d\u0437\u0438\u0448\u0435\u043d\u043e\u0432 \u043d\u0430 iOS 12+.<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0430\u043b\u0438 \u0443\u043d\u0438\u043a\u0430\u043b\u044c\u043d\u044b\u0439 \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u044b\u0439 pop.<\/p>\n<\/li>\n<\/ol>\n<h2>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435<\/h2>\n<p>\u0421\u043f\u0430\u0441\u0438\u0431\u043e, \u0447\u0442\u043e \u0434\u043e\u0447\u0438\u0442\u0430\u043b\u0438 \u0434\u043e \u043a\u043e\u043d\u0446\u0430! \u0422\u0435\u043f\u0435\u0440\u044c \u043c\u043e\u0436\u043d\u043e \u0438 \u0440\u0435\u0437\u044e\u043c\u0435 \u043e\u0431\u043d\u043e\u0432\u0438\u0442\u044c. <\/p>\n<p>\u041c\u044b \u043d\u0430\u0447\u0430\u043b\u0438 \u0441 \u043d\u0443\u043b\u044f, \u0430 \u0432 \u043a\u043e\u043d\u0446\u0435 \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438 <a href=\"https:\/\/github.com\/joomcode\/BottomSheet\">\u043c\u043d\u043e\u0433\u043e\u0444\u0443\u043d\u043a\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u044b\u0439 Bottom Sheet<\/a>. \u041e\u0442\u0432\u0435\u0442\u0438\u043b\u0438 \u043d\u0430 \u0432\u043e\u043f\u0440\u043e\u0441\u044b, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0441\u0442\u043e\u044f\u043b\u0438 \u0432 \u043d\u0430\u0447\u0430\u043b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438, \u0438 \u0434\u0430\u0436\u0435 \u0432\u0441\u043f\u043e\u043c\u043d\u0438\u043b\u0438 \u0448\u043a\u043e\u043b\u044c\u043d\u0443\u044e \u0444\u0438\u0437\u0438\u043a\u0443. \u041d\u0430\u0434\u0435\u044e\u0441\u044c, \u0447\u0442\u043e \u0441\u0442\u0430\u0442\u044c\u044f \u043e\u043a\u0430\u0437\u0430\u043b\u0430\u0441\u044c \u043f\u043e\u043b\u0435\u0437\u043d\u043e\u0439, \u0438 \u0432 \u0432\u0430\u0448\u0435\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0438 Bottom Sheet \u0437\u0430\u0438\u0433\u0440\u0430\u0435\u0442 \u043d\u043e\u0432\u044b\u043c\u0438 \u043a\u0440\u0430\u0441\u043a\u0430\u043c\u0438!<\/p>\n<\/div>\n<\/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\/company\/joom\/blog\/596821\/\"> https:\/\/habr.com\/ru\/company\/joom\/blog\/596821\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<div><\/div>\n<div id=\"post-content-body\" class=\"article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>Bottom Sheet \u043f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u043b\u044f\u043b\u0441\u044f \u043c\u043d\u0435 \u0441\u043b\u043e\u0436\u043d\u044b\u043c \u0438 \u043d\u0435\u0434\u043e\u0441\u044f\u0433\u0430\u0435\u043c\u044b\u043c. \u042d\u0442\u043e \u0431\u044b\u043b \u0432\u044b\u0437\u043e\u0432! \u042f \u043d\u0435 \u043f\u043e\u043d\u0438\u043c\u0430\u043b, \u0441 \u0447\u0435\u0433\u043e \u043d\u0430\u0447\u0430\u0442\u044c. \u0412\u043e\u0437\u043d\u0438\u043a\u0430\u043b\u043e \u043c\u043d\u043e\u0433\u043e \u0432\u043e\u043f\u0440\u043e\u0441\u043e\u0432: \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c view \u0438\u043b\u0438 view controller? Auto \u0438\u043b\u0438 manual layout? \u041a\u0430\u043a \u0430\u043d\u0438\u043c\u0438\u0440\u043e\u0432\u0430\u0442\u044c? \u041a\u0430\u043a \u0441\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e?<\/p>\n<p>\u041d\u043e \u0432\u0441\u0451 \u0438\u0437\u043c\u0435\u043d\u0438\u043b\u043e\u0441\u044c \u043f\u043e\u0441\u043b\u0435 \u0440\u0430\u0431\u043e\u0442\u044b \u043d\u0430\u0434 Bottom Sheet \u0434\u043b\u044f \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f Joom, \u0433\u0434\u0435 \u043e\u043d \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u043f\u043e\u0432\u0441\u0435\u043c\u0435\u0441\u0442\u043d\u043e. \u0412 \u0442\u043e\u043c \u0447\u0438\u0441\u043b\u0435 \u0438 \u0432 \u0442\u0430\u043a\u0438\u0445 \u043a\u0440\u0438\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u0445 \u0441\u0446\u0435\u043d\u0430\u0440\u0438\u044f\u0445, \u043a\u0430\u043a \u043e\u043f\u043b\u0430\u0442\u0430. \u0422\u0430\u043a \u0447\u0442\u043e \u043c\u043e\u0433\u0443 \u0442\u043e\u0447\u043d\u043e \u0441\u043a\u0430\u0437\u0430\u0442\u044c, \u0447\u0442\u043e \u0432 \u044d\u0442\u043e\u043c \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442\u0435 \u043c\u044b \u0443\u0432\u0435\u0440\u0435\u043d\u044b. \u041d\u0430\u0441\u0442\u043e\u043b\u044c\u043a\u043e \u0443\u0432\u0435\u0440\u0435\u043d\u044b, \u0447\u0442\u043e \u044f \u0434\u0430\u0436\u0435 <a href=\"https:\/\/youtu.be\/4VLYzbrKe7c\">\u0440\u0430\u0441\u0441\u043a\u0430\u0437\u044b\u0432\u0430\u043b<\/a> \u043e \u043d\u0451\u043c \u043d\u0430 Podlodka iOS crew #7. \u0412 \u0440\u0430\u043c\u043a\u0430\u0445 \u0432\u043e\u0440\u043a\u0448\u043e\u043f\u0430 \u044f \u043f\u043e\u043a\u0430\u0437\u0430\u043b, \u043a\u0430\u043a \u0441\u0434\u0435\u043b\u0430\u0442\u044c Bottom Sheet, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0443\u043c\u0435\u0435\u0442 \u043f\u043e\u0434\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430, \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u0438 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 <em>UINavigationController.<\/em><\/p>\n<p>\u0421\u0442\u043e\u043f, \u043d\u043e Apple \u0436\u0435 <a href=\"https:\/\/developer.apple.com\/videos\/play\/wwdc2021\/10063\/\">\u043f\u0440\u0435\u0434\u043e\u0441\u0442\u0430\u0432\u0438\u043b\u0430<\/a> \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 <a href=\"https:\/\/developer.apple.com\/design\/human-interface-guidelines\/ios\/views\/sheets\/\">Bottom Sheet<\/a>. \u0417\u0430\u0447\u0435\u043c \u043f\u0438\u0441\u0430\u0442\u044c \u0441\u0432\u043e\u0439? \u0414\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e, \u044d\u0442\u043e \u0442\u0430\u043a, \u043d\u043e \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442\u0441\u044f \u0442\u043e\u043b\u044c\u043a\u043e \u0441 iOS 15. \u0410 \u044d\u0442\u043e \u0437\u043d\u0430\u0447\u0438\u0442, \u0447\u0442\u043e \u043f\u043e\u043b\u043d\u043e\u0446\u0435\u043d\u043d\u043e \u0435\u0433\u043e \u043c\u043e\u0436\u043d\u043e \u0431\u0443\u0434\u0435\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u0447\u0435\u0440\u0435\u0437 2-3 \u0433\u043e\u0434\u0430. \u041a \u0442\u043e\u043c\u0443 \u0436\u0435 \u0447\u0430\u0441\u0442\u043e \u0442\u0440\u0435\u0431\u043e\u0432\u0430\u043d\u0438\u044f \u0434\u0438\u0437\u0430\u0439\u043d\u0435\u0440\u043e\u0432 \u0432\u044b\u0445\u043e\u0434\u044f\u0442 \u0437\u0430 \u0440\u0430\u043c\u043a\u0438 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0445 iOS-\u044d\u043b\u0435\u043c\u0435\u043d\u0442\u043e\u0432.<\/p>\n<p>\u0412 \u0440\u0430\u043c\u043a\u0430\u0445 \u0441\u0442\u0430\u0442\u044c\u0438 \u0445\u043e\u0447\u0443 \u0440\u0430\u0437\u0432\u0435\u044f\u0442\u044c \u0442\u0443\u043c\u0430\u043d \u043d\u0430\u0434 Bottom Sheet, \u043e\u0442\u0432\u0435\u0442\u0438\u0442\u044c \u043d\u0430 \u0432\u043e\u043f\u0440\u043e\u0441\u044b, \u043a\u043e\u0442\u043e\u0440\u044b\u043c\u0438 \u0437\u0430\u0434\u0430\u0432\u0430\u043b\u0441\u044f \u044f \u0441\u0430\u043c \u0438 \u043f\u0440\u0435\u0434\u043b\u043e\u0436\u0438\u0442\u044c \u043e\u0434\u0438\u043d \u0438\u0437 \u0432\u0430\u0440\u0438\u0430\u043d\u0442\u043e\u0432 \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0438. \u0427\u0442\u043e\u0431\u044b \u0432 \u043a\u043e\u043d\u0446\u0435 \u0432\u044b \u043c\u043e\u0433\u043b\u0438 \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432 \u0440\u0435\u0437\u044e\u043c\u0435 \u0441\u0442\u0440\u043e\u0447\u043a\u0443 \u00ab\u041f\u0440\u043e\u0444\u0435\u0441\u0441\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e \u0434\u0435\u043b\u0430\u044e Bottom Sheet&#8217;\u044b\u00bb.<\/p>\n<figure class=\"full-width\"><figcaption><\/figcaption><\/figure>\n<p>\u0415\u0441\u043b\u0438 \u0437\u0430\u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043e\u0432\u0430\u043b, \u0442\u043e \u043d\u0430\u0447\u043d\u0451\u043c! \u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u043f\u0440\u043e\u0441\u0442\u043e\u0439 Bottom Sheet \u0438 \u0448\u0430\u0433 \u0437\u0430 \u0448\u0430\u0433\u043e\u043c \u0435\u0433\u043e \u043f\u0440\u043e\u043a\u0430\u0447\u0430\u0435\u043c. <\/p>\n<ol>\n<li>\n<p>\u041d\u0430\u0443\u0447\u0438\u043c\u0441\u044f \u043f\u043e\u0434\u0441\u0442\u0440\u0430\u0438\u0432\u0430\u0442\u044c\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0438 \u0437\u0430\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet.<\/p>\n<\/li>\n<li>\n<p>\u0414\u043e\u0431\u0430\u0432\u0438\u043c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435, \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u044f \u043a\u043e\u043d\u0442\u0435\u043d\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0441\u043a\u0440\u043e\u043b\u043b\u0438\u0442\u0441\u044f.<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u043c <em>UINavigationController<\/em> \u0441 \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u0435\u0439 \u0432\u043d\u0443\u0442\u0440\u0438 Bottom Sheet.<\/p>\n<\/li>\n<\/ol>\n<h3>\u0427\u0430\u0441\u0442\u044c 1. \u0410\u0434\u0430\u043f\u0442\u0438\u0440\u0443\u0435\u043c\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u0417\u0430\u043a\u0440\u044b\u0432\u0430\u0435\u043c Bottom Sheet. \u0411\u0430\u0437\u043e\u0432\u044b\u0439 \u0434\u0438\u0437\u0430\u0439\u043d<\/h3>\n<p>\u0421\u043f\u0435\u0440\u0432\u0430 \u0443\u0431\u0435\u0434\u0438\u043c\u0441\u044f, \u0447\u0442\u043e \u043c\u044b \u043f\u043e\u043d\u0438\u043c\u0430\u0435\u043c \u043f\u043e\u0434 &#171;Bottom Sheet&#187; \u043e\u0434\u043d\u043e \u0438 \u0442\u043e \u0436\u0435. Bottom Sheet &#8212; \u044d\u0442\u043e \u043a\u043e\u043c\u043f\u043e\u043d\u0435\u043d\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0440\u0430\u0441\u043f\u043e\u043b\u0430\u0433\u0430\u0435\u0442\u0441\u044f \u0441\u043d\u0438\u0437\u0443 \u0438 \u0430\u0434\u0430\u043f\u0442\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u043f\u043e\u0434 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u041f\u0440\u0438\u043c\u0435\u0440\u044b \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f \u0435\u0441\u0442\u044c \u0432 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0445 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f\u0445: <a href=\"https:\/\/apps.apple.com\/us\/app\/apple-maps\/id915056765\">Apple Maps<\/a> (\u043f\u043e\u0438\u0441\u043a), <a href=\"https:\/\/apps.apple.com\/us\/app\/stocks\/id1069512882\">Stocks<\/a> (\u043d\u043e\u0432\u043e\u0441\u0442\u0438), <a href=\"https:\/\/apps.apple.com\/us\/app\/voice-memos\/id1069512134\">Voice Memos<\/a> (\u0437\u0430\u043f\u0438\u0441\u044c \u0433\u043e\u043b\u043e\u0441\u0430) \u0438 \u0442.\u0434.<\/p>\n<figure class=\"full-width\"><figcaption>Apple Maps, Stocks, Voice Memos<\/figcaption><\/figure>\n<h3>\u0421\u0442\u0430\u0440\u0442\u043e\u0432\u044b\u0439 \u043f\u0440\u043e\u0435\u043a\u0442<\/h3>\n<p>\u0412\u043e\u0442 \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u0441\u0442\u0430\u0440\u0442\u043e\u0432\u044b\u0439 \u043f\u0440\u043e\u0435\u043a\u0442 \u043d\u0430 <a href=\"https:\/\/github.com\/joomcode\/BottomSheet\">Github<\/a>. \u0412 \u043f\u0440\u043e\u0435\u043a\u0442\u0435 \u0435\u0441\u0442\u044c \u0434\u0432\u0430 \u0442\u0430\u0440\u0433\u0435\u0442\u0430: <em>BottomSheetDemo<\/em> \u0438 <em>BottomSheet<\/em> \u2014 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435 \u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430 \u0441 Bottom Sheet.<\/p>\n<details class=\"spoiler\">\n<summary>\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430<\/summary>\n<div class=\"spoiler__content\">\n<pre><code class=\"yaml\">BottomSheetDemo     Sources\/User Interface         Screens             Resize                 ResizeViewController.swift             Root                 RootViewController.swift BottomSheet     Core         BottomSheetModalDismissalHandler.swift         BottomSheetPresentationController.swift         BottomSheetPresentationController+PullBar.swift         BottomSheetTransitioningDelegate.swift     Helpers         ...<\/code><\/pre>\n<\/div>\n<\/details>\n<p><em>RootViewController<\/em> \u2014 \u044d\u0442\u043e \u043f\u0435\u0440\u0432\u044b\u0439 \u044d\u043a\u0440\u0430\u043d \u0432 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0438. \u0412 \u043d\u0451\u043c \u0435\u0441\u0442\u044c \u0432\u0441\u0435\u0433\u043e \u043e\u0434\u043d\u0430 \u043a\u043d\u043e\u043f\u043a\u0430 Show Bottom Sheet. \u041f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e \u043f\u043e\u043a\u0430\u0436\u0435\u0442\u0441\u044f <em>ResizeViewController.<\/em><\/p>\n<pre><code class=\"swift\">@objc private func handleShowBottomSheet() {     let viewController = ResizeViewController(initialHeight: 300)     present(viewController, animated: true, completion: nil) }<\/code><\/pre>\n<p>\u0412 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0442\u043e\u0440\u0435 <em>ResizeViewController<\/em> \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u0435\u0442 \u0432\u044b\u0441\u043e\u0442\u0443 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430. \u0422\u0430\u043a\u0436\u0435 \u0435\u0441\u0442\u044c \u0447\u0435\u0442\u044b\u0440\u0435 \u043a\u043d\u043e\u043f\u043a\u0438, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0438\u0437\u043c\u0435\u043d\u044f\u044e\u0442 \u0432\u044b\u0441\u043e\u0442\u0443 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430: \u043d\u0430 +100 \u0438 -100, \u0432 2 \u0438 0.5 \u0440\u0430\u0437.<\/p>\n<p>\u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<figure class=\"full-width\"><figcaption><\/figcaption><\/figure>\n<h3>\u0422\u0435\u043e\u0440\u0438\u044f. \u041a\u0430\u043a \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c Bottom Sheet?<\/h3>\n<p>\u041d\u0430\u043c \u043d\u0443\u0436\u043d\u0430 \u0441\u0443\u0449\u043d\u043e\u0441\u0442\u044c, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0431\u0443\u0434\u0435\u0442 \u0443\u043f\u0440\u0430\u0432\u043b\u044f\u0442\u044c \u043f\u043e\u043a\u0430\u0437\u043e\u043c. \u041e\u043d\u0430 \u0434\u043e\u0431\u0430\u0432\u0438\u0442 Bottom Sheet \u0432 UI-\u0438\u0435\u0440\u0430\u0440\u0445\u0438\u044e, \u0440\u0430\u0441\u043f\u043e\u043b\u043e\u0436\u0438\u0442 \u0435\u0433\u043e \u043d\u0430 \u044d\u043a\u0440\u0430\u043d\u0435, \u0443\u0447\u0442\u0451\u0442 \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430, \u0431\u0443\u0434\u0435\u0442 \u0440\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430 \u0435\u0433\u043e \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f, \u043f\u043e\u0437\u0430\u0431\u043e\u0442\u0438\u0442\u0441\u044f \u043e\u0431 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0438, \u0438 \u043c\u043e\u0436\u043d\u043e \u0431\u0443\u0434\u0435\u0442 \u0441\u0434\u0435\u043b\u0430\u0442\u044c \u0438\u043d\u0442\u0435\u0440\u0430\u043a\u0442\u0438\u0432\u043d\u043e\u0435 \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u0435.<\/p>\n<p>\u042d\u0442\u043e \u043f\u043e\u0445\u043e\u0436\u0435 \u043d\u0430 \u0437\u043e\u043d\u0443 \u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u043e\u0441\u0442\u0438 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uipresentationcontroller\">UIPresentationController<\/a>. \u0421 \u043c\u043e\u043c\u0435\u043d\u0442\u0430 \u043f\u043e\u044f\u0432\u043b\u0435\u043d\u0438\u044f view controller&#8217;\u0430 \u0438 \u0434\u043e \u043c\u043e\u043c\u0435\u043d\u0442\u0430 \u0441\u043a\u0440\u044b\u0442\u0438\u044f, UIKit \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 presentation controller \u0434\u043b\u044f \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u043e\u043c \u043f\u043e\u043a\u0430\u0437\u0430.<\/p>\n<p>\u0414\u043b\u044f \u0435\u0433\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f \u043d\u0430\u0434\u043e \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u0442\u044c <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontroller\/1621355-modalpresentationstyle\">modalPresentationStyle<\/a> \u0438 \u043f\u0435\u0440\u0435\u0434\u0430\u0442\u044c presentation controller \u0447\u0435\u0440\u0435\u0437 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontroller\/1621421-transitioningdelegate\">transitioningDelegate<\/a>.<\/p>\n<p>\u0412\u043e\u043e\u0440\u0443\u0436\u0438\u0432\u0448\u0438\u0441\u044c \u044d\u0442\u0438\u043c \u0437\u043d\u0430\u043d\u0438\u0435\u043c, \u043d\u0430\u0447\u043d\u0451\u043c \u0434\u0435\u043b\u0430\u0442\u044c Bottom Sheet!<\/p>\n<h3>\u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c presentation controller<\/h3>\n<p>\u0414\u043b\u044f \u043f\u043e\u043a\u0430\u0437\u0430 Bottom Sheet \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043c <em>modalPresentationStyle<\/em> \u0438 <em>transitioningDelegate<\/em>. \u041d\u0435 \u0437\u0430\u0431\u044b\u0432\u0430\u0435\u043c, \u0447\u0442\u043e <em>transitioningDelegate<\/em> \u2014 \u044d\u0442\u043e weak \u0441\u0441\u044b\u043b\u043a\u0430, \u0438 \u043d\u0430\u043c \u043f\u043e\u043d\u0430\u0434\u043e\u0431\u0438\u0442\u0441\u044f strong \u0441\u0441\u044b\u043b\u043a\u0430, \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u043f\u043e\u0442\u0435\u0440\u044f\u0442\u044c \u043e\u0431\u044a\u0435\u043a\u0442.<\/p>\n<pre><code class=\"swift\">private var bottomSheetTransitioningDelegate: UIViewControllerTransitioningDelegate?  @objc private func handleShowBottomSheet() {     let viewController = ResizeViewController(initialHeight: 300)     \/\/ TODO: bottomSheetTransitioningDelegate = ...     viewController.modalPresentationStyle = .custom     viewController.transitioningDelegate = bottomSheetTransitioningDelegate     present(viewController, animated: true, completion: nil) }<\/code><\/pre>\n<p>\u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c BottomSheetTransitioningDelegate \u2014 \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e <em>transitioningDelegate.<\/em><\/p>\n<pre><code class=\"swift\">public final class BottomSheetTransitioningDelegate: NSObject, UIViewControllerTransitioningDelegate {     private func _presentationController(         forPresented presented: UIViewController,         presenting: UIViewController?,         source: UIViewController     ) -> BottomSheetPresentationController {         BottomSheetPresentationController(             presentedViewController: presented,             presenting: presenting         )     } }<\/code><\/pre>\n<p>\u0418 presentation controller.<\/p>\n<pre><code class=\"swift\">public final class BottomSheetPresentationController: UIPresentationController {}<\/code><\/pre>\n<p>\u041d\u0430\u043a\u043e\u043d\u0435\u0446 \u0432\u0435\u0440\u043d\u0451\u043c\u0441\u044f \u0432 <em>RootViewController<\/em> \u0438 \u0437\u0430\u043a\u0440\u043e\u0435\u043c TODO.<\/p>\n<pre><code class=\"swift\">\/\/ TODO: bottomSheetTransitioningDelegate = ... bottomSheetTransitioningDelegate = BottomSheetTransitioningDelegate()<\/code><\/pre>\n<p>\u0414\u0430\u0432\u0430\u0439\u0442\u0435 \u0437\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<figure class=\"full-width\"><figcaption><\/figcaption><\/figure>\n<p>\u041a\u0430\u043a \u0431\u0443\u0434\u0442\u043e \u0441\u0442\u0430\u043b\u043e \u0442\u043e\u043b\u044c\u043a\u043e \u0445\u0443\u0436\u0435. View controller \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u0432\u043e \u0432\u0435\u0441\u044c \u044d\u043a\u0440\u0430\u043d \u0438 \u0441\u043a\u0440\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u0437\u0430 status bar. \u041c\u044b \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043b\u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 presentation controller, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u043b view controller \u043a\u0440\u0430\u0441\u0438\u0432\u043e, \u043f\u043e\u0437\u0438\u0446\u0438\u043e\u043d\u0438\u0440\u043e\u0432\u0430\u043b \u0435\u0433\u043e \u0441 \u0443\u0447\u0435\u0442\u043e\u043c safeArea. \u0412 \u043d\u0430\u0448\u0435\u043c presentation controller \u043d\u0438\u0447\u0435\u0433\u043e \u043f\u043e\u0434\u043e\u0431\u043d\u043e\u0433\u043e \u043d\u0435\u0442, \u043c\u044b \u043d\u0438\u043a\u0430\u043a \u043d\u0435 \u0443\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u0435 view controller, \u0434\u0430\u0432\u0430\u0439\u0442\u0435 \u0438\u0441\u043f\u0440\u0430\u0432\u0438\u043c\u0441\u044f.<\/p>\n<h3>\u0423\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u043c \u0440\u0430\u0437\u043c\u0435\u0440 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430<\/h3>\n<p>\u0412\u0435\u0440\u043d\u0451\u043c\u0441\u044f \u043a <em>ResizeViewController.<\/em> \u041f\u043e\u043b\u0435 <em>currentHeight<\/em> \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 \u0442\u0435\u043a\u0443\u0449\u0443\u044e \u0432\u044b\u0441\u043e\u0442\u0443. \u0427\u0442\u043e\u0431\u044b \u043d\u0435 \u0441\u043e\u0437\u0434\u0430\u0432\u0430\u0442\u044c \u043b\u0438\u0448\u043d\u0438\u0445 \u043f\u0440\u043e\u0442\u043e\u043a\u043e\u043b\u043e\u0432, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uiviewcontroller\/1621476-preferredcontentsize\">preferredContentSize<\/a>. \u041e\u043d \u0431\u0443\u0434\u0435\u0442 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0442\u044c \u0442\u0435\u043a\u0443\u0449\u0438\u0439 \u0436\u0435\u043b\u0430\u0435\u043c\u044b\u0439 \u0440\u0430\u0437\u043c\u0435\u0440 \u0434\u043b\u044f Bottom Sheet.<\/p>\n<p>\u0412 presentation controller \u043f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0438\u043c <em>frameOfPresentedViewInContainerView<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 \u043f\u043e\u043b\u043e\u0436\u0435\u043d\u0438\u0435 <em>presentedView<\/em>. \u0412 \u043d\u0430\u0448\u0435\u043c \u0441\u043b\u0443\u0447\u0430\u0435 <em>presentedView<\/em> \u2014 \u044d\u0442\u043e view <em>ResizeViewController<\/em>. <em>containerView<\/em> \u2014 \u044d\u0442\u043e view, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u0442 <em>presentedView<\/em> \u0438 \u043a\u0443\u0434\u0430 \u043c\u043e\u0436\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0442\u0435\u043d\u044c.<\/p>\n<pre><code class=\"swift\">public override var frameOfPresentedViewInContainerView: CGRect {     targetFrameForPresentedView() }  private func targetFrameForPresentedView() -> CGRect {     guard let containerView = containerView else {         return .zero     }      let windowInsets = presentedView?.window?.safeAreaInsets ?? .zero      let preferredHeight = presentedViewController.preferredContentSize.height + windowInsets.bottom     let maxHeight = containerView.bounds.height - windowInsets.top     let height = min(preferredHeight, maxHeight)      return .init(         x: 0,         y: (containerView.bounds.height - height).pixelCeiled,         width: containerView.bounds.width,         height: height.pixelCeiled     ) }<\/code><\/pre>\n<p>\u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0443\u043a\u0430\u0436\u0435\u043c <em>shouldPresentInFullscreen<\/em> \u0432 <em>false<\/em>, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e Bottom Sheet \u043f\u043e\u043a\u0440\u044b\u0432\u0430\u0435\u0442 \u043d\u0435 \u0432\u0435\u0441\u044c \u044d\u043a\u0440\u0430\u043d.<\/p>\n<pre><code>public override var shouldPresentInFullscreen: Bool {     false }<\/code><\/pre>\n<p>\u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c.<\/p>\n<figure class=\"full-width\"><figcaption><\/figcaption><\/figure>\n<p>\u0418\u0437\u043d\u0430\u0447\u0430\u043b\u044c\u043d\u044b\u0439 \u0440\u0430\u0437\u043c\u0435\u0440 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0435\u0442\u0441\u044f, \u043d\u043e \u043d\u0435\u0442 \u0440\u0435\u0430\u043a\u0446\u0438\u0438 \u043d\u0430 \u0435\u0433\u043e \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f.<\/p>\n<h3>\u0420\u0435\u0430\u0433\u0438\u0440\u0443\u0435\u043c \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u0435 \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430<\/h3>\n<p>\u0420\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c <em>UIPresentationController<\/em>. \u041e\u043d \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u0442 <a href=\"https:\/\/developer.apple.com\/documentation\/uikit\/uicontentcontainer\">UIContentContainer<\/a>, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u043d\u0430\u043c \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u0435\u043d <em>preferredContentSizeDidChange(forChildContentContainer:)<\/em>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u043f\u0440\u0438 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f\u0445 <em>preferredContentSize<\/em> \u0432 \u0434\u043e\u0447\u0435\u0440\u043d\u0438\u0445 view controller&#8217;\u0430\u0445.<\/p>\n<pre><code class=\"swift\">public override func preferredContentSizeDidChange(forChildContentContainer container: UIContentContainer) {     updatePresentedViewSize() }  private func updatePresentedViewSize() {     guard let presentedView = presentedView else {         return     }      let oldFrame = presentedView.frame     let targetFrame = targetFrameForPresentedView()     if !oldFrame.isAlmostEqual(to: targetFrame) {         presentedView.frame = targetFrame     } }<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u0442\u0435\u043a\u0443\u0449\u0438\u0439 frame \u0438 \u0442\u043e\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043c\u044b \u0441\u0447\u0438\u0442\u0430\u0435\u043c \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u044b\u043c. \u0415\u0441\u043b\u0438 \u043e\u043d\u0438 \u0440\u0430\u0437\u043d\u044b\u0435, \u0442\u043e \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0435\u043c <em>presentedView.frame.<\/em> \u0417\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.<\/p>\n<figure class=\"full-width\"><figcaption><\/figcaption><\/figure>\n<p>\u0420\u0430\u0437\u043c\u0435\u0440 \u0438\u0437\u043c\u0435\u043d\u044f\u0435\u0442\u0441\u044f \u043d\u0435\u0440\u0430\u0432\u043d\u043e\u043c\u0435\u0440\u043d\u043e \u0431\u0435\u0437 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u0438. \u041f\u043e\u0447\u0435\u043c\u0443? \u041f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043c\u044b \u043d\u0438\u043a\u0430\u043a \u043d\u0435 \u0443\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u043c \u044d\u0442\u0443 \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044e. \u0414\u043e\u0431\u0430\u0432\u0438\u043c \u0430\u043d\u0438\u043c\u0430\u0446\u0438\u044e \u043d\u0430 \u0438\u0437\u043c\u0435\u043d\u0435\u043d\u0438\u044f <em>preferredContentSize<\/em> \u0432 <em>ResizeViewController.<\/em><\/p>\n<pre><code class=\"swift\">UIView.animate(     withDuration: 0.25,     animations: { [self] in         preferredContentSize = CGSize(             width: UIScreen.main.bounds.width,             height: newValue         )     } )<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c.<\/p>\n<figure class=\"full-width\"><figcaption><\/figcaption><\/figure>\n<p>\u0420\u0430\u0431\u043e\u0442\u0430\u0435\u0442! \u041d\u043e \u043c\u044b \u043d\u0438\u043a\u0430\u043a \u043d\u0435 \u043c\u043e\u0436\u0435\u043c \u0443\u0439\u0442\u0438 \u0441 Bottom Sheet.<\/p>\n<h3>\u0417\u0430\u043a\u0440\u044b\u0432\u0430\u0435\u043c Bottom Sheet<\/h3>\n<p>\u0414\u043b\u044f \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f \u0434\u043e\u0431\u0430\u0432\u0438\u043c \u0442\u0435\u043d\u044c \u0438 \u043f\u043e \u043d\u0430\u0436\u0430\u0442\u0438\u044e \u0431\u0443\u0434\u0435\u043c \u0441\u043a\u0440\u044b\u0432\u0430\u0442\u044c Bottom Sheet. \u0421\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f, \u0447\u0435\u0440\u0435\u0437 \u043a\u043e\u0442\u043e\u0440\u044b\u0439 presentation controller \u0431\u0443\u0434\u0435\u0442 \u0441\u043e\u043e\u0431\u0449\u0430\u0442\u044c, \u0447\u0442\u043e \u0433\u043e\u0442\u043e\u0432, \u0447\u0442\u043e\u0431\u044b \u0435\u0433\u043e \u0437\u0430\u043a\u0440\u044b\u043b\u0438.<\/p>\n<pre><code class=\"swift\">public protocol BottomSheetModalDismissalHandler {     func performDismissal(animated: Bool) }<\/code><\/pre>\n<p>\u041f\u0435\u0440\u0435\u0434\u0430\u0451\u043c \u0435\u0433\u043e \u0432 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0430\u0442\u043e\u0440 presentation controller.<\/p>\n<pre><code class=\"swift\">private let dismissalHandler: BottomSheetModalDismissalHandler  public init(     presentedViewController: UIViewController,     presentingViewController: UIViewController?,     dismissalHandler: BottomSheetModalDismissalHandler ) {     self.dismissalHandler = dismissalHandler     super.init(presentedViewController: presentedViewController, presenting: presentingViewController) }<\/code><\/pre>\n<p>\u0414\u043b\u044f \u0443\u0434\u043e\u0431\u0441\u0442\u0432\u0430 \u0441\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u0444\u0430\u0431\u0440\u0438\u043a\u0443 presentation controller&#8217;\u0430.<\/p>\n<pre><code class=\"swift\">public protocol BottomSheetPresentationControllerFactory {     func makeBottomSheetPresentationController(         presentedViewController: UIViewController,         presentingViewController: UIViewController?     ) -> BottomSheetPresentationController }<\/code><\/pre>\n<p>\u041a\u043e\u0442\u043e\u0440\u0430\u044f \u0431\u0443\u0434\u0435\u0442 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c\u0441\u044f \u0432\u043d\u0443\u0442\u0440\u0438 <em>BottomSheetTransitioningDelegate.<\/em> <\/p>\n<pre><code class=\"swift\">private let factory: BottomSheetPresentationControllerFactory  public init(factory: BottomSheetPresentationControllerFactory) {     self.factory = factory }  public func presentationController(     forPresented presented: UIViewController,     presenting: UIViewController?,     source: UIViewController ) -> UIPresentationController? {     factory.makeBottomSheetPresentationController(         presentedViewController: presented,         presentingViewController: presenting     ) }<\/code><\/pre>\n<p>\u0412 <em>RootViewController<\/em> \u0440\u0435\u0430\u043b\u0438\u0437\u0443\u0435\u043c \u0444\u0430\u0431\u0440\u0438\u043a\u0443 \u0438 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u0437\u0430\u043a\u0440\u044b\u0442\u0438\u044f. \u0421\u043a\u0440\u044b\u0432\u0430\u0435\u043c <em>presentedViewController<\/em>,<\/p>\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-327331","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/327331","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=327331"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/327331\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=327331"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=327331"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=327331"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}