Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
ブラウザアプリを自作してわかったWebViewの扱い方/iOS Meetup in 福岡
Search
Kyome (Takuto Nakamura)
March 24, 2023
Programming
4.6k
1
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
ブラウザアプリを自作してわかったWebViewの扱い方/iOS Meetup in 福岡
Kyome (Takuto Nakamura)
March 24, 2023
More Decks by Kyome (Takuto Nakamura)
See All by Kyome (Takuto Nakamura)
Swiftで高速フーリエ変換してオーディオビジュアライザーを作る / iOSDC Japan 2024 Day1 Track D
kyome22
3
1.5k
Accelerate.vDSPとSwift Chartsでぶち上がろう!/サイボウズモバイル Meetup 2023.04.20
kyome22
0
290
iOSのウィジェットでも猫走らせたい / iOSDC Japan 2022 Day2 Unconference
kyome22
3
1.4k
Hacking Xcode Behaviors / macOS native symposium #08
kyome22
2
1.9k
AppKitでお絵描きをしてみよう / macOS native symposium #06
kyome22
2
1.1k
Finder Sync Extension で Mac 向け便利ツールを作ろう / iOSDC Japan 2021
kyome22
6
6.2k
iOS Custom Keyboardsでできること/できないこと/やってはいけないこと / iOSDC Japan 2020 LT
kyome22
3
2.1k
Other Decks in Programming
See All in Programming
「エンジニアインターン、どうやって取った?」準備のリアルを語るLT会 Progate BAR
akiomatic
0
120
過去最大のMCPアップデート! 2026-07-28 RC版の謎に迫る
licux
6
180
決定論的オーケストレーションの設計と実装 / Design and Implementation of Deterministic Orchestration
nrslib
3
1.2k
Oxcを導入して開発体験が向上した話
yug1224
4
300
ADKを使って簡単にAIエージェントを作ってみよう
k1mu21
0
240
Make SRE Operations Easier with Azure SRE Agent
kkamegawa
0
4.8k
net-httpのHTTP/2対応について
naruse
0
460
軽量Java基盤の設計 DIコンテナに頼らない、長期保守と1秒起動の実現 JJUG CCC 2026 Spring
macha64
0
480
These Five Tricks Can Make Your Apps Greener, Cheaper, & Nicer
hollycummins
0
280
脅威をエンジニアリングの糧にして――現場編 / Turning Threats into Engineering Fuel — Field Edition
nrslib
0
260
Old Dog, New Tricks: The Java 25 Reinvention - JNation
bazlur_rahman
0
150
柔軟なPDFレイアウトエディタを支える型システム設計 — Discriminated UnionとConditional Typeの実践
minako__ph
4
1.6k
Featured
See All Featured
The World Runs on Bad Software
bkeepers
PRO
72
12k
Measuring Dark Social's Impact On Conversion and Attribution
stephenakadiri
2
210
Code Review Best Practice
trishagee
74
20k
Efficient Content Optimization with Google Search Console & Apps Script
katarinadahlin
PRO
1
600
Faster Mobile Websites
deanohume
310
31k
Making the Leap to Tech Lead
cromwellryan
135
9.9k
Exploring the relationship between traditional SERPs and Gen AI search
raygrieselhuber
PRO
2
4k
Amusing Abliteration
ianozsvald
1
200
No one is an island. Learnings from fostering a developers community.
thoeni
21
3.7k
Helping Users Find Their Own Way: Creating Modern Search Experiences
danielanewman
31
3.2k
Designing Dashboards & Data Visualisations in Web Apps
destraynor
231
55k
New Earth Scene 8
popppiees
3
2.3k
Transcript
Kyome 2023/03/24 (ۚ) @LINE FukuokaΦϑΟε ϒϥβΞϓϦΛࣗ࡞ͯ͠Θ͔ͬͨ WebViewͷѻ͍ํ J04.FFUVQ JOԬ
ࣗݾհ IUUQTLZPNFJP ,ZPNF ,ZPNFTVLF CybozuͰkintoneϞόΠϧͷiOSΛ୲ ۀޮԽOSSϥΠϒϥϦͷ։ൃӡ༻ʹੵۃత झຯͰmacOS͚ͷϢʔςΟϦςΟΞϓϦ։ൃ
ࣗݾհ RunCat activity indicator HandyPalette color picker ScreenPointer presentation tool
QuickMIDI MIDI keyboard player
ͳͥϒϥβΞϓϦΛ ࣗ࡞͠Α͏ͱࢥ͔ͬͨ
ͳͥϒϥβΞϓϦΛࣗ࡞͠Α͏ͱࢥ͔ͬͨ kintoneʹʮJavaScriptΧελϚΠζʯͱ͍͏ϢʔβʔಠࣗͷػೳΛ ՃͰ͖ΔΈ͕͋Δ kintoneϞόΠϧWebViewϕʔεͷΞϓϦͱͳ͍ͬͯΔ WebViewϕʔεͷΞϓϦ։ൃΛ୲͢Δʹ͋ͨͬͯɺWebViewΛѻͬͨ ͜ͱ͕ͳ͔ͬͨͨΊɺϒϥβΞϓϦΛؙ͝ͱࣗ࡞͢Δ͜ͱʹͨ͠ MinBrowser https://github.com/kyome22/MinBrowserɹ AppStoreͰ৴த
ͳͥϒϥβΞϓϦΛࣗ࡞͠Α͏ͱࢥ͔ͬͨ ϒϥβΞϓϦΛؙ͝ͱࣗ࡞ֶͯ͠Μͩ͜ͱ WebViewͰͷΠϕϯτΛϑοΫͯ͠ωΠςΟϒಈ࡞Λͭͳ͛Δํ๏ WebView্Ͱ࣮ߦ͞ΕͨJavaScriptͷϩάΛरͬͯग़ྗ͢Δํ๏ WebViewΛབྷΊͨςετέʔεΛ҆ఆ࣮ͯ͠ߦ͢Δํ๏ etc. ࠓճͷͷൣғ֎
SwiftUIͰWKWebViewΛѻ͏
UIViewRepresentableͰWKWebViewΛϥοϓͯ͠Viewͱͯ͠͏ ݕࡧɺΔɺਐΉɺPull to refreshͳͲ͕ՄೳͳΠϯλϥΫςΟϒͳ WebViewΛ࡞Γ͍ͨ UIViewRepresentableͷ֎ͷViewͱͷ࿈ܞ͕ඞཁ SwiftUIͰWKWebViewΛѻ͏ UIViewRepresentable WKWebView SearchBar
ProgressView ϫʔυݕࡧ ϓϩάϨεͷߋ৽
GitHubΛړ͍ͬͯΔͱΑ͘ݟΔख๏ enumͷϓϩύςΟΛࢹͯ͠WKWebViewͷΞΫγϣϯΛ੍ޚ͢Δ UIViewRepresentableͷCoordinatorͰWKWebViewͷঢ়ଶΛࢹ͢Δ SwiftUIͰWKWebViewΛѻ͏
GitHubΛړ͍ͬͯΔͱΑ͘ݟΔख๏ enumͷϓϩύςΟΛࢹͯ͠WKWebViewͷΞΫγϣϯΛ੍ޚ͢Δ UIViewRepresentableͷCoordinatorͰWKWebViewͷঢ়ଶΛࢹ͢Δ SwiftUIͰWKWebViewΛѻ͏ struct WrappedWKWebView: UIViewRepresentable { enum Action
{ case none case search(String) } @Binding private var action: Action ϫʔυݕࡧ༻ͷaction
GitHubΛړ͍ͬͯΔͱΑ͘ݟΔख๏ enumͷϓϩύςΟΛࢹͯ͠WKWebViewͷΞΫγϣϯΛ੍ޚ͢Δ UIViewRepresentableͷCoordinatorͰWKWebViewͷঢ়ଶΛࢹ͢Δ SwiftUIͰWKWebViewΛѻ͏ func updateUIView(_ webView: WKWebView, context: Context)
{ switch action { case .none: break case .search(let searchText): context.coordinator.search(text: searchText) action = .none } } action͕searchʹͳͬͨΒcoordinatorΛհͯ͠ WKWebViewͷload(request:)Λୟ͘ actionΛ͍ऴΘͬͨΒ.noneʹ͢
GitHubΛړ͍ͬͯΔͱΑ͘ݟΔख๏ enumͷϓϩύςΟΛࢹͯ͠WKWebViewͷΞΫγϣϯΛ੍ޚ͢Δ UIViewRepresentableͷCoordinatorͰWKWebViewͷঢ়ଶΛࢹ͢Δ SwiftUIͰWKWebViewΛѻ͏ struct WrappedWKWebView: UIViewRepresentable { @Binding private
var progress: Double private let webView = WKWebView() func makeCoordinator() -> Coordinator { return Coordinator(webView: webView, progress: $progress) } coordinatorʹwebViewͱBinding͍ͨ͠ϓϩύςΟΛ͢
GitHubΛړ͍ͬͯΔͱΑ͘ݟΔख๏ enumͷϓϩύςΟΛࢹͯ͠WKWebViewͷΞΫγϣϯΛ੍ޚ͢Δ UIViewRepresentableͷCoordinatorͰWKWebViewͷঢ়ଶΛࢹ͢Δ SwiftUIͰWKWebViewΛѻ͏ class Coordinator { init(webView: WKWebView, progress:
Binding<Double>) { // ॳظԽলུ self.webView .publisher(for: \.estimatedProgress) .sink { [weak self] value in self?.progress = value } .store(in: &cancellables) } } WKWebViewͷঢ়ଶΛࢹͯ͠Bindingʹྲྀ͢
SwiftUIͰWKWebViewΛѻ͏ ൃੜ ϫʔυݕࡧͨ͠ޙϩʔυॲཧ͕ऴΘΒͣɺ Կϩʔυ͞ΕΔΑ͏ͳڍಈʹͳͬͨ ͔͠XcodeͰେྔͷܯࠂ͕ʂ 🤯 ѹతڍಈෆ৹
⚠ ܯࠂʹ͋Δ௨ΓɺViewͷߋ৽தʹBinding͍ͯ͠Δͷߋ৽Λ͍ͯ͠Δ͜ͱ WebView͕ViewͰ͋Δͷʹঢ়ଶΛ͓࣋ͬͯΓɺ View֎෦͔Βঢ়ଶΛมߋͨ͠Γࢹͨ͠Γ͢Δඞཁ͕͋Δ͜ͱ͕ݪҼ WebViewએݴతUIͱͷ૬ੑ͕ѱ͘ɺೃછΈͮΒ͍ SwiftUIͰWKWebViewΛѻ͏
💡ղܾࡦ UIViewRepresentableͷ֎ʹWKWebViewͷΠϯελϯεΛ࣋ͨͤɺ ͦͪΒͰঢ়ଶͷࢹͱ੍ޚΛߦ͏ SwiftUIͰWKWebViewΛѻ͏ UIViewRepresentable WKWebView SearchBar ProgressView ϫʔυݕࡧ ϓϩάϨεͷߋ৽
WebViewModel ϨϯμϦϯάػೳ͚ͩ WKWebView ঢ়ଶͷࢹͱ੍ޚ
💡ղܾࡦ UIViewRepresentableͷ֎ʹWKWebViewͷΠϯελϯεΛ࣋ͨͤɺ ͦͪΒͰঢ়ଶͷࢹͱ੍ޚΛߦ͏ SwiftUIͰWKWebViewΛѻ͏ struct WrappedWKWebView: UIViewRepresentable { let setWebViewHandler:
(WKWebView) -> Void func makeUIView(context: Context) -> some WKWebView { let webView = WKWebView() setWebViewHandler(webView) return webView } func updateUIView(_ uiView: UIViewType, context: Context) {} } WKWebViewΛ֎ʹ࣋ͪग़ͭ͢ Կ͠ͳ͍
💡ղܾࡦ UIViewRepresentableͷ֎ʹWKWebViewͷΠϯελϯεΛ࣋ͨͤɺ ͦͪΒͰঢ়ଶͷࢹͱ੍ޚΛߦ͏ SwiftUIͰWKWebViewΛѻ͏ class WebViewModel: ObservableObject { private weak
var webView: WKWebView? func setWebView(_ webView: WKWebView) { self.webView = webView // webViewͷঢ়ଶΛࢹ(ߪಡ)ͯ͠PublishedͳϓϩύςΟʹྲྀ͢ } func search() { // URLRequestΛ࡞ͬͯ webView?.load() Λୟ͘ } }
💡ղܾࡦ UIViewRepresentableͷ֎ʹWKWebViewͷΠϯελϯεΛ࣋ͨͤɺ ͦͪΒͰঢ়ଶͷࢹͱ੍ޚΛߦ͏ SwiftUIͰWKWebViewΛѻ͏ struct WebView: View { @StateObject private
var webViewModel = WebViewModel() var body: some View { VStack(spacing: 0) { SearchBar(inputText: $webViewModel.inputText) .onSubmit { webViewModel.search() } ProgressView(value: webViewModel.progress) WrappedWKWebView(setWebViewHandler: { webView in webViewModel.setWebView(webView) }) } } } Modelͷsearch()Λୟ͘ WKWebViewͷΠϯελϯεΛModelʹ͢
💡ղܾࡦ UIViewRepresentableͷ֎ʹWKWebViewͷΠϯελϯεΛ࣋ͨͤɺ ͦͪΒͰঢ়ଶͷࢹͱ੍ޚΛߦ͏ SwiftUIͰWKWebViewΛѻ͏ struct WebView: View { @StateObject private
var webViewModel = WebViewModel() var body: some View { VStack(spacing: 0) { SearchBar(inputText: $webViewModel.inputText) .onSubmit { webViewModel.search() } ProgressView(value: webViewModel.progress) WrappedWKWebView(setWebViewHandler: { webView in webViewModel.setWebView(webView) }) } } } Modelͷsearch()Λୟ͘ WKWebViewͷΠϯελϯεΛModelʹ͢ Q, Կݺͼग़͞Εͯ͠·͏ͷͰʁ A, UIViewRepresentableͷmakeUIView() Ұ͔͠ݺΕͳ͍ͷͰͳ͍
💡ղܾࡦ UIViewRepresentableͷ֎ʹWKWebViewͷΠϯελϯεΛ࣋ͨͤɺ ͦͪΒͰঢ়ଶͷࢹͱ੍ޚΛߦ͏ SwiftUIͰWKWebViewΛѻ͏ struct WebView: View { @StateObject private
var webViewModel = WebViewModel() var body: some View { VStack(spacing: 0) { SearchBar(inputText: $webViewModel.inputText) .onSubmit { webViewModel.search() } ProgressView(value: webViewModel.progress) WrappedWKWebView(setWebViewHandler: { webView in webViewModel.setWebView(webView) }) } } } Modelͷsearch()Λୟ͘ WKWebViewͷΠϯελϯεΛModelʹ͢ Q, Կݺͼग़͞Εͯ͠·͏ͷͰʁ A, UIViewRepresentableͷmakeUIView() Ұ͔͠ݺΕͳ͍ͷͰͳ͍
Thank you! WKWebViewʹ͍ͭͯ MinBrowserͷϦϦʔεʹۤઓͨ݅͠ ݸਓΞϓϦ։ൃʹ͍ͭͯ ࠙ձͰ͓͠͠·͠ΐ͏ʂ
খ
MinBrowserͷϦϦʔεʹۤઓͨ݅͠
ग़དྷ্͕ͬͨͷͰϦϦʔε͠Α͏ͱࢥͬͨΒɺ৹ࠪʹ6ϲ݄͔͔ͬͨ😵 ʢ2022/09/04 ৹ࠪఏग़ɺ2023/03/02 ৴։࢝ʣ WebϒϥβͳͷͰʮແ੍ݶͷWebΞΫηεؚ͕·ΕΔʯʹ ☑ ͕ඞཁ ྗతͳදݱɺΪϟϯϒϧɺաܹͳੑత༰ͳͲશͯͷྸ੍ݶ߲ ʹ͍ͭͯʮසൟ/ۃʯʹ ☑
͕ඞཁͩͱࢥ͍ࠐΜͰ͍ͨ ݁Ռతʹ৹ࠪʹ͔͔࣌ؒΔϧʔτʹೖͬͯ͠·ͬͨ MinBrowserͷϦϦʔεʹۤઓͨ͠
৹ࠪʹ͔͔࣌ؒΔ߹ɺεςʔλεRejectʹͳΓ ʮ͔͔࣌ؒΔ͚Ͳ٫ԼͰͳ͍͔ΒͬͯͯͶʯͱϝοηʔδ͕ಧ͘ ΞϓϦࣗମ͕ྸ੍ݶ߲ΛҙਤతʹؚΜͰ͍ͳ͚Ε ☑ ෆཁͩͬͨ ʮແ੍ݶͷWebΞΫηεؚ͕·ΕΔʯ͚ͩʹ ☑ ͨ͠ΒͰ৹ࠪ௨ͬͨ MinBrowserͷϦϦʔεʹۤઓͨ͠ ͱ΄΄…