iOS用WKWebView与JS交互获取系统图片及WKWebView的Alert,Confirm,TextInput的监听代理方法使用,屏蔽WebView的可选菜单

最近做一个项目,开始是使用WebView与JS交互的,由于内存管理方面WebView欠佳。WKWebVIew的内存线程管理好,所以选择使用 WKWebVIew(使用WKWebView 的缺点在于,这个控件加载的H5页面不支持ajax请求,所以需要自己把网络请求在OC上实现)。

一、首先说下应该注意的问题:

1.要获取拍照或相册的图片,如果是iOS 10系统,需要设置访问权限(在 Info-plist 中设置)

相机权限: Privacy - Camera Usage Description 是否允许此App使用你的相机?
相册权限: Privacy - Photo Library Usage Description 是否允许此App访问你的媒体资料库?

2.WebView和WKWebView和JS互调的方法和使用的传参类型不同

WebView 使用 (window.iosModel.getImage(JSON.stringify(parameter)); //JSON.stringify(参数字符串) 这个方法是 把字符串转换成json字符串 parameter是参数字符串
   )传值给 OC

WKWebView 使用 (window.webkit.messageHandlers.iosModel.postMessage(parameter))

3.需要特别注意的是:WKWebView 不执行JS写的 ajax请求(WKWebView 可能由于基于 WebKit,并不会执行 C socket 相关的函数对 HTTP 请求进行处理)如果有网络请求,需要自己用OC实现

4.使用的时候不要忘记挂上使用到的代理,和导入代理

@interface ViewController () <WKScriptMessageHandler, WKNavigationDelegate, WKUIDelegate,UINavigationControllerDelegate,UIImagePickerControllerDelegate>

5.屏蔽WebView的可选菜单(即不会出现拷贝、全选等弹出菜单)在加载完成后的代理中执行以下两段JS

// 导航完成时,会回调(也就是页面载入完成了)
- (void)webView:(WKWebView *)webView didFinishNavigation:(null_unspecified WKNavigation *)navigation {
    NSLog(@"66===%s", __FUNCTION__);
    // 禁用选中效果
    [self.webView evaluateJavaScript:@"document.documentElement.style.webkitUserSelect='none'" completionHandler:nil];
    [self.webView evaluateJavaScript:@"document.documentElement.style.webkitTouchCallout='none'" completionHandler:nil];
}




二、代码:

<!DOCTYPE html>
<html>
	<head>
		<meta charset="UTF-8">
		<title></title>
	</head>
	<body>
		<div>
            <h3>JS与iOS交互</h3>
            <h4>JS页面获取iOS系统图片</h5>
        </div>
		<div>
			<input type = "button" style="width: 50%;height: 5%;" id="Button" value="打开相机获取图片" onclick="getIOSImage()"></button>
		</div><dir />
		<div>
			<img src="testImage.png" id="changeImage"style="width: 30%; height: 30%;" onclick="getIOSImage()"><!--src="图片的相对路径" 如果把html文件导入工程中,图片路径和OC一样只写图片名字和后缀就可以,(记得要先把图片添加到工程) 图片也可以实现按钮的方法getIOSImage -->
		</div>
        <span id="iosParame" style="width: 200px; height: 50%; color:orangered; font-size:15px" value="等待获取ios参数" >
            </div>
		<script>
			var getIOSImage = function(){
				var parameter = {'title':'JS调OC','describe':'这里就是JS传给OC的参数'};
				// 在下面这里实现js 调用系统原生api iosDelegate
                //JSON.stringify(参数字符串) 这个方法是 把字符串转换成json字符串
	window.iosDelegate.getImage(JSON.stringify(parameter));// 实现数据的 json 格式字符串
			}
        // 这里是 iOS调用js的方法
        function setImageWithPath(arguments){
            document.getElementById('changeImage').src = arguments['imagePath'];
            document.getElementById('iosParame').innerHTML = arguments['iosContent'];
        }
		</script>
	</body>
</html>

<pre name="code" class="objc">#import "ViewController.h"
#import <JavaScriptCore/JavaScriptCore.h>
#import <WebKit/WebKit.h>
#import "SaveImage_Util.h"

@interface ViewController () <WKScriptMessageHandler, WKNavigationDelegate, WKUIDelegate,UINavigationControllerDelegate,UIImagePickerControllerDelegate>

@property (nonatomic, strong) WKWebView *webView;
@property (nonatomic, strong) UIProgressView *progressView;

@end

@implementation ViewController
{
    int indextNumb;// 交替图片名字
    UIImage *getImage;//获取的图片
}
- (void)viewDidLoad {
  [super viewDidLoad];
  
  self.edgesForExtendedLayout = UIRectEdgeNone;
  self.automaticallyAdjustsScrollViewInsets = NO;
  
  WKWebViewConfiguration *config = [[WKWebViewConfiguration alloc] init];
  // 设置偏好设置
  config.preferences = [[WKPreferences alloc] init];
  // 默认为0
  config.preferences.minimumFontSize = 10;
  // 默认认为YES
  config.preferences.javaScriptEnabled = YES;
  // 在iOS上默认为NO,表示不能自动通过窗口打开
  config.preferences.javaScriptCanOpenWindowsAutomatically = NO;


  // web内容处理池
  config.processPool = [[WKProcessPool alloc] init];

  // 通过JS与webview内容交互
  config.userContentController = [[WKUserContentController alloc] init];
  // 注入JS对象名称AppModel,当JS通过AppModel来调用时,
  // 我们可以在WKScriptMessageHandler代理中接收到
  [config.userContentController addScriptMessageHandler:self name:@"iosModel"];



        //通过默认的构造器来创建对象
  self.webView = [[WKWebView alloc] initWithFrame:self.view.bounds configuration:config];
    // 导航代理
    self.webView.navigationDelegate = self;
    // 与webview UI交互代理
    self.webView.UIDelegate = self;
  NSURL *path = [[NSBundle mainBundle] URLForResource:@"testJS" withExtension:@"html"];
  [self.webView loadRequest:[NSURLRequest requestWithURL:path]];
  [self.view addSubview:self.webView];
  
  

  // 添加KVO监听
  [self.webView addObserver:self
                 forKeyPath:@"loading"
                    options:NSKeyValueObservingOptionNew
                    context:nil];
  [self.webView addObserver:self
                 forKeyPath:@"title"
                    options:NSKeyValueObservingOptionNew
                    context:nil];
  [self.webView addObserver:self
                 forKeyPath:@"estimatedProgress"
                    options:NSKeyValueObservingOptionNew
                    context:nil];

  // 添加进入条
  self.progressView = [[UIProgressView alloc] init];
  self.progressView.frame = self.view.bounds;
  [self.view addSubview:self.progressView];
  self.progressView.backgroundColor = [UIColor blackColor];
  
  self.navigationItem.leftBarButtonItem = [[UIBarButtonItem alloc] initWithTitle:@"后退" style:UIBarButtonItemStyleDone target:self action:@selector(goback)];
  self.navigationItem.rightBarButtonItem = [[UIBarButtonItem alloc] initWithTitle:@"前进" style:UIBarButtonItemStyleDone target:self action:@selector(gofarward)];
}

- (void)goback {
  if ([self.webView canGoBack]) {
    [self.webView goBack];
  }
}

- (void)gofarward {
  if ([self.webView canGoForward]) {
    [self.webView goForward];
  }
}

#pragma mark - WKScriptMessageHandler
// 通过这个方法获取 JS传来的json字符串
- (void)userContentController:(WKUserContentController *)userContentController
      didReceiveScriptMessage:(WKScriptMessage *)message
{
  if ([message.name isEqualToString:@"iosModel"]) {
    // 打印所传过来的参数,只支持NSNumber, NSString, NSDate, NSArray,
    // NSDictionary, and NSNull类型
      NSLog(@"JS传来的json字符串 :  %@", message.body);
      NSDictionary *jsDictionary = message.body;
      if ([jsDictionary[@"means"] isEqualToString:@"获取系统图片"])
      {
          [self beginOpenPhoto];
      }
  }
}

#pragma mark - KVO
- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary<NSString *,id> *)change
                       context:(void *)context {
  if ([keyPath isEqualToString:@"loading"]) {
    NSLog(@"loading");
  } else if ([keyPath isEqualToString:@"title"]) {
    self.title = self.webView.title;
  } else if ([keyPath isEqualToString:@"estimatedProgress"]) {
    NSLog(@"progress: %f", self.webView.estimatedProgress);
    self.progressView.progress = self.webView.estimatedProgress;
  }
  if (!self.webView.loading) {
    [UIView animateWithDuration:0.5 animations:^{
      self.progressView.alpha = 0;
    }];
  }
}

#pragma mark - WKNavigationDelegate
    // 请求开始前,会先调用此代理方法
    // 与UIWebView的
    // - (BOOL)webView:(UIWebView *)webView
    // shouldStartLoadWithRequest:(NSURLRequest *)request
    // navigationType:(UIWebViewNavigationType)navigationType;
    // 类型,在请求先判断能不能跳转(请求)

