成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

HarmonyOS 項目實戰之通訊錄(Java)

開發 后端 OpenHarmony
通訊錄demo主要分為聯系人界面、設置緊急聯系人、服務卡片3個模塊,分為Java和JS兩個版本,本篇主要講解用盡可能的用Java去實現。

想了解更多內容,請訪問:

51CTO和華為官方合作共建的鴻蒙技術社區

https://harmonyos.51cto.com

1 簡介

通訊錄demo主要分為聯系人界面、設置緊急聯系人、服務卡片3個模塊,分為Java和JS兩個版本,本篇主要講解用盡可能的用Java去實現。

1.1 原型

感興趣的小伙伴,可以自己根據原型效果自己嘗試著去實現【通訊錄demo簡易原型】。

#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區

1.2 場景示例

通過學習與練習本demo,可以延伸至以下場景。

#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區

1.3 項目實戰

《HarmonyOS 項目實戰之通訊錄Demo(JS)》

《HarmonyOS 項目實戰之通訊錄(Java)》

《HarmonyOS 項目實戰之新聞頭條(ArkUI-TS》

2 功能開發

2.1 聯系人列表

2.1.1 實現效果

2.1.2 核心代碼

參考:ListContainer-常用組件開發指導-Java UI框架-UI-開發-HarmonyOS應用開發

  • ListContainer設置StickyContactProvider適配器
  • HeaderDecor頭部聯動效果設置
  • ContactData數據處理相關類,sortContactData方法用于排序等數據處理
  1. ContactData categoryData = ContactData.get(); 
  2.     categoryData.sortContactData(); 
  3.  
  4.     contactList = (ListContainer) findComponentById(ResourceTable.Id_contactList); 
  5.     Text headerText = (Text) findComponentById(ResourceTable.Id_sticky_text); 
  6.     List<ContactBean> dataList = categoryData.getResultList(); 
  7.  
  8.     mStickyContactProvider = new StickyContactProvider(this, dataList); 
  9.     contactList.setItemProvider(mStickyContactProvider); 
  10.     HeaderDecor headerDecor = new HeaderDecor(contactList, headerText); 

 sortContactData方法數據處理,排序,字母索引:

  1. public void sortContactData() { 
  2.     List<ContactBean> mContactList = new ArrayList<>(); 
  3.     Map<String, String> map = new HashMap<>(); 
  4.  
  5.     for (ContactBean contactBean : mContactBeans) { 
  6.         String pinyin = Utils.getPingYin(contactBean.getName()); 
  7.         map.put(pinyin, contactBean.getName()); 
  8.         contactBean.setNamepy(pinyin); 
  9.         mContactList.add(contactBean); 
  10.     } 
  11.     mContactList.sort(new ContactComparator()); 
  12.     characterList = new ArrayList<>(); 
  13.     resultList = new ArrayList<>(); 
  14.     for (ContactBean contactBean : mContactList) { 
  15.         String namepy = contactBean.getNamepy(); 
  16.         String character = (namepy.charAt(0) + "").toUpperCase(Locale.ENGLISH); 
  17.         if (!characterList.contains(character)) { 
  18.             if (character.hashCode() >= "A".hashCode() && character.hashCode() <= "Z".hashCode()) { // 是字母 
  19.                 characterList.add(character); 
  20.                 resultList.add(new ContactBean(character, ContactBean.ITEM_TYPE.ITEM_TYPE_CHARACTER.ordinal())); 
  21.             } else { 
  22.                 if (!characterList.contains("#")) { 
  23.                     characterList.add("#"); 
  24.                     resultList.add(new ContactBean("#", ContactBean.ITEM_TYPE.ITEM_TYPE_CHARACTER.ordinal())); 
  25.                 } 
  26.             } 
  27.         } 
  28.  
  29.         resultList.add(new ContactBean(contactBean.getName(), contactBean.getTelephone(), map.get(namepy), ContactBean.ITEM_TYPE.ITEM_TYPE_CONTACT.ordinal())); 
  30.     } 

2.2 數據的增刪改查

2.2.1 實現效果

#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區

2.2.2 增刪改查實現

ListContainer刪除實現

  1. categoryData.deleteContactBeans(item); 
  2. categoryData.sortContactData(); 
  3. mStickyContactProvider.setDataListChanged(categoryData.getResultList()); 

 隨機添加一個聯系人

  1. categoryData.addContactBean("胡六一""15269856587"); 
  2. categoryData.sortContactData(); 
  3. mStickyContactProvider.setDataListChanged(categoryData.getResultList()); 

 ContactData數據處理效果類,實現數據增刪改查

  1. // Generate the javaBean of ContactData 
  2. public static ContactData get() { 
  3.     return new ContactData(); 
  4.  
  5. public List<ContactBean> getDefaultContactBeans() { 
  6.     return mDefaultContactBeans; 
  7.  
  8. public List<ContactBean> getContactBeans() { 
  9.     return mContactBeans; 
  10.  
  11. public void addContactBean(String name, String phone) { 
  12.     mContactBeans.add(new ContactBean(name, phone)); 
  13.  
  14. public List<ContactBean> deleteContactBeans(ContactBean item) { 
  15.     mContactBeans.removeIf(contactBean -> contactBean.getName().equals(item.getName())); 
  16.  
  17.     return mContactBeans; 
  18.  
  19. public List<ContactBean> getResultList() { 
  20.     return resultList; 
  21.  
  22. public List<String> getCharacterList() { 
  23.     return characterList; 
  24.  
  25. public int getScrollPosition(String character) { 
  26.     if (characterList.contains(character)) { 
  27.         for (int i = 0; i < resultList.size(); i++) { 
  28.             if (resultList.get(i).getCharacter().equals(character)) { 
  29.                 return i; 
  30.             } 
  31.         } 
  32.     } 
  33.  
  34.     return -1; // -1不會滑動 

2.2.3 緊急聯系人數據存儲

輕量級數據存儲:輕量級數據存儲概述-輕量級數據存儲-數據管理-開發-HarmonyOS應用開發

Key-Value數據結構

一種鍵值結構數據類型。Key是不重復的關鍵字,Value是數據值。

運作機制

  • 本模塊提供輕量級數據存儲的操作類,應用通過這些操作類完成數據庫操作。
  • 借助DatabaseHelper API,應用可以將指定文件的內容加載到Preferences實例,每個文件最多有一個Preferences實例,系統會通過靜態容器將該實例存儲在內存中,直到應用主動從內存中移除該實例或者刪除該文件。
  • 獲取到文件對應的Preferences實例后,應用可以借助Preferences API,從Preferences實例中讀取數據或者將數據寫入Preferences實例,通過flush或者flushSync將Preferences實例持久化。

核心代碼實現

添加緊急聯系人,并通知java卡片更新。

  1. ZSONObject zsonObject = new ZSONObject(); 
  2. zsonObject.put("urgent1", nameTf1.getText()); 
  3. zsonObject.put("urgentPhone1", phoneTf1.getText()); 
  4. zsonObject.put("urgent2", nameTf2.getText()); 
  5. zsonObject.put("urgentPhone2", phoneTf2.getText()); 
  6. PreferenceUtils.putString(getContext(),"urgentPerson", ZSONObject.toZSONString(zsonObject)); 
  7. FormBindingData formBindingData = new FormBindingData(zsonObject); 
  8. ((ContactPersonAbility) getAbility()).confirmUpdateForm(formBindingData); 

 PreferenceUtils封裝工具類,實現數據存儲。

  1. public class PreferenceUtils { 
  2.     private static String PREFERENCE_FILE_NAME = "prefrence_file"
  3.     private static Preferences preferences; 
  4.     private static DatabaseHelper databaseHelper; 
  5.     private static Preferences.PreferencesObserver mPreferencesObserver; 
  6.  
  7.     private static void initPreference(Context context) { 
  8.         if (databaseHelper == null) { 
  9.             databaseHelper = new DatabaseHelper(context); 
  10.         } 
  11.         if (preferences == null) { 
  12.             preferences = databaseHelper.getPreferences(PREFERENCE_FILE_NAME); 
  13.         } 
  14.     } 
  15.  
  16.     //存放、獲取時傳入的context必須是同一個context,否則存入的數據無法獲取 
  17.     public static void putString(Context context, String key, String value) { 
  18.         initPreference(context); 
  19.         preferences.putString(key, value); 
  20.         preferences.flush(); 
  21.     } 
  22.  
  23.     /** 
  24.      * @param context 上下文 
  25.      * @param key     鍵 
  26.      * @return 獲取的String 默認值為:null 
  27.      */ 
  28.     public static String getString(Context context, String key) { 
  29.         initPreference(context); 
  30.         return preferences.getString(keynull); 
  31.     } 
  32.  
  33.     public static String getString(Context context, String key, String d) { 
  34.         initPreference(context); 
  35.         return preferences.getString(key, d); 
  36.     } 
  37.  
  38.     public static void putInt(Context context, String keyint value) { 
  39.         initPreference(context); 
  40.         preferences.putInt(key, value); 
  41.         preferences.flush(); 
  42.     } 
  43.  
  44.     /** 
  45.      * @param context 上下文 
  46.      * @param key     鍵 
  47.      * @return 獲取int的默認值為:-1 
  48.      */ 
  49.     public static int getInt(Context context, String key) { 
  50.         initPreference(context); 
  51.         return preferences.getInt(key, -1); 
  52.     } 
  53.     public static void putLong(Context context, String key, long value) { 
  54.         initPreference(context); 
  55.         preferences.putLong(key, value); 
  56.         preferences.flush(); 
  57.     } 
  58.  
  59.     /** 
  60.      * @param context 上下文 
  61.      * @param key     鍵 
  62.      * @return 獲取long的默認值為:-1 
  63.      */ 
  64.     public static long getLong(Context context, String key) { 
  65.         initPreference(context); 
  66.         return preferences.getLong(key, -1L); 
  67.     } 
  68.  
  69.     public static void putBoolean(Context context, String key, boolean value) { 
  70.         initPreference(context); 
  71.         preferences.putBoolean(key, value); 
  72.         preferences.flush(); 
  73.     } 
  74.  
  75.     /** 
  76.      * @param context 上下文 
  77.      * @param key     鍵 
  78.      * @return 獲取boolean的默認值為:false 
  79.      */ 
  80.     public static boolean getBoolean(Context context, String key) { 
  81.         initPreference(context); 
  82.         return preferences.getBoolean(keyfalse); 
  83.     } 
  84.  
  85.     public static void putFloat(Context context, String keyfloat value) { 
  86.         initPreference(context); 
  87.         preferences.putFloat(key, value); 
  88.         preferences.flush(); 
  89.     } 
  90.  
  91.     /** 
  92.      * @param context 上下文 
  93.      * @param key     鍵 
  94.      * @return 獲取float的默認值為:0.0 
  95.      */ 
  96.     public static float getFloat(Context context, String key) { 
  97.         initPreference(context); 
  98.         return preferences.getFloat(key, 0.0F); 
  99.     } 
  100.  
  101.     public static void putStringSet(Context context, String keySet<String> set) { 
  102.         initPreference(context); 
  103.         preferences.putStringSet(keyset); 
  104.         preferences.flush(); 
  105.     } 
  106.  
  107.     /** 
  108.      * @param context 上下文 
  109.      * @param key     鍵 
  110.      * @return 獲取set集合的默認值為:null 
  111.      */ 
  112.     public static Set<String> getStringSet(Context context, String key) { 
  113.         initPreference(context); 
  114.         return preferences.getStringSet(keynull); 
  115.     } 
  116.  
  117.     public static boolean deletePreferences(Context context) { 
  118.         initPreference(context); 
  119.         boolean isDelete = databaseHelper.deletePreferences(PREFERENCE_FILE_NAME); 
  120.         return isDelete; 
  121.     } 
  122.  
  123.     public static void registerObserver(Context context, Preferences.PreferencesObserver preferencesObserver) { 
  124.         initPreference(context); 
  125.         mPreferencesObserver = preferencesObserver; 
  126.         preferences.registerObserver(mPreferencesObserver); 
  127.     } 
  128.  
  129.     public static void unregisterObserver() { 
  130.         if (mPreferencesObserver != null) { 
  131.             // 向preferences實例注銷觀察者 
  132.             preferences.unregisterObserver(mPreferencesObserver); 
  133.         } 
  134.     } 

2.3 第三方跳轉

2.3.1 實現效果

#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區

2.3.2 撥打電話與發送短信

  1. /** 
  2.      * 跳轉系統短信 
  3.      */ 
  4.     private void doMessage(String telephone) { 
  5.         Intent intent = new Intent(); 
  6.         Operation operation = new Intent.OperationBuilder() 
  7. //                .withAction("android.intent.action.SENDTO") // Android寫法 android.intent.action.SENDTO 
  8.                 .withAction(IntentConstants.ACTION_SEND_SMS) 
  9.                 .withUri(Uri.parse("smsto:" + telephone)) // 設置號碼 
  10.                 .withFlags(Intent.FLAG_NOT_OHOS_COMPONENT) 
  11.                 .build(); 
  12.         intent.setOperation(operation); 
  13.         context.startAbility(intent, 11); 
  14.     } 
  15.  
  16.     /** 
  17.      * 申請撥打電話權限 
  18.      */ 
  19.     private boolean requestPermissions() { 
  20.         if (context.verifySelfPermission("android.permission.CALL_PHONE") != IBundleManager.PERMISSION_GRANTED) { 
  21.             // 應用未被授予權限 
  22.             if (context.canRequestPermission("android.permission.CALL_PHONE")) { 
  23.                 // 是否可以申請彈框授權(首次申請或者用戶未選擇禁止且不再提示) 
  24.                 context.requestPermissionsFromUser(new String[]{"android.permission.CALL_PHONE"}, 100); 
  25.             } 
  26.             return false
  27.         } else { 
  28.             // 權限已被授予 
  29.             return true
  30.         } 
  31.     } 
  32.  
  33.     /** 
  34.      * 直接撥打電話 
  35.      * 需要申請權限 
  36.      */ 
  37.     private void doCall(String destinationNum) { 
  38.         if (!requestPermissions()) { 
  39.             return
  40.         } 
  41.         Intent intent = new Intent(); 
  42.         Operation operation = new Intent.OperationBuilder() 
  43.                 .withAction("android.intent.action.CALL") // 系統應用撥號盤 
  44.                 .withUri(Uri.parse("tel:" + destinationNum)) // 設置號碼 
  45.                 .withFlags(2) 
  46.                 .build(); 
  47.         intent.setOperation(operation); 
  48.         // 啟動Ability 
  49.         context.startAbility(intent, 10); 
  50.  
  51.     } 
  52.  
  53.     /** 
  54.      * 跳轉系統撥打電話界面 
  55.      */ 
  56.     private void doDial(String destinationNum) { 
  57.         Intent intent = new Intent(); 
  58.         Operation operation = new Intent.OperationBuilder() 
  59.                 .withAction(IntentConstants.ACTION_DIAL) // 系統應用撥號盤 
  60. //                .withBundleName(context.getCallingBundle()) // 應用撥號選擇器 
  61.                 .withUri(Uri.parse("tel:" + destinationNum)) // 設置號碼 
  62.                 .withFlags(2) 
  63.                 .build(); 
  64.         intent.setOperation(operation); 
  65.         // 啟動Ability 
  66.         context.startAbility(intent, 10); 
  67.  
  68.     } 

2.4 JS服務卡片

2.4.1 實現效果

#星光計劃2.0# HarmonyOS 項目實戰之通訊錄(Java)-鴻蒙HarmonyOS技術社區

2.4.2 創建卡片模板

使用DevEco Studio創建卡片工程

創建成功后,在config.json的module中會生成js模塊,用于對應卡片的js相關資源,配置示例如下:

  1. "js": [ 
  2.   { 
  3.     "pages": [ 
  4.       "pages/index/index" 
  5.     ], 
  6.     "name""widget"
  7.     "window": { 
  8.       "designWidth": 720, 
  9.       "autoDesignWidth"true 
  10.     }, 
  11.     "type""form" 
  12.   } 

 config.json文件“abilities”配置forms模塊細節如下:

  1. "name""com.huhu.contact.ContactPersonAbility"
  2. "icon""$media:icon"
  3. "description""$string:contactpersonability_description"
  4. "formsEnabled"true
  5. "label""$string:contact_ContactPersonAbility"
  6. "type""page"
  7. "forms": [ 
  8.   { 
  9.     "jsComponentName""widget"
  10.     "isDefault"true
  11.     "scheduledUpdateTime""10:30"
  12.     "defaultDimension""2*2"
  13.     "name""widget"
  14.     "description""This is a service widget"
  15.     "colorMode""auto"
  16.     "type""JS"
  17.     "supportDimensions": [ 
  18.       "2*2" 
  19.     ], 
  20.     "updateEnabled"true
  21.     "updateDuration": 1 
  22.   } 

 創建一個ContactPersonAbility,覆寫卡片相關回調函數。

  • onCreateForm(Intent intent)
  • onUpdateForm(long formId)
  • onDeleteForm(long formId)
  • onCastTempForm(long formId)
  • onEventNotify(Map
  • onTriggerFormEvent(long formId, String message)
  • onAcquireFormState(Intent intent)

當卡片使用方請求獲取卡片時,卡片提供方會被拉起并調用onCreateForm(Intent intent)回調,intent中會帶有卡片ID、卡片名稱和卡片外觀規格信息,可按需獲取使用。

開發JS卡片時,FormAbility可以繼承AceAbility或Ability,繼承Ability時,需在onStart()方法中額外設置路由信息。

2.4.3 卡片數據綁定

  1. @Override 
  2.  public ProviderFormInfo bindFormData() { 
  3.      HiLog.info(TAG, "bind form data"); 
  4.      ProviderFormInfo providerFormInfo = new ProviderFormInfo(); 
  5.      String urgentPersonStr = PreferenceUtils.getString(context, "urgentPerson"""); 
  6.      ZSONObject zsonObject = ZSONObject.stringToZSON(urgentPersonStr); 
  7.      if (dimension == DEFAULT_DIMENSION_2X2) { 
  8.          if (zsonObject != null) { 
  9.              providerFormInfo.setJsBindingData(new FormBindingData(zsonObject)); 
  10.          } 
  11.      } 
  12.      return providerFormInfo; 
  13.  } 

 2.4.4 卡片數據更新

  1. public void confirmUpdateForm(FormBindingData formBindingData) { 
  2.     FormControllerManager formControllerManager = FormControllerManager.getInstance(this); 
  3.     List<Long> allFormIdFromSharePreference = formControllerManager.getAllFormIdFromSharePreference(); 
  4.     if (allFormIdFromSharePreference == null || allFormIdFromSharePreference.isEmpty()) return
  5.     Long formId = allFormIdFromSharePreference.get(0); 
  6.     try { 
  7.         updateForm(formId,formBindingData); 
  8.     } catch (FormException e) { 
  9.         e.printStackTrace(); 
  10.     } 

 2.4.5 卡片事件處理

  1.   "data": { 
  2.     "text_content""Name"
  3.     "cardPrimaryText""Contacts"
  4.     "cardSecondaryText""+8612345678912"
  5.     "urgent1""無"
  6.     "urgent2""無"
  7.     "urgentPhone1""+8612345678912"
  8.     "urgentPhone2""+8612345678915" 
  9.   }, 
  10.   "actions": { 
  11.     "urgentCall1": { 
  12.       "action""message"
  13.       "params": { 
  14.         "action""urgentCall1"
  15.         "phoneNumber""10086" 
  16.       } 
  17.     }, 
  18.     "urgentCall2": { 
  19.       "action""message"
  20.       "params": { 
  21.         "action""urgentCall2"
  22.         "phoneNumber""15565339857" 
  23.       } 
  24.     }, 
  25.     "startMainRouter": { 
  26.       "action""router"
  27.       "abilityName""com.huhu.contact.ContactPersonAbility" 
  28.     } 
  29.   } 

卡片支持觸發事件,覆寫onTriggerFormEvent方法實現對事件的觸發,doCall就是前面的播打電話的方法

  1. @Override 
  2. protected void onTriggerFormEvent(long formId, String message) { 
  3.     super.onTriggerFormEvent(formId, message); 
  4.     HiLog.info(loglabel, "onTriggerFormEvent: " + message); 
  5.     FormControllerManager formControllerManager = FormControllerManager.getInstance(this); 
  6.     FormController formController = formControllerManager.getController(formId); 
  7.     formController.onTriggerFormEvent(formId, message); 
  8.     ZSONObject params = ZSONObject.stringToZSON(message); 
  9.     String action = params.getString("action"); 
  10.     String phoneNumber = params.getString("phoneNumber"); 
  11.     HiLog.info(loglabel, "onTriggerFormEvent: action:" + action); 
  12.  
  13.     String urgentPersonStr = PreferenceUtils.getString(this, "urgentPerson"""); 
  14.     ZSONObject zsonObject = ZSONObject.stringToZSON(urgentPersonStr); 
  15.     switch (action) { 
  16.         case "urgentCall1"
  17.             String urgentPhone1 = zsonObject.getString("urgentPhone1"); 
  18.             doCall(urgentPhone1); 
  19.             break; 
  20.         case "urgentCall2"
  21.             String urgentPhone2 = zsonObject.getString("urgentPhone2"); 
  22.             doCall(urgentPhone2); 
  23.             break; 
  24.         default
  25.             break; 
  26.     } 

3 注意事項

Demo還有很多需要完善的地方

  • 滑動時,上滑頭部聯動效果,索引有時會錯亂;
  • 搜索功能未實現;
  • 緊急聯系人沒有和列表數據聯動。

4 總結

代碼地址: https://gitee.com/hu-lingqing/contact-person.git

想了解更多內容,請訪問:

51CTO和華為官方合作共建的鴻蒙技術社區

https://harmonyos.51cto.com

責任編輯:jianghua 來源: 鴻蒙社區
相關推薦

2021-12-10 10:12:44

鴻蒙HarmonyOS應用

2011-09-05 14:08:21

微信Andriod安卓

2014-09-24 10:29:14

微信企業號開發

2011-09-16 14:05:42

Andrioid應用iPhone應用Symbian應用

2014-09-28 22:30:13

微信企業號

2014-09-28 22:26:11

微信企業號

2012-02-09 09:10:44

Path通訊錄隱私

2010-06-18 22:42:42

智能手機平臺Android網秦

2013-03-18 10:19:27

安卓軟件手機通訊錄隱私信息

2012-06-05 13:53:03

天天聯系華為

2011-07-20 08:49:24

jQuery MobiAndroid

2010-09-26 08:35:01

火種通訊錄

2012-12-21 14:51:52

手機中國

2010-09-08 23:11:01

2015-07-30 15:58:15

EC企信企業即時通訊

2010-05-12 14:42:20

2011-08-12 10:16:10

iPhone通訊錄聯系人

2011-11-28 14:37:32

點心通訊錄

2012-01-09 16:43:13

點心通訊錄

2011-07-19 17:25:14

jQuery MobiAndroid
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 中文字幕免费 | 久久久久久九九九九九九 | 盗摄精品av一区二区三区 | 中文字幕电影在线观看 | 久久小视频 | 日韩一区二区免费视频 | 日韩第一区 | 国产在线精品一区二区三区 | 视频一区中文字幕 | 国产成人99久久亚洲综合精品 | 欧美xxxx黑人又粗又长 | 久久久国产一区 | 久久9视频 | 九色91视频 | 欧美精品一区二区三区蜜臀 | 精品一区在线免费观看 | 丝袜美腿一区二区三区动态图 | 中文av在线播放 | 天天插天天操 | 新疆少妇videos高潮 | 亚洲国产精品成人无久久精品 | 成人国产精品久久 | 91 在线| 亚洲第一天堂 | 亚洲国产精品日韩av不卡在线 | 狠狠av | 日韩欧美二区 | 成年人在线视频 | 国产午夜三级一区二区三 | 久久精品久久久久久 | 亚洲国产欧美日韩 | 免费视频一区二区 | 91中文字幕 | www.久久| 国产精品永久免费视频 | www日本高清 | 亚洲欧美一区二区三区国产精品 | 国产精品视频久久 | 国产精品久久国产精品 | 国产成人自拍一区 | 成人伊人 |