文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

卡片服务开发之如何开发一个地图服务卡片

2024-12-03 01:02

关注

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

前言

处于隐私保护借用熊猫基地定位,代码层实现了获取实时定位功能。

效果图

卡片效果2x2+2x44x4缩放

关键技术及实现原理

卡片现有支持的基础组件有:button、calendar、chart、clock、divider、image、input、progress、span、text

可以看到现有的卡片组件并不支持地图的开发,那么如何在卡片上显示地图尼?

通过image组件+高德地图WebAPI的静态地图即可实现地图的显示。

-以上方便有开发卡片经验的开发者提供思路,具体方式方法如下

从零开始

创建项目

打开DevEco Studio工具,点击File->New->New Project创建一个Empty Ability(JS)如下图:

SDK选用API 5

创建后的结构:


首先修改程序的配置文件:

打开config.json,修改卡片支持类型情况:


添加权限:


配置完成还需要在MainAbility中显示的申明使用权限信息,详情参考文档配置相关内容:

打开MainAbility添加方法:

  1. //获取权限 
  2.    private void requestPermission() { 
  3.        String[] permission = { 
  4.                "ohos.permission.LOCATION"
  5.                "ohos.permission.LOCATION_IN_BACKGROUND"
  6.        }; 
  7.        List applyPermissions = new ArrayList<>(); 
  8.        for (String element : permission) { 
  9.            if (verifySelfPermission(element) != 0) { 
  10.                if (canRequestPermission(element)) { 
  11.                    applyPermissions.add(element); 
  12.                } 
  13.            } 
  14.        } 
  15.        requestPermissionsFromUser(applyPermissions.toArray(new String[0]), 0); 
  16.    } 

并在onStart方法中调用requestPermission();方法。

修改界面:

打开widget下的pages/index/imdex.hml

  1. "container"
  2.         "container-inner" > 
  3.             "container-img"
  4.                  
  5.                     "{{imgSrc}}" class="bg-img"
  6.                     "container-show-text" > 
  7.                         "show-text" >当前检索项: 
  8.                         "show-text" style="color: coral;"  >{{searchText}} 
  9.                     
 
  •                      
  •                     "container-map-ctl"
  •                         "map-ctl-btn" @click="mapAddEvent"  type="circle">+ 
  •                         "map-ctl-btn" @click="mapReduceEvent"  type="circle">- 
  •                     
  •  
  •                     "{{showCtlButton}}" class="container-ctl" > 
  •                         "ctl-btn" @click="searchCheckedEvent0">{{searchBtns[0]}} 
  •                         "ctl-btn" @click="searchCheckedEvent1">{{searchBtns[1]}} 
  •                         "ctl-btn" @click="searchCheckedEvent2">{{searchBtns[2]}} 
  •                         "ctl-btn" @click="searchCheckedEvent3">{{searchBtns[3]}} 
  •                         "ctl-btn" @click="searchCheckedEvent4">{{searchBtns[4]}} 
  •                      
  •                  
  •              
  •          
  •  
  • 需要注意:卡片的事件不能使用表达式,不能使用for语句循环构建

    样式调整文件pages/index/imdex.css:

    1. .container { 
    2.     flex-direction: column
    3.     justify-content: center; 
    4.     align-items: center; 
    5.  
    6. .bg-img { 
    7.     flex-shrink: 0; 
    8.     height: 100%; 
    9.     object-fit: cover; 
    10.  
    11. .container-ctl{ 
    12.     opacity: 0.9; 
    13.     width: 100%; 
    14.     height: 100%; 
    15.     justify-content: center; 
    16.     flex-direction: row; 
    17.     align-items: flex-end
    18.     bottom: 3px; 
    19. .ctl-btn{ 
    20.     padding: 3px 6px; 
    21.     margin:3px 6px; 
    22.     font-size: 12px; 
    23.     border-radius: 3px; 
    24.     background-color: #409eff; 
    25.     border: 1px solid #cbcbcb; 
    26.     box-shadow: 1px 1px 3px #a8a8a8; 
    27. .container-map-ctl{ 
    28.     opacity: 0.8; 
    29.     justify-content: flex-end
    30.     margin-right: 3px; 
    31. .map-ctl-btn{ 
    32.     background-color: #409eff; 
    33.     border: 1px solid #cbcbcb; 
    34.     box-shadow: 1px 1px 3px #a8a8a8; 
    35.     width: 24px; 
    36.     height: 24px; 
    37.     margin:3px; 
    38.  
    39. .container-show-text{ 
    40.     padding: 9px; 
    41. .show-text{ 
    42.     font-size: 8px; 
    43.     font-weight: bolder; 
    44.  

    json配置信息修改pages/index/index.json:

    1.   "data": { 
    2.     "showCtlButton"false,//是否显示button。由Java传值且在2x2的界面不显示 
    3.     "imgSrc""/common/ic_default_image@3x.png",//默认图片 
    4.     "searchText"""
    5.     "searchBtns": []//配置的button按钮信息 
    6.   }, 
    7.   "actions": { 
    8.     "searchCheckedEvent0": { 
    9.       "action""message"
    10.       "params": { 
    11.         "index": 0, 
    12.         "name""checkSearch" 
    13.       } 
    14.     }, 
    15.     "searchCheckedEvent1": { 
    16.       "action""message"
    17.       "params": { 
    18.         "index": 1, 
    19.         "name""checkSearch" 
    20.       } 
    21.     }, 
    22.     "searchCheckedEvent2": { 
    23.       "action""message"
    24.       "params": { 
    25.         "index": 2, 
    26.         "name""checkSearch" 
    27.       } 
    28.     }, 
    29.     "searchCheckedEvent3": { 
    30.       "action""message"
    31.       "params": { 
    32.         "index": 3, 
    33.         "name""checkSearch" 
    34.       } 
    35.     }, 
    36.     "searchCheckedEvent4": { 
    37.       "action""message"
    38.       "params": { 
    39.         "index": 4, 
    40.         "name""checkSearch" 
    41.       } 
    42.     }, 
    43.     "mapAddEvent": { 
    44.       "action""message"
    45.       "params": { 
    46.         "name""mapAdd" 
    47.       } 
    48.     }, 
    49.     "mapReduceEvent": { 
    50.       "action""message"
    51.       "params": { 
    52.         "name""mapReduce" 
    53.       } 
    54.     } 
    55.   } 

    后台逻辑

    由于更新卡片时需要提供formId,我们对FormController及FormControllerManager这两个帮助类进行一个修改。

    打开java目录下的FormController文件并添加受保护的属性 formId,并修改构造函数。

    然后进入FormControllerManager找到createFormController、getController、newInstance进行修改。

    createFormController:

    在newInstance方法中添加参数formId,如下图:

    getController:

    在newInstance方法中添加参数formId,如下图:

    newInstace:

    该方法是动态的创建WidgetImpl方法,类似于IOC作用。

    找到java目录下的widget/widget/widgetImpl,卡片的所有逻辑都在该文件内

    首先修改构造函数及定义基础属性等

    因上述修改了FormController及FormControllerManager构造函数必须增加Long formId参数

    1. private static Location slocation=null;//当前位置信息 
    2.    private Boolean slocationChanged=false;//位置是否修改 
    3.    private  int dimension=2;//当前卡片模式  2x2=2;2x4=3;4x4=4; 
    4.    private List defualtBtn=new ArrayList<>();//界面下方的按钮列表 
    5.    private static Locator locator=null;//坐标获取类 
    6.    private LocatorCallBack locatorCallBack=new LocatorCallBack();//坐标获取后返回调用类 
    7.    private int mRoom=16;//静态地图显示层级 
    8.    private String markType="";//静态地图周边搜索关键字 
    9.    private String mSize="500*500";//静态地图大小 
    10.    private List mKeyLocation=new ArrayList<>();//静态地图获取周边标记的坐标 
    11.    RequestParam requestParam = new RequestParam(RequestParam.PRIORITY_ACCURACY, 20, 0); 
    12.  
    13.  
    14.    public WidgetImpl(Context context, String formName, Integer dimension,Long formId) { 
    15.        super(context, formName, dimension,formId); 
    16.        this.dimension=dimension; 
    17.        //获取当前定位 
    18.        if(locator==null){ 
    19.            locator=new Locator(context); 
    20.            locator.startLocating(requestParam,locatorCallBack); 
    21.        } 
    22.        switch (dimension){ 
    23.            case 2:{ 
    24.                mSize="300*300"
    25.                mRoom=13; 
    26.                break; 
    27.            } 
    28.            case 3:{ 
    29.              mSize="500*250"
    30.              mRoom=13; 
    31.              break; 
    32.            } 
    33.            case 4:{ 
    34.                mSize="500*500"
    35.                mRoom=15; 
    36.                break; 
    37.            } 
    38.        } 
    39.    } 
    40.  
    41.    public class LocatorCallBack implements LocatorCallback{ 
    42.  
    43.        @Override 
    44.        public void onLocationReport(Location location) { 
    45.            slocation=location; 
    46.            //周边信息接口额度有限,限制为当坐标改变时刷新坐标mark信息,并更新卡片 
    47.            if(location==slocation || slocation==null
    48.                return
    49.            refreshMark(); 
    50.            updateFormData(formId); 
    51.        } 
    52.  
    53.        @Override 
    54.        public void onStatusChanged(int i) { 
    55.  
    56.        } 
    57.  
    58.        @Override 
    59.        public void onErrorReport(int i) { 
    60.  
    61.        } 
    62.    } 

    修改createFormController,该方法在卡片创建时调用,我们需要把页面需要的参数传递过去

    注意网络图片需要使用“通过内存图片方式使用image组件”

    1. @Override 
    2.    public ProviderFormInfo bindFormData(){ 
    3.        defualtBtn=new ArrayList<>(); 
    4.        defualtBtn.add("酒店"); 
    5.        defualtBtn.add("餐饮"); 
    6.        defualtBtn.add("景点"); 
    7.        defualtBtn.add("加油站"); 
    8.        if(defualtBtn.size()<5){ 
    9.            for(int i=defualtBtn.size();i<5;i++){ 
    10.                defualtBtn.add("未设置"); 
    11.            } 
    12.        } 
    13.        this.markType=defualtBtn.get(0); 
    14.        this.refreshMark(); 
    15.        FormBindingData formBindingData=null
    16.        ZSONObject zsonObject =new ZSONObject(); 
    17.        zsonObject.put("imgSrc","memory://amap.png"); 
    18.        zsonObject.put("showCtlButton",this.dimension!=2); 
    19.        zsonObject.put("searchBtns",defualtBtn); 
    20.        zsonObject.put("searchText",markType); 
    21.        formBindingData=new FormBindingData(zsonObject); 
    22.        ProviderFormInfo formInfo = new ProviderFormInfo(); 
    23.        formInfo.setJsBindingData(formBindingData); 
    24.        String amapUrl=getMapImageUrl(mKeyLocation); 
    25.        byte[] bytes= HttpImageUtils.doGetRequestForFile(amapUrl); 
    26.        formBindingData.addImageData("amap.png",bytes); 
    27.        return formInfo; 
    28.    } 

    初始化卡片后改进onTriggerFormEvent

    该方法为接收卡片事件,message为事件传递的params参数

    1. @Override 
    2.   public void onTriggerFormEvent(long formId, String message) { 
    3.       ZSONObject request=ZSONObject.stringToZSON(message); 
    4.       String EventName=request.getString("name"); 
    5.       switch (EventName){ 
    6.           case "checkSearch":{ 
    7.               int index=request.getIntValue("index"); 
    8.               markType=defualtBtn.get(index); 
    9.               this.refreshMark(); 
    10.               break; 
    11.           } 
    12.           case "mapAdd":{ 
    13.               if(mRoom<17){ 
    14.                   mRoom+=1; 
    15.               } 
    16.               break; 
    17.           } 
    18.           case "mapReduce":{ 
    19.               if(mRoom>0){ 
    20.                   mRoom-=1; 
    21.               } 
    22.               break; 
    23.           } 
    24.       } 
    25.       updateFormData(formId); 
    26.   } 

    修改更新卡片信息的方法(此方法不仅是系统会定时刷新,也有主动刷新的调用如:卡片事件改变后调用,坐标改变后的调用,这也是需要修改FormController、FormControllerManager增加formId属性的原因,因为在主动刷新时需要formId参数)

    此处还有一个重点就是 ((Ability)context).updateForm(formId,bindingData);

    1. @Override 
    2.    public void updateFormData(long formId, Object... vars) { 
    3.        ZSONObject zsonObject=new ZSONObject(); 
    4.        zsonObject.put("searchBtns",defualtBtn); 
    5.        zsonObject.put("searchText",markType); 
    6.        String mapName="amap"+System.currentTimeMillis()+".png"
    7.        zsonObject.put("imgSrc","memory://"+mapName); 
    8.        FormBindingData bindingData = new FormBindingData(zsonObject); 
    9.        String amapUrl=getMapImageUrl(mKeyLocation); 
    10.        byte[] bytes= HttpImageUtils.doGetRequestForFile(amapUrl); 
    11.        bindingData.addImageData(mapName,bytes); 
    12.        try{ 
    13.            ((Ability)context).updateForm(formId,bindingData); 
    14.        }catch (Exception ex){ 
    15.            ex.printStackTrace(); 
    16.        } 
    17.  
    18.    } 

    其它一些上述方法中调用的私有方法及类

    私有方法:

    1. private void refreshMark(){ 
    2.       try{ 
    3.           this.mKeyLocation= HttpImageUtils.SearchByKeyUrl(getMapMarkUrl(10)); 
    4.       }catch (Exception ex){ 
    5.           ex.printStackTrace(); 
    6.       } 
    7.   } 
    8.  
    9.   private String getMapImageUrl(List Position){ 
    10.       String url="https://restapi.amap.com/v3/staticmap"
    11.       String params="key="
    12.       params+="&zoom="+mRoom; 
    13.       params+="&size="+mSize; 
    14.       if(slocation!=null
    15.           params+="&location="+slocation.getLongitude()+","+slocation.getLatitude(); 
    16.       params+="&markers=large,0xea7700,H:"+slocation.getLongitude()+","+slocation.getLatitude(); 
    17.       if(Position==null || Position.size()==0) 
    18.           return  url+"?"+params; 
    19.       String markers="|mid,0xFF0000,:"
    20.  
    21.       for(int i=0;isize();i++){ 
    22.           markers+=Position.get(i)+";"
    23.       } 
    24.       params+=markers.substring(0,markers.length()-1); 
    25.       return url+"?"+params; 
    26.   } 
    27.  
    28.   private  String getMapMarkUrl(int size){ 
    29.       String Url="https://restapi.amap.com/v5/place/around?key="
    30.       Url+="&keywords="+(markType=="未设置"?"":markType); 
    31.       if(slocation!=null
    32.           Url+="&location="+slocation.getLongitude()+","+slocation.getLatitude(); 
    33.       Url+="&size="+size
    34.       return Url; 
    35.   } 

    HttpImageUtils类

    1. package com.panda_coder.amapcard.utils; 
    2.  
    3. import com.panda_coder.amapcard.MainAbility; 
    4. import ohos.hiviewdfx.HiLog; 
    5. import ohos.hiviewdfx.HiLogLabel; 
    6. import ohos.utils.zson.ZSONArray; 
    7. import ohos.utils.zson.ZSONObject; 
    8.  
    9. import java.io.*; 
    10. import java.net.HttpURLConnection; 
    11. import java.net.URL; 
    12. import java.util.ArrayList; 
    13. import java.util.List; 
    14.  
    15. public class HttpImageUtils { 
    16.     private static final HiLogLabel TAG = new HiLogLabel(HiLog.DEBUG, 0x0, MainAbility.class.getName()); 
    17.  
    18.     public final static byte[] doGetRequestForFile(String urlStr) { 
    19.         InputStream is = null
    20.         HttpURLConnection conn = null
    21.         byte[] buff = new byte[1024]; 
    22.         try { 
    23.             URL url = new URL(urlStr); 
    24.             conn = (HttpURLConnection) url.openConnection(); 
    25.  
    26.             conn.setDoInput(true); 
    27.             conn.setRequestMethod("GET"); 
    28.             conn.setReadTimeout(6000); 
    29.             conn.connect(); 
    30.             is = conn.getInputStream(); 
    31.             if (conn.getResponseCode() == 200) { 
    32.                 buff = readInputStream(is); 
    33.             } else
    34.                 buff=null
    35.             } 
    36.         } catch (Exception e) { 
    37.             HiLog.error(TAG,"【获取图片异常】",e); 
    38.         } 
    39.         finally { 
    40.             try { 
    41.                 if(is != null){ 
    42.                     is.close(); 
    43.                 } 
    44.             } catch (IOException e) { 
    45.                 e.printStackTrace(); 
    46.             } 
    47.             conn.disconnect(); 
    48.         } 
    49.  
    50.         return buff; 
    51.     } 
    52.  
    53.     public static byte[] readInputStream(InputStream is) { 
    54.         ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
    55.         byte[] buffer = new byte[1024]; 
    56.         int length = -1; 
    57.         try { 
    58.             while ((length = is.read(buffer)) != -1) { 
    59.                 baos.write(buffer, 0, length); 
    60.             } 
    61.             baos.flush(); 
    62.         } catch (IOException e) { 
    63.             e.printStackTrace(); 
    64.         } 
    65.         byte[] data = baos.toByteArray(); 
    66.         try { 
    67.             is.close(); 
    68.             baos.close(); 
    69.         } catch (IOException e) { 
    70.             e.printStackTrace(); 
    71.         } 
    72.         return data; 
    73.     } 
    74.  
    75.     public static String httpGet(String urlStr){ 
    76.         InputStream is = null
    77.         HttpURLConnection conn = null
    78.         String response=""
    79.         StringBuffer buffer = new StringBuffer(); 
    80.         try { 
    81.             URL url = new URL(urlStr); 
    82.             conn = (HttpURLConnection) url.openConnection(); 
    83.  
    84.             conn.setDoInput(true); 
    85.             conn.setRequestMethod("GET"); 
    86.             conn.setReadTimeout(6000); 
    87.             conn.connect(); 
    88.             is = conn.getInputStream(); 
    89.             if (conn.getResponseCode() == 200) { 
    90.                 String str=null
    91.                 InputStreamReader isr = new InputStreamReader(is,"utf-8"); 
    92.                 BufferedReader br = new BufferedReader(isr); 
    93.                 while((response = br.readLine())!=null){ 
    94.                     buffer.append(response); 
    95.                 } 
    96.             } 
    97.             response=buffer.toString(); 
    98.  
    99.         } catch (Exception e) { 
    100.             HiLog.error(TAG,"【访问异常】",e); 
    101.         } 
    102.         finally { 
    103.             try { 
    104.                 if(is != null){ 
    105.                     is.close(); 
    106.                 } 
    107.             } catch (IOException e) { 
    108.                 e.printStackTrace(); 
    109.             } 
    110.             conn.disconnect(); 
    111.         } 
    112.         return response; 
    113.     } 
    114.  
    115.     public final  static List SearchByKeyUrl(String urlStr){ 
    116.         List result=new ArrayList<>(); 
    117.         String response= httpGet(urlStr); 
    118.         if(response==null || response==""
    119.             return result; 
    120.         ZSONObject zson=ZSONObject.stringToZSON(response); 
    121.         if(zson.getIntValue("infocode")!=10000) 
    122.             return result; 
    123.         ZSONArray zsonArray=zson.getZSONArray("pois"); 
    124.         for(int i=0;isize();i++){ 
    125.             ZSONObject child= (ZSONObject)zsonArray.get(i); 
    126.             String location=child.getString("location"); 
    127.             result.add(location); 
    128.         } 
    129.         return result; 
    130.     } 

    想了解更多内容,请访问:

    51CTO和华为官方合作共建的鸿蒙技术社区

    https://harmonyos.51cto.com

     

    来源:鸿蒙社区内容投诉

    免责声明:

    ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

    ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

    软考中级精品资料免费领

    • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

      难度     813人已做
      查看
    • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

      难度     354人已做
      查看
    • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

      难度     318人已做
      查看
    • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

      难度     435人已做
      查看
    • 2024年上半年系统架构设计师考试综合知识真题

      难度     224人已做
      查看

    相关文章

    发现更多好内容

    猜你喜欢

    AI推送时光机
    位置:首页-资讯-后端开发
    咦!没有更多了?去看看其它编程学习网 内容吧
    首页课程
    资料下载
    问答资讯