- (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:
(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler {
  NSString *hostname = navigationAction.request.URL.host.lowercaseString;
  if (navigationAction.navigationType == WKNavigationTypeLinkActivated
      && ![hostname containsString:@".lanou.com"]) {
// 对于跨域,需要手动跳转, 用系统浏览器(Safari)打开
    [[UIApplication sharedApplication] openURL:navigationAction.request.URL];
    
    // 不允许web内跳转
    decisionHandler(WKNavigationActionPolicyCancel);
  } else {
    self.progressView.alpha = 1.0;
    decisionHandler(WKNavigationActionPolicyAllow);
  }
  
}


    // 在响应完成时,会回调此方法
    // 如果设置为不允许响应,web内容就不会传过来
- (void)webView:(WKWebView *)webView
decidePolicyForNavigationResponse:(WKNavigationResponse *)navigationResponse
decisionHandler:(void (^)(WKNavigationResponsePolicy))decisionHandler {
  decisionHandler(WKNavigationResponsePolicyAllow);
  
}


// 开始导航跳转时会回调
- (void)webView:(WKWebView *)webView didStartProvisionalNavigation:(null_unspecified WKNavigation *)navigation {
   
}

    // 接收到重定向时会回调
- (void)webView:(WKWebView *)webView
didReceiveServerRedirectForProvisionalNavigation:(null_unspecified WKNavigation *)navigation {
   
}


    // 导航失败时会回调
- (void)webView:(WKWebView *)webView
didFailProvisionalNavigation:(null_unspecified WKNavigation *)navigation withError:(NSError *)error {
    
}


// 页面内容到达main frame时回调
- (void)webView:(WKWebView *)webView didCommitNavigation:(null_unspecified WKNavigation *)navigation {
    
}

// 导航完成时,会回调(也就是页面载入完成了)
- (void)webView:(WKWebView *)webView didFinishNavigation:(null_unspecified WKNavigation *)navigation {
    NSLog(@"66===%s", __FUNCTION__);
    // 禁用选中效果
    [self.webView evaluateJavaScript:@"document.documentElement.style.webkitUserSelect='none'" completionHandler:nil];
    [self.webView evaluateJavaScript:@"document.documentElement.style.webkitTouchCallout='none'" completionHandler:nil];
}
    // 导航失败时会回调
- (void)webView:(WKWebView *)webView didFailNavigation:
(null_unspecified WKNavigation *)navigation withError:(NSError *)error
{
    
}


/* 对于HTTPS的都会触发此代理,如果不要求验证,传默认就行
    如果需要证书验证,与使用AFN进行HTTPS证书验证是一样的 */

- (void)webView:(WKWebView *)webView didReceiveAuthenticationChallenge:
(NSURLAuthenticationChallenge *)challenge completionHandler:
(void (^)(NSURLSessionAuthChallengeDisposition disposition,
          NSURLCredential *__nullable credential))completionHandler
{
  completionHandler(NSURLSessionAuthChallengePerformDefaultHandling, nil);
}


// 9.0才能使用,web内容处理中断时会触发
/*
- (void)webViewWebContentProcessDidTerminate:(WKWebView *)webView {
}
*/
#pragma mark - WKUIDelegate
- (void)webViewDidClose:(WKWebView *)webView {
    
}
/* 在JS端调用alert函数时,会触发此代理方法。JS端调用alert时所传的数据可以通过message拿到 在原生得到结果后,需要回调JS,是通过completionHandler回调 */
- (void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message
initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(void))completionHandler
{
  UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"alert" message:message preferredStyle:UIAlertControllerStyleAlert];
  [alert addAction:[UIAlertAction actionWithTitle:@"确定" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
    completionHandler();
  }]];
  
  [self presentViewController:alert animated:YES completion:NULL];
  NSLog(@"%@", message);
 
}

    // JS端调用confirm函数时,会触发此方法
    // 通过message可以拿到JS端所传的数据
    // 在iOS端显示原生alert得到YES/NO后
    // 通过completionHandler回调给JS端
- (void)webView:(WKWebView *)webView
runJavaScriptConfirmPanelWithMessage:(NSString *)message
initiatedByFrame:(WKFrameInfo *)frame
completionHandler:(void (^)(BOOL result))completionHandler {
  UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"confirm" message:message preferredStyle:UIAlertControllerStyleAlert];
  [alert addAction:[UIAlertAction actionWithTitle:@"确定"
    style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action){
    completionHandler(YES);
  }]];
  [alert addAction:[UIAlertAction actionWithTitle:@"取消"
  style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action)
    {
    completionHandler(NO);
  }]];
  [self presentViewController:alert animated:YES completion:NULL];
  NSLog(@"%@", message);
}
 // JS端调用prompt函数时,会触发此方法
    // 要求输入一段文本
    // 在原生输入得到文本内容后,通过completionHandler回调给JS
