// ignore_for_file: depend_on_referenced_packages import 'dart:io'; import 'package:desktop_webview_window/desktop_webview_window.dart'; import 'package:flutter/material.dart'; import 'package:flutter_inappwebview/flutter_inappwebview.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_windows_webview/flutter_windows_webview.dart'; import 'package:mangayomi/providers/l10n_providers.dart'; import 'package:mangayomi/services/http/m_client.dart'; import 'package:mangayomi/utils/global_style.dart'; import 'package:url_launcher/url_launcher.dart'; import 'package:share_plus/share_plus.dart'; import 'package:path/path.dart' as p; import 'package:path_provider/path_provider.dart'; class MangaWebView extends ConsumerStatefulWidget { final String url; final String sourceId; final String title; const MangaWebView( {super.key, required this.url, required this.sourceId, required this.title}); @override ConsumerState createState() => _MangaWebViewState(); } class _MangaWebViewState extends ConsumerState { final GlobalKey webViewKey = GlobalKey(); double _progress = 0; @override void initState() { if (Platform.isWindows || Platform.isLinux || Platform.isMacOS) { _runWebViewDesktop(); } else { setState(() { _isNotDesktop = true; }); } super.initState(); } final _windowsWebview = FlutterWindowsWebview(); Webview? _desktopWebview; void _runWebViewDesktop() async { if (Platform.isLinux || Platform.isMacOS) { _desktopWebview = await WebviewWindow.create( configuration: CreateConfiguration( userDataFolderWindows: await getWebViewPath(), ), ); _desktopWebview! ..launch(widget.url) ..addOnWebMessageReceivedCallback((s) { if (s.substring(0, 2) == "UA") { MClient.setCookie(_url, s.replaceFirst("UA", "")); } }) ..addScriptToExecuteOnDocumentCreated( "window.chrome.webview.postMessage(\"UA\" + navigator.userAgent)") ..onClose.whenComplete(() async { if (Platform.isMacOS) { final cookieList = await _desktopWebview!.getCookies(widget.url); for (var c in cookieList) { final cookie = c.entries.map((e) => "${e.key}=${e.value}").join(";"); await MClient.setCookie(_url, "", cookie: cookie); } } if (mounted) { Navigator.pop(context); } }); } //credit: https://github.com/wgh136/PicaComic/blob/master/lib/network/nhentai_network/cloudflare.dart else if (Platform.isWindows && await FlutterWindowsWebview.isAvailable()) { _windowsWebview.launchWebview( widget.url, WebviewOptions(messageReceiver: (s) { if (s.substring(0, 2) == "UA") { MClient.setCookie(_url, s.replaceFirst("UA", "")); } }, onTitleChange: (_) { _windowsWebview.runScript( "window.chrome.webview.postMessage(\"UA\" + navigator.userAgent)"); _windowsWebview.getCookies(widget.url).then((cookies) { final cookie = cookies.entries.map((e) => "${e.key}=${e.value}").join("; "); MClient.setCookie(_url, "", cookie: cookie); }); })); } } bool _isNotDesktop = false; InAppWebViewController? _webViewController; late String _url = widget.url; late String _title = widget.title; bool _canGoback = false; bool _canGoForward = false; @override Widget build(BuildContext context) { final l10n = l10nLocalizations(context); return !_isNotDesktop ? Scaffold( appBar: AppBar( title: Text( _title, style: const TextStyle( overflow: TextOverflow.ellipsis, fontWeight: FontWeight.bold), ), leading: IconButton( onPressed: () { if (Platform.isLinux || Platform.isMacOS) { _desktopWebview!.close(); } Navigator.pop(context); }, icon: const Icon(Icons.close)), ), ) : SafeArea( child: WillPopScope( onWillPop: () async { _webViewController?.goBack(); return false; }, child: Column( children: [ SizedBox( height: AppBar().preferredSize.height, child: Row( children: [ Expanded( child: ListTile( dense: true, subtitle: Text( _url, style: const TextStyle( fontSize: 10, overflow: TextOverflow.ellipsis), ), title: Text( _title, style: const TextStyle( overflow: TextOverflow.ellipsis, fontWeight: FontWeight.bold), ), leading: IconButton( onPressed: () { Navigator.pop(context); }, icon: const Icon(Icons.close)), ), ), IconButton( icon: Icon(Icons.arrow_back, color: _canGoback ? null : Colors.grey), onPressed: _canGoback ? () { _webViewController?.goBack(); } : null, ), IconButton( icon: Icon(Icons.arrow_forward, color: _canGoForward ? null : Colors.grey), onPressed: _canGoForward ? () { _webViewController?.goForward(); } : null, ), PopupMenuButton( popUpAnimationStyle: popupAnimationStyle, itemBuilder: (context) { return [ PopupMenuItem( value: 0, child: Text(l10n!.refresh)), PopupMenuItem( value: 1, child: Text(l10n.share)), PopupMenuItem( value: 2, child: Text(l10n.open_in_browser)), PopupMenuItem( value: 3, child: Text(l10n.clear_cookie)), ]; }, onSelected: (value) async { if (value == 0) { _webViewController?.reload(); } else if (value == 1) { Share.share(_url); } else if (value == 2) { await InAppBrowser.openWithSystemBrowser( url: Uri.parse(_url)); } else if (value == 3) { CookieManager.instance().deleteAllCookies(); } }), ], ), ), _progress < 1.0 ? LinearProgressIndicator(value: _progress) : Container(), Expanded( child: InAppWebView( key: webViewKey, onWebViewCreated: (controller) async { _webViewController = controller; }, onLoadStart: (controller, url) async { setState(() { _url = url.toString(); }); }, shouldOverrideUrlLoading: (controller, navigationAction) async { var uri = navigationAction.request.url!; if (![ "http", "https", "file", "chrome", "data", "javascript", "about" ].contains(uri.scheme)) { if (await canLaunchUrl(uri)) { // Launch the App await launchUrl( uri, ); // and cancel the request return NavigationActionPolicy.CANCEL; } } return NavigationActionPolicy.ALLOW; }, onLoadStop: (controller, url) async { if (mounted) { setState(() { _url = url.toString(); }); } }, onProgressChanged: (controller, progress) async { if (mounted) { setState(() { _progress = progress / 100; }); } }, onUpdateVisitedHistory: (controller, url, isReload) async { final ua = await controller.evaluateJavascript( source: "navigator.userAgent") ?? ""; await MClient.setCookie(url.toString(), ua); final canGoback = await controller.canGoBack(); final canGoForward = await controller.canGoForward(); final title = await controller.getTitle(); if (mounted) { setState(() { _url = url.toString(); _title = title!; _canGoback = canGoback; _canGoForward = canGoForward; }); } }, initialUrlRequest: URLRequest(url: Uri.parse(widget.url)), ), ), ], ), ), ); } } Future getWebViewPath() async { final document = await getApplicationDocumentsDirectory(); return p.join( document.path, 'desktop_webview_window', ); }