- (void)webView:(WKWebView *)webView
runJavaScriptTextInputPanelWithPrompt:(NSString *)prompt
    defaultText:(nullable NSString *)defaultText
initiatedByFrame:(WKFrameInfo *)frame
completionHandler:(void (^)(NSString * __nullable result))completionHandler
{
  UIAlertController *alert = [UIAlertController alertControllerWithTitle:prompt message:defaultText preferredStyle:UIAlertControllerStyleAlert];
  [alert addTextFieldWithConfigurationHandler:^(UITextField * _Nonnull textField) {
    textField.textColor = [UIColor redColor];
  }];
  [alert addAction:[UIAlertAction actionWithTitle:@"确定"
    style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
    completionHandler([[alert.textFields lastObject] text]);
  }]];
  
  [self presentViewController:alert animated:YES completion:NULL];
}

//  获取图片
- (void)beginOpenPhoto
{
    // 主队列 异步打开相机
    dispatch_async(dispatch_get_main_queue(), ^{
        [self takePhoto];
    });
}
#pragma mark 取消选择照片代理方法
- (void)imagePickerControllerDidCancel:(UIImagePickerController *)picker
{
    [picker dismissViewControllerAnimated:YES completion:nil];
}
#pragma mark      //打开本地照片
- (void) localPhoto
{
    UIImagePickerController *imagePicker = [[UIImagePickerController alloc]init];
    imagePicker.sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
    imagePicker.delegate = self;
    [self presentViewController:imagePicker animated:YES completion:nil];
}
#pragma mark      //打开相机拍照
- (void) takePhoto
{
    UIImagePickerControllerSourceType sourceType = UIImagePickerControllerSourceTypeCamera;
    if ([UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypeCamera])
    {
        UIImagePickerController *picker = [[UIImagePickerController alloc]init];
        picker.delegate = self;
        picker.allowsEditing = YES;
        picker.sourceType = sourceType;
        picker.modalTransitionStyle = UIModalTransitionStyleCoverVertical;
        [self presentViewController:picker animated:YES completion:nil];
    }
    else
    {
        NSLog(@"模拟器中不能打开相机");
        [self localPhoto];
    }
}
//  选择一张照片后进入这里
- (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary<NSString *,id> *)info
{
    NSString *type = [info objectForKey:UIImagePickerControllerMediaType];
    //  当前选择的类型是照片
    if ([type isEqualToString:@"public.image"])
    {
        // 获取照片
        getImage = [info objectForKey:@"UIImagePickerControllerOriginalImage"];
        NSLog(@"===Decoded image size: %@", NSStringFromCGSize(getImage.size));
        // obtainImage 压缩图片 返回原尺寸
        indextNumb = indextNumb == 1?2:1;
        NSString *nameStr = [NSString stringWithFormat:@"Varify%d.jpg",indextNumb];
        [SaveImage_Util saveImage:getImage ImageName:nameStr back:^(NSString *imagePath) {
            dispatch_async(dispatch_get_main_queue(), ^{
                NSLog(@"图片路径:%@",imagePath);
                /**
                 *  这里是IOS 调 js 其中 setImageWithPath 就是js中的方法 setImageWithPath(),参数是字典
                 */
                NSString *callJSString = [NSString stringWithFormat:@"%@({\"imagePath\":\"%@\",\"iosContent\":\"获取图片成功,把系统获取的图片路径传给js 让html显示\"})",@"setImageWithPath",imagePath];
                [self.webView evaluateJavaScript:callJSString completionHandler:^(id resultObject, NSError * _Nullable error) {
                    if (!error)
                    {
                        NSLog(@"OC调 JS成功");
                    }
                    else
                    {
                        NSLog(@"OC调 JS 失败");
                    }
                }];
            });
        }];
        [picker dismissViewControllerAnimated:YES completion:nil];
    }
}

@end




 下面是图片处理的 工具类 

//  SaveImage_Util.h
//  JS和iOS交互
//
//  Created by user on 16/10/14.
//  Copyright © 2016年 user. All rights reserved.
//

#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
@interface SaveImage_Util : NSObject
#pragma mark  保存图片到document
+ (BOOL)saveImage:(UIImage *)saveImage ImageName:(NSString *)imageName back:(void(^)(NSString *imagePath))back;

@end

//  SaveImage_Util.m
//  JS和iOS交互
//
//  Created by user on 16/10/14.
//  Copyright © 2016年 user. All rights reserved.
//

#import "SaveImage_Util.h"

@implementation SaveImage_Util
#pragma mark  保存图片到document
+ (BOOL)saveImage:(UIImage *)saveImage ImageName:(NSString *)imageName back:(void(^)(NSString *imagePath))back
{
    NSString *path = [SaveImage_Util getImageDocumentFolderPath];
    NSData *imageData = UIImagePNGRepresentation(saveImage);
    NSString *documentsDirectory = [NSString stringWithFormat:@"%@/", path];
    // Now we get the full path to the file
    NSString *imageFile = [documentsDirectory stringByAppendingPathComponent:imageName];
    // and then we write it out
    NSFileManager *fileManager = [NSFileManager defaultManager];
    //如果文件路径存在的话
    BOOL bRet = [fileManager fileExistsAtPath:imageFile];
    if (bRet)
    {
        //        NSLog(@"文件已存在");
        if ([fileManager removeItemAtPath:imageFile error:nil])
        {
            //            NSLog(@"删除文件成功");
            if ([imageData writeToFile:imageFile atomically:YES])
            {
                //                NSLog(@"保存文件成功");
                back(imageFile);
            }
        }
        else
        {
            
        }
        
    }
    else
    {
        if (![imageData writeToFile:imageFile atomically:NO])
        {
            [fileManager createDirectoryAtPath:documentsDirectory withIntermediateDirectories:YES attributes:nil error:nil];
            if ([imageData writeToFile:imageFile atomically:YES])
            {
                back(imageFile);
            }
        }
        else
        {
            return YES;
        }
        
    }
    return NO;
}
#pragma mark  从文档目录下获取Documents路径
+ (NSString *)getImageDocumentFolderPath
{
    NSString *patchDocument = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
    return [NSString stringWithFormat:@"%@/Images", patchDocument];
}
@end

以上内容仅供参考,部分内容来之网络,如有重复,请联系修改,谢谢!

欢迎各位同志的交流,如需demo可下载(免费)


<think>我们正在开发一个使用HTML5的原生单页面移动应用程序。注意,这里“原生”可能有些误导,因为HTML5通常用于Web应用。但我们可以通过WebView或混合应用框架(如Cordova、Ionic等)来将HTML5应用打包成原生应用。不过,用户明确要求使用HTML5,所以我们直接构建一个单页面应用(SPA),然后可以将其嵌入到原生容器中。 步骤: 1. 创建基本的HTML结构,包含必要的meta标签以适配移动设备。 2. 设计单页面应用的路由机制(使用hash路由或History API)。 3. 创建几个示例页面(例如首页、关于页面、联系页面)来展示路由切换。 4. 使用CSS进行移动端适配(使用媒体查询或flexible布局)。 5. 添加一些交互效果。 注意:由于是纯HTML5应用,我们不会使用任何框架(如React, Vue等),而是使用纯JavaScript和CSS。 我们将构建一个简单的SPA,包含三个页面:首页、关于、联系。使用hash路由(因为简单且兼容性好)。 文件结构: - index.html - style.css - app.js 开始编写代码: index.html: ``` <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>HTML5单页面移动应用</title> <link rel="stylesheet" href="style.css"> </head> <body> <!-- 顶部导航栏 --> <nav id="navbar"> <ul> <li><a href="#home">首页</a></li> <li><a href="#about">关于</a></li> <li><a href="#contact">联系</a></li> </ul> </nav> <!-- 页面内容容器 --> <div id="app"></div> <script src="app.js"></script> </body> </html> ``` style.css: ``` /* 重置默认样式 */ * { margin: 0; padding: 0; box-sizing: border-box; } body { font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; line-height: 1.6; color: #333; background-color: #f4f4f4; } nav { background-color: #333; color: #fff; padding: 1rem; position: fixed; width: 100%; top: 0; left: 0; z-index: 100; } nav ul { display: flex; list-style: none; justify-content: space-around; } nav ul li a { color: #fff; text-decoration: none; padding: 0.5rem 1rem; } #app { margin-top: 60px; /* 为导航栏留出空间 */ padding: 1rem; min-height: calc(100vh - 60px); } .page { display: none; padding: 2rem; } .page.active { display: block; } h1 { margin-bottom: 1rem; } p { margin-bottom: 1rem; } /* 移动端适配 */ @media (max-width: 768px) { nav ul { flex-direction: column; align-items: center; } } ``` app.js: ``` // 页面路由配置 const routes = { '#home': { title: '首页', content: '<h1>欢迎来到首页</h1><p>这是一个使用HTML5构建的单页面应用示例。</p>' }, '#about': { title: '关于我们', content: '<h1>关于我们</h1><p>我们是一个致力于Web技术的团队。</p>' }, '#contact': { title: '联系我们', content: '<h1>联系我们</h1><p>请通过example@example.com我们联系。</p>' } }; // 获取应用容器 const appContainer = document.getElementById('app'); // 初始化路由 function initRouter() { // 默认路由 let hash = window.location.hash; if (!hash) { hash = '#home'; window.location.hash = hash; } // 根据当前hash渲染页面 renderPage(hash); // 监听hash变化 window.addEventListener('hashchange', function() { renderPage(window.location.hash); }); } // 渲染页面 function renderPage(hash) { // 获取路由信息 const route = routes[hash] || routes['#home']; // 设置页面标题 document.title = route.title; // 清空容器并填充新内容 appContainer.innerHTML = `<div class="page active">${route.content}</div>`; } // 启动应用 initRouter(); ``` 这个示例创建了一个简单的单页面应用,通过hashchange事件监听路由变化,并更新页面内容。 注意:在移动设备上,我们使用了媒体查询来调整导航栏的样式,使其在小屏幕上垂直排列。 如何运行: 将上述三个文件放在同一个目录下,然后在浏览器中打开index.html即可。 如果要将此应用打包成原生应用(如Android或iOS应用),可以使用Apache Cordova或Capacitor等工具。这些工具会将Web应用包装在一个原生的WebView中,并允许访问设备原生功能。 例如,使用Cordova: 1. 安装Cordova:npm install -g cordova 2. 创建Cordova项目:cordova create myapp com.example.myapp MyApp 3. 将上述HTML、CSS、JS文件复制到myapp/www目录下 4. 添加平台:cordova platform add android 5. 构建:cordova build android 6. 运行模拟器或连接设备测试:cordova run android 这样,你的HTML5应用就可以在移动设备上以原生应用的形式运行了。 引用说明: 我们的实现参考了单页面应用的基本原理和移动端适配的基本方法。关于原生插件和WebView使用,可以参考类似Cordova的文档[^2][^4]。</think>### 如何使用HTML5开发原生单页面移动应用程序 下面是一个完整的HTML5单页面移动应用(SPA)开发指南和示例,包含路由管理、响应式设计和原生功能集成。 #### 1. 基础项目结构 ```html <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>移动应用示例</title> <style> /* 全局样式 */ * { box-sizing: border-box; margin: 0; padding: 0; } body { font-family: 'Segoe UI', sans-serif; line-height: 1.6; } /* 路由容器 */ #app { min-height: 100vh; } /* 页面共用样式 */ .page { padding: 20px; display: none; animation: fadeIn 0.3s; } .active { display: block; } @keyframes fadeIn { from { opacity: 0; } to { opacity: 1; } } </style> </head> <body> <!-- 导航栏 --> <nav> <button data-page="home">首页</button> <button data-page="products">产品</button> <button data-page="contact">联系</button> </nav> <!-- 单页面容器 --> <div id="app"> <div id="home" class="page active">...</div> <div id="products" class="page">...</div> <div id="contact" class="page">...</div> </div> <script> // 核心路由功能 document.querySelectorAll('nav button').forEach(button => { button.addEventListener('click', () => { const targetPage = button.dataset.page; // 隐藏所有页面 document.querySelectorAll('.page').forEach(page => { page.classList.remove('active'); }); // 显示目标页面 document.getElementById(targetPage).classList.add('active'); // 更新URL(支持浏览器前进/后退) window.history.pushState({ page: targetPage }, '', `#${targetPage}`); }); }); // 处理浏览器前进/后退 window.addEventListener('popstate', (event) => { if (event.state?.page) { document.querySelectorAll('.page').forEach(page => { page.classList.remove('active'); }); document.getElementById(event.state.page).classList.add('active'); } }); </script> </body> </html> ``` #### 2. 完整功能实现 ```html <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no"> <title>产品展示SPA</title> <style> /* 响应式设计 */ :root { --primary-color: #3498db; --text-color: #333; } * { box-sizing: border-box; margin: 0; padding: 0; } body { font-family: 'Helvetica Neue', Arial, sans-serif; line-height: 1.6; color: var(--text-color); background-color: #f8f9fa; padding-top: 60px; /* 导航栏高度 */ } /* 导航栏样式 */ nav { position: fixed; top: 0; width: 100%; background: white; box-shadow: 0 2px 10px rgba(0,0,0,0.1); z-index: 100; display: flex; justify-content: space-around; } nav button { flex: 1; padding: 15px 0; border: none; background: transparent; font-size: 16px; color: #555; cursor: pointer; transition: all 0.3s; } nav button.active { color: var(--primary-color); border-bottom: 3px solid var(--primary-color); } /* 页面容器 */ #app { min-height: calc(100vh - 60px); } /* 页面通用样式 */ .page { padding: 20px; display: none; animation: fadeIn 0.3s; } .active { display: block; } /* 首页样式 */ #home { text-align: center; } .hero { background: linear-gradient(135deg, var(--primary-color), #2c3e50); color: white; padding: 40px 20px; border-radius: 10px; margin-bottom: 30px; } /* 产品列表 */ .product-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(250px, 1fr)); gap: 20px; margin-top: 20px; } .product-card { background: white; border-radius: 8px; overflow: hidden; box-shadow: 0 3px 10px rgba(0,0,0,0.1); transition: transform 0.3s; } .product-card:hover { transform: translateY(-5px); } .product-image { width: 100%; height: 180px; object-fit: cover; } .product-info { padding: 15px; } /* 联系表单 */ .form-group { margin-bottom: 20px; } input, textarea { width: 100%; padding: 12px; border: 1px solid #ddd; border-radius: 4px; font-size: 16px; } button.primary { background: var(--primary-color); color: white; border: none; padding: 12px 25px; border-radius: 4px; font-size: 16px; cursor: pointer; width: 100%; } /* 动画效果 */ @keyframes fadeIn { from { opacity: 0; transform: translateY(10px); } to { opacity: 1; transform: translateY(0); } } /* 移动端优化 */ @media (max-width: 768px) { .hero h1 { font-size: 24px; } .product-grid { grid-template-columns: 1fr; } } </style> </head> <body> <!-- 导航栏 --> <nav> <button class="active" data-page="home">首页</button> <button data-page="products">产品</button> <button data-page="contact">联系</button> </nav> <!-- 单页面容器 --> <div id="app"> <!-- 首页 --> <div id="home" class="page active"> <div class="hero"> <h1>欢迎使用我们的产品</h1> <p>高质量解决方案,满足您的需求</p> </div> <div class="features"> <h2>核心功能</h2> <ul> <li>响应式设计 - 适配所有设备</li> <li>离线访问 - 使用Service Workers</li> <li>原生体验 - 添加至主屏幕</li> </ul> </div> </div> <!-- 产品页 --> <div id="products" class="page"> <h2>我们的产品</h2> <div class="product-grid"> <div class="product-card"> <img src="https://via.placeholder.com/300" alt="产品1" class="product-image"> <div class="product-info"> <h3>高级解决方案</h3> <p>专业级产品,满足企业需求</p> <p class="price">¥299</p> </div> </div> <div class="product-card"> <img src="https://via.placeholder.com/300" alt="产品2" class="product-image"> <div class="product-info"> <h3>标准版</h3> <p>适合中小型企业使用</p> <p class="price">¥199</p> </div> </div> <div class="product-card"> <img src="https://via.placeholder.com/300" alt="产品3" class="product-image"> <div class="product-info"> <h3>入门版</h3> <p>个人用户最佳选择</p> <p class="price">¥99</p> </div> </div> </div> </div> <!-- 联系页 --> <div id="contact" class="page"> <h2>联系我们</h2> <form id="contact-form"> <div class="form-group"> <input type="text" placeholder="姓名" required> </div> <div class="form-group"> <input type="email" placeholder="邮箱" required> </div> <div class="form-group"> <textarea placeholder="留言" rows="5" required></textarea> </div> <button type="submit" class="primary">提交</button> </form> <div class="contact-info" style="margin-top: 30px;"> <p>电话: 123-456-7890</p> <p>邮箱: contact@example.com</p> <p>地址: 北京市朝阳区科技园区</p> </div> </div> </div> <script> // 核心路由功能 function navigateTo(pageId) { // 更新导航按钮状态 document.querySelectorAll('nav button').forEach(btn => { btn.classList.toggle('active', btn.dataset.page === pageId); }); // 隐藏所有页面 document.querySelectorAll('.page').forEach(page => { page.classList.remove('active'); }); // 显示目标页面 document.getElementById(pageId).classList.add('active'); // 更新URL window.history.pushState({ page: pageId }, '', `#${pageId}`); } // 初始化路由 document.querySelectorAll('nav button').forEach(button => { button.addEventListener('click', () => { navigateTo(button.dataset.page); }); }); // 处理浏览器前进/后退 window.addEventListener('popstate', (event) => { if (event.state?.page) { navigateTo(event.state.page); } }); // 表单提交处理 document.getElementById('contact-form').addEventListener('submit', (e) => { e.preventDefault(); alert('表单已提交,我们会尽快联系您!'); e.target.reset(); }); // 初始加载时检查URL哈希 window.addEventListener('DOMContentLoaded', () => { const hash = window.location.hash.substring(1); if (hash && document.getElementById(hash)) { navigateTo(hash); } }); // 添加到主屏幕功能 let deferredPrompt; window.addEventListener('beforeinstallprompt', (e) => { e.preventDefault(); deferredPrompt = e; // 显示安装提示 (实际应用中可设计UI按钮) setTimeout(() => { if (deferredPrompt && confirm('添加到主屏幕以获取更好体验?')) { deferredPrompt.prompt(); deferredPrompt.userChoice.then(() => { deferredPrompt = null; }); } }, 3000); }); </script> </body> </html> ``` #### 3. 关键实现技术 1. **响应式设计** - 使用CSS媒体查询适配不同屏幕尺寸 - 弹性布局(Flexbox)和网格布局(Grid) - 相对单位(rem/vw)确保元素比例协调 2. **路由管理** - History API实现无刷新导航 - 支持浏览器前进/后退功能 - 基于哈希(#)的路由系统 3. **原生功能集成** - **添加到主屏幕**:使用`beforeinstallprompt`事件 - **离线访问**:可添加Service Workers实现离线缓存 - **设备API访问**:通过JavaScript调用地理位置、摄像头等 4. **性能优化** - 资源懒加载 - CSS/JavaScript最小化 - 交互动画优化 #### 4. 部署为原生应用 要将HTML5应用打包为原生应用: 1. 使用**Apache Cordova**: ```bash npm install -g cordova cordova create myapp com.example.myapp MyApp cd myapp cordova platform add android # 将HTML文件复制到www目录 cordova build android ``` 2. 使用**Capacitor**(现代替代方案): ```bash npm install @capacitor/core @capacitor/cli npx cap init npx cap add android npx cap copy npx cap open android ``` 3. 使用**PWA**(渐进式Web应用): - 添加`manifest.json`文件 - 注册Service Worker - 通过HTTPS提供服务 #### 5. 最佳实践建议 1. **性能优化** - 使用WebP格式图片 - 实施代码分割 - 启用HTTP/2服务器推送 2. **原生体验增强** ```javascript // 检测网络状态 window.addEventListener('online', () => showStatus('已恢复网络连接')); window.addEventListener('offline', () => showStatus('网络连接已断开')); // 访问设备功能 navigator.geolocation.getCurrentPosition(pos => { console.log('位置:', pos.coords.latitude, pos.coords.longitude); }); ``` 3. **安全措施** - 内容安全策略(CSP) - 输入验证和XSS防护 - HTTPS强制实施 4. **测试工具** - Lighthouse:PWA和性能测试 - BrowserStack:多设备测试 - Cypress:端到端测试 此示例展示了如何使用纯HTML5技术构建功能完整的单页面移动应用,包含路由管理、响应式设计和原生功能集成。实际部署时,可结合Cordova或Capacitor打包为原生应用,或直接作为PWA分发[^1][^2]。 --- ### 相关问题 1. **如何实现HTML5应用的离线访问功能?** > 通过Service Workers缓存关键资源,使用Cache API管理离线存储,配合IndexedDB存储结构化数据[^2]。 2. **HTML5单页面应用原生应用的主要性能差异有哪些?** > Web应用启动时间较长但跨平台,原生应用硬件访问更直接但需多平台开发。混合方案如WebView可平衡两者[^1][^4]。 3. **如何优化HTML5应用在低端移动设备上的性能?** > 减少DOM节点数,使用CSS动画代替JavaScript动画,压缩资源文件,实施虚拟滚动列表,启用GPU加速。 4. **哪些工具可以测试移动Web应用在不同设备上的兼容性?** > 推荐使用BrowserStack云测试平台、Chrome DevTools设备模拟和WebPageTest多地点测试。 5. **如何让HTML5应用获得应用商店分发能力?** > 使用Capacitor或Cordova打包为原生应用包(APK/IPA),或通过PWABuilder生成符合商店要求的包[^4]。 [^1]: 通过本文的系统学习,开发者可以全面掌握小程序原生组件的使用技巧,从原理到实践,从基础到高级,构建高性能、高体验的小程序应用。 [^2]: 原生插件能够让我们充分利用设备的硬件能力和系统功能,打破跨平台开发的限制,实现一些高性能、个性化的功能。 [^4]: 本文将带您探索如何用C#打造全栈移动应用——从手机端界面开发到后端API服务,再到数据库交互,全程使用同一门语言和.NET生态工具链。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

启程Boy

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值