超碰91资源站-超碰97豆花-超碰97人妻-超碰97人人干-超碰97人人香蕉-超碰97天天操-超碰97在线资源站-超碰97资源站共享-超碰97资源站总站-超碰aa在线91-超碰av操-超碰爱爱

半岛外围网上直营

3D格式轉換神器HOOPS Exchange使用教程(二):檢索可視化工作流的圖形數據

原創|3D產品功能介紹|編輯:何家巧|2022-08-31 11:45:40.093|閱讀 236 次

概述:本教程將說明如何使用 HOOPS Exchange 檢索可視化工作流的圖形數據。完成本教程后,您將對 HOOPS Exchange 如何提供對零件三角形網格的訪問、如何在 3D 空間中正確定位以及如何確定每個零件的基本顏色有一個基本的了解。

# 界面/圖表報表/文檔/IDE等千款熱門軟控件火熱銷售中 >>

相關鏈接:

HOOPS Exchange是什么?

 是一組軟件庫,可以幫助開發人員在開發應用程序時讀取和寫入主流的 2D 和 3D 格式。HOOPS Exchange 支持在主流的3D 文件格式中讀取 CAD 數據,并支持將 3D 數據轉換為 PRC 數據格式,這是一種高度可壓縮和開放的文件格式,并已通過國際標準化組織 (ISO 14739-1:2014) 的認證。PRC 也是 Adobe PDF 中用于 3D 的格式之一。HOOPS Exchange 持續優化讀取各種 3D 數據的功能,尤其是對于來自計算機輔助設計 (CAD) 系統的數據。


本章我們學習創建一個使用 加載文件并使用 Qt3D 將其可視化的跨平臺應用程序。

介紹
本教程將向大家說明如何使用 檢索可視化工作流的圖形數據。學習完本教程后,您將對 HOOPS Exchange 如何提供對零件三角形網格的訪問、如何在 3D 空間中正確定位它們以及如何確定每個零件的基本顏色有一個基本的了解。
本教程有一些先決條件。首先,您應該已經完成了“打印裝配結構”教程,該教程涵蓋了文件加載和數據檢索等幾個基本概念,這些話題在此不再贅述。
HOOPS Exchange 是一個支持 Windows、macOS 和 Linux 的 SDK。我們將使用最流行的跨平臺 GUI 工具包 Qt,具體來說,我們將依賴 Qt3D 來實現跨平臺的圖形功能。我們將盡一切努力將工具包所需的專業知識降至最低,但是,您必須在計算機上安裝 Qt 6才能完成本教程。
像許多跨平臺開發社區一樣,Qt 已經開始向使用 CMake 作為默認構建系統的方向遷移。可以在此處找到有關使用 CMake 構建 Qt 應用程序的信息。本教程包括基于這些概念的完整 CMakeLists.txt 文件。Qt 的最新發行版包括 bin/qt-cmake,如果您尚未安裝 CMake,則可以使用它們。
不需要深入了解 Qt 和 CMake,但兩者都必須安裝并準備好使用。
第 0 步:項目設置
克隆項目
我們提供了一個 git 存儲庫來支持本教程。克隆主分支以建立項目的起點。
git 克隆 //github.com/techsoft3d/he_qt_basic_view.git
配置
使用您喜歡的文本編輯器打開文件CMakeLists.txt。在文件的頂部,您將看到HOOPS_EXCHANGE_DIR已設置變量。更新分配給此變量的值以反映您的特定安裝位置。
建造
由于本教程的目標是提供對 HOOPS Exchange 的理解,因此我們不會花太多時間在如何構建和運行 Qt 應用程序或 IDE 選擇和配置的主題上。但以防萬一您不熟悉它是如何完成的,我們將在此處提供一些提示。
視覺工作室代碼
Visual Studio Code 是跨平臺開發的絕佳選擇。它支持 C/C++ 開發和 CMake 作為構建配置系統。Microsoft在此處提供了此用例的出色概述。
編輯文件 _.vscode/settings.json_ 并更新 Qt 路徑以反映您本地安裝的 Qt。安裝 CMake Tools 擴展后,您可以使用狀態欄上的按鈕來配置、構建和運行應用程序。
Windows 上的 Visual C++
打開 Visual Studio 命令提示符并執行位于 Qt 安裝的 bin 文件夾中的 qtenv2.bat。接下來,在項目目錄中創建一個名為build的子文件夾并更改為它。運行qt-cmake ..以生成所需的文件。這將創建qt_he_viewer.sln,您可以使用命令evenv qt_he_viewer.sln 打開它。
開始運作
構建項目后,您就可以運行應用程序了。當您運行二進制文件時,您將看到一個標準的文件打開對話框。對話框的默認位置是包含 HOOPS Exchange 附帶的示例數據的文件夾。導航到 PRC 子文件夾并選擇helloworld.prc。該文件加載迅速,并出現空的 3D 視圖。
查看main.cpp的實現以熟悉程序流程。您會注意到 HOOPS Exchange 已初始化,并提示用戶輸入一個輸入文件,然后加載該文件。加載文件后,代碼繼續調用createScene,配置視圖、相機和光源。
我們將從創建場景開始,以一種有點抽象的方式。
第 1 步:創建場景
要創建場景,我們必須實現Scene.cppcreateScene中定義的函數。在編輯器中打開文件。你會注意到它被存根返回一個空對象。
在 HOOPS Exchange 數據模型中,曲面細分存在于表示項級別。這意味著我們將需要實現遍歷裝配結構、輸入每個零件定義并提取其中包含的表示項的功能。對于我們遇到的每個表示項目,我們需要做一些事情:
  1. 確定是否應顯示表示項。
  2. 生成我們可以輕松渲染的細分數據。
  3. Qt3D從 細分創建網格。
  4. Qt3D從HOOPS Exchange 樣式定義創建材質。
  5. Qt3D從世界位置創建一個變換。
我們剛剛列出的所有功能都已在您克隆的項目中被刪除,因此我們可以編寫完整的 createScene 主體,而無需過多關注每個步驟的實現方式。
首先,我們將聲明并初始化一個結構來控制如何為表示項生成鑲嵌。創建后添加以下代碼行rootEntity.
// 創建曲面細分參數來控制行為
A3DRWParamsTessellationData tess_params;
A3D_INITIALIZE_DATA(A3DRWParamsTessellationData, tess_params);
// 使用“預設”選項獲得中等詳細程度
tess_params.m_eTessellationLevelOfDetail = kA3DTessLODMedium;
為簡單起見,我們在 options 結構中使用詳細級別枚舉,它控制一組特定的細分選項。這適用于基本的查看工作流程。我們將很快使用這個選項對象。
forEach_RepresentationItem接下來,我們將使用稍后實現的函數來迭代每個表示項。現在,讓我們假設它存在并且做我們想做的事——也就是說,它遍歷裝配結構,并且對于它遇到的每個零件,它都提取表示項。對于每個表示項,調用提供的 lambda。設置細分參數后添加以下代碼行。
// 遍歷每個表示項
forEach_RepresentationItem(model_file, [&](EntityArray const &path) {
});
lambda 的參數是 an EntityArray,,它是 的類型別名QVector<A3DEntity*>。它包含指向程序集層次結構中每個節點的有序指針列表。數組中的第一項是模型文件,然后是一系列產品,然后是零件。最后,數組以遇到的表示項結束。
對于這一步的其余部分,我們將按順序將代碼添加到 lambda 的主體中。
有時不應繪制表示項。為了確定這一點,我們將使用一種稱為級聯屬性的機制。級聯屬性允許我們在實例化它的組件的上下文中計算零件的屬性。特定裝配可以覆蓋特定零件的顏色或可見性。我們將把我們對級聯屬性的使用封裝在一個名為的簡單結構CascadedAttributes中,稍后我們將實現該結構。它被淘汰了,所以現在讓我們假設它的行為符合我們的需要。
在 lambda 的主體中添加以下代碼行:
CascadedAttributes ca( 路徑 );
// 確定是否應該跳過此項
如果( ca->m_bRemoved || !ca->m_bShow ) {
  返回
}
CascadedAttributes重載,提供對其中包含的結構operator->的直接訪問。A3DMiscCascadedAttributesData如果表示項目的這個實例被刪除或不應該顯示,我們會提前退出。
如果我們不及早退出,下一步就是在 Exchange 中生成曲面細分。為此,我們添加以下代碼行:
A3DRiRepresentationItem *ri = path.back();
// 使用我們上面聲明的選項生成曲面細分
A3DRiRepresentationItemComputeTessellation(ri, &tess_params);
現在我們已經對表示項進行了細分,我們可以訪問數據。
// 獲取此表示項的數據
A3DRiRepresentationItemData擺脫;
A3D_INITIALIZE_DATA(A3DRiRepresentationItemData,擺脫);
if ( A3D_SUCCESS != A3DRiRepresentationItemGet( ri, &rid ) ) {
  返回
}
// 曲面細分存儲在 m_pTessBase 中
自動tess_base = rid.m_pTessBase;
您應該非常熟悉上面介紹的模式,它使用不透明的對象句柄 ( ri) 將其關聯數據讀入結構。然后從結構中獲得鑲嵌句柄,我們就可以使用它了。
使用曲面細分的句柄,我們接下來嘗試創建一個Qt3D網格。如果我們成功了,我們就會創造并應用它的材料并進行轉換。這是通過以下方式完成的,使用了一些已經被刪除的附加函數:
// 創建網格
如果(自動網格= createMesh(tess_base)){
  自動節點 =新Qt3DCore::QEntity(rootEntity);
  節點->添加組件(網格);
  // 創建材質
  如果自動材料= createMaterial(ca->m_sStyle)){
    節點->添加組件(材料);
  }
  // 創建變換
  如果自動變換 = createTransform(路徑)){
    節點->添加組件(變換);
  }
}
如果獲得了網格,我們將創建一個節點來保存它,以及材質和變換。該節點是rootEntity.
仍然在 lambda 的主體內工作,我們還有最后一項任務。回想一下,每當您從 Exchange 讀取數據時,您必須確保通過第二次調用 getter 并提供空句柄來釋放任何關聯的內存。
使用 lambda 主體內的以下(也是最終)代碼行釋放表示項數據:
A3DRiRepresentationItemGet( nullptr , &rid);
這樣就完成了構建場景的高層實現。我們顯然為以后的步驟留下了許多實現細節,但我們已經完成了構成渲染模型所需的基本場景圖的任務。
第 2 步:程序集遍歷
從上一步來看,應該有點清楚還剩下什么要做。我們將以系統的方式攻擊每個任務,首先通過實現 ForEach_RepresentationItem 遍歷程序集層次結構。
讓我們從函數必須如何運行的簡短描述開始。在您的編輯器中打開文件 ForEachRepresentationItem.cpp,您將找到代碼的存根版本:
命名空間{
  void forEach_Impl( EntityArray const &path, std::function< void (EntityArray
  常量&)>常量&fcn ) {
    Q_UNUSED(路徑);
    Q_UNUSED(fcn);
  }
}
無效forEach_RepresentationItem(A3DAsmModelFile *model_file,
std::function< void (EntityArray const &)> const &fcn ) {
  forEach_Impl( { model_file }, fcn );
}
該函數有兩個參數。第一個是模型文件的不透明句柄。第二個參數是作為回調調用的函數對象。并且,正如我們在第 1 步中所討論的,實現預計將遍歷裝配結構并為遇到的每個表示項調用回調。
回調函數使用單個參數調用:一個EntityArray包含 Exchange 對象的不透明句柄的有序列表。該列表是順序的,從A3DAsmModelFile句柄開始,然后是一個或多個A3DAsmProductOccurrence句柄。句柄代表通向零件的裝配層次。當然,接下來就是A3DAsmPartDefinition手柄了。最后,路徑包含A3DRiRepresentationItem遇到的句柄。如果部件定義包含A3DRiSet對象(表示項集),則路徑中將有多個A3DRiRepresentationItem句柄。
公共函數立即調用一個匿名實現,該實現采用一個EntityArray而不是一個A3DAsmModelFile句柄。這樣做的用處很快就會變得清晰。該實現將只關心提供的路徑中的最后一個句柄。
一個很好的起點是一開始。所以,讓我們實現我們已經知道的情況——當這個函數被路徑中的單個對象調用時,它是一個A3DAsmModelFile句柄。在這種情況下,我們希望將每個子A3DAsmProductOccurrence句柄添加到路徑并再次調用該函數以進行更深入的挖掘。它應該看起來像這樣:
auto  const ntt = path.back();
自動類型 = kA3DTypeUnknown;
if (A3D_SUCCESS != A3DEntityGetType(ntt, &type) ) {
  返回
}
EntityArray children;
如果(kA3DTypeAsmModelFile == 類型){
  A3DAsmModelFileData mfd;
 ; A3D_INITIALIZE_DATA(A3DAsmModelFileData, mfd);
  如果(A3D_SUCCESS!= A3DAsmModelFileGet(ntt,&mfd)){
    返回
  }
  children = EntityArray(mfd.m_ppPOOccurrences,mfd.m_ppPOOccurrences +
  mfd.m_uiPOOccurrencesSize);
  A3DAsmModelFileGet( nullptr , &mfd);
}
對于(auto child : children ){
  自動child_path = 路徑;
  child_path.push_back(children auto child : children);
  forEach_Impl(child_path, fcn);
}
A3DAsmProductOccurrence此實現是遞歸的,并使用句柄作為 的值調用自身path.back()。讓我們通過添加 if 子句來擴充處理這種情況的代碼。

否則 if ( kA3DTypeAsmProductOccurrence == type ) {
  A3DAsmProductOccurrenceData 吊艙;
  A3D_INITIALIZE_DATA(A3DAsmProductOccurrenceData, pod);
&nbsp; if (A3D_SUCCESS != A3DAsmProductOccurrenceGet(ntt, &pod) ) {
    返回;
  }
 child = EntityArray( pod.m_ppPOccurrences, pod.m_ppPOccurrences +
  pod.m_uiPOOccurrencesSize );
  A3DAsmProductOccurrenceGet( nullptr , &pod);
}
從這里去哪里?這將處理整個裝配層次結構,直至節點包含零件。所以,除了上面實現中所示的處理children外,我們還必須檢查an是否A3DAsmProductOccurrence包含一個part。
確定零件是否存在有時就像檢查m_pPart產品出現結構中的字段一樣簡單。但這并沒有捕捉到共享部件實例化的常見情況。零件實例化是通過使用m_pPrototype句柄來實現的,該句柄引用了裝配節點的共享定義。如果一個節點有一個空m_pPart句柄,你還必須遞歸檢查它的原型,如果它有一個。要實現此邏輯,請在匿名命名空間的頂部添加 getPart 函數。
A3DAsmPartDefinition *getPart( A3DAsmProductOccurrence *po ) {
  if ( nullptr == po ) {
    返回 空指針;
  }
  A3DAsmProductOccurrenceData 吊艙;
  A3D_INITIALIZE_DATA(A3DAsmProductOccurrenceData, pod);
  if (A3D_SUCCESS != A3DAsmProductOccurrenceGet( po, &pod ) ) {
    返回 空指針
  }
  汽車零件 = pod.m_pPart ?pod.m_pPart : getPart( pod.m_pPrototype );
  A3DAsmProductOccurrenceGet( nullptr , &pod);
  返回部分;
  }
現在,我們可以在剛剛添加的處理A3DAsmPartDefinition對象的子句中使用這個函數:
否則 if ( kA3DTypeAsmProductOccurrence == type ) {
  A3DAsmProductOccurrenceData 吊艙;
  A3D_INITIALIZE_DATA(A3DAsmProductOccurrenceData, pod);
&nbsp; if (A3D_SUCCESS != A3DAsmProductOccurrenceGet(ntt, &pod) ) {
    返回;
  }
  孩子 = EntityArray( pod.m_ppPOccurrences, pod.m_ppPOccurrences +
  pod.m_uiPOOccurrencesSize );
  如果(汽車零件= pod.m_pPart?pod.m_pPart:getPart(pod.m_pPrototype)){
&nbsp;   children.insert(children.begin(), part);
  }
  A3DAsmProductOccurrenceGet( nullptr , &pod);
}
我們已經完成了零件定義!所以讓我們在子句中添加部分定義遍歷:
} else  if ( kA3DTypeAsmPartDefinition == type ) {
  A3DAsmPartDefinitionData pdd;
  A3D_INITIALIZE_DATA(A3DAsmPartDefinitionData, pdd);
  if (A3D_SUCCESS != A3DAsmPartDefinitionGet(ntt, &pdd) ) {
    返回
  }
 children = EntityArray(pdd.m_ppRepItems,pdd.m_ppRepItems +
  pdd.m_uiRepItemsSize );
  A3DAsmPartDefinitionGet( nullptr , &pdd);
將我們帶到表示項目上,我們應該在其中調用回調函數,提供用于將我們帶到這里的路徑。但在我們這樣做之前,我們不能忘記作為集合的特定表示項類型。如果遇到這種對象類型,我們必須進一步遍歷。
處理所有這些細節應該看起來像這樣,作為條件的最后一個 else 子句:
 否則{
  如果(kA3DTypeRiSet == 類型){
    A3DRiSetData risd;
    A3D_INITIALIZE_DATA(A3DRiSetData, risd);
    if (A3D_SUCCESS != A3DRiSetGet(ntt, &risd) ) {
      返回
    }
    children = EntityArray(risd.m_ppRepItems, risd.m_ppRepItems + risd.m_uiRepItemsSize);
    A3DRiSetGet( nullptr , &risd);
  }其他{
    fcn(路徑);
  }
}
如果您現在感覺有點頭暈,請不要擔心,這是完全正常的。我們一起成功地實現了一個行為良好的函數,用于以對我們非常有用的方式遍歷 Exchange 產品結構。通過使用函數對象,我們將遍歷與構建場景圖的工作分開。在此過程中,您可能已經對 Exchange 的數據結構有所了解。
第 3 步:級聯屬性
讓我們繼續實現我們在步驟 1 中創建場景時使用的每個函數。我們遇到的下一個存根函數是 lambda 內部的CascadedAttributes結構。此結構在文件CascadedAddtributes.h中實現。打開它看看。您將找到一個空的構造函數和析構函數,我們現在將實現它們。
構造函數有一個參數,你現在應該很熟悉了。它是一個 EntityArray,表示從模型文件到我們感興趣的表示項的 Exchange 對象的路徑。我們的構造函數的工作是計算A3DMiscCascadedAttributesData與該路徑對應的對象。我們將按照此處的編程指南關于級聯屬性的部分提供的指導來執行此操作。
實現構造函數如下:
// 創建一個向量來保存級聯屬性句柄
QVector<A3DMiscCascadedAttributes*> cascaded_attribs;
// 創建“根”級聯屬性句柄
cascaded_attribs.push_back( nullptr );
A3DMiscCascadedAttributesCreate( &cascaded_attribs.back() );
// 對于路徑中的每個實體
對于(自動ntt:路徑){
  如果(A3DEntityIsBaseWithGraphicsType(ntt)){
    // 獲取之前級聯屬性的句柄
    自動父親 = cascaded_attribs.back();

// 為這個實體創建一個新的級聯屬性句柄

    cascaded_attribs.push_back( nullptr );
    A3DMiscCascadedAttributesCreate( &cascaded_attribs.back() );

    // 將此句柄壓入堆棧
    A3DMiscCascadedAttributesPush( cascaded_attribs.back(), ntt, 父親);
  }
}
// 計算級聯屬性數據
A3D_INITIALIZE_DATA(A3DMiscCascadedAttributesData, d);
A3DMiscCascadedAttributesGet( cascaded_attribs.back(), &d );
對于自動屬性:cascaded_attribs){
&nbsp;   A3DMiscCascadedAttributesDelete(attrib);
}
代碼中的注釋應該合理地解釋方法是什么。
一旦構造了這個對象,我們就適當地填充了數據字段。剩下要做的就是釋放析構函數中的對象。將這行代碼添加到析構函數中:
A3DMiscCascadedAttributesGet( nullptr , &d);
僅此而已。
完成此步驟意味著您已經創建了一個簡單的結構來管理任意 EntityArray 的級聯屬性。這與我們工作流程的其余部分很好地結合在一起,并直接利用了我們實現的方法來遍歷產品結構。
第 4 步:創建網格
在下一步中,我們將介紹從 HOOPS Exchange 讀取曲面細分所需的代碼,并創建Qt3D適合渲染的相應對象。這項工作將在文件中完成Mesh.cpp。現在在你的編輯器中打開它,你會發現熟悉的 stubbed out 實現。
要開始這項任務,我們應該對傳入的句柄執行一些健全性檢查。具體來說,我們要確保它是我們要為這個基本查看工作流處理的正確的具體對象類型。
A3DEEntityType tess_type = kA3DTypeUnknown;
if (A3D_SUCCESS != A3DEntityGetType( tess_base, &tess_type ) ) {
  返回 空指針
}
// 確保我們只處理我們關心的類型
如果(苔絲類型!= kA3DTypeTess3D){
  返回 空指針
}
傳遞給函數的句柄是一個名為的基類型A3DTessBase.對于這個基本的查看工作流,我們將只處理具體類型A3DTess3D.如果傳入一個空句柄,此代碼將正確處理它并退出。
基本鑲嵌類型包含我們需要的所有派生類型共有的信息,特別是坐標數組。添加代碼以從 HOOPS Exchange 讀取基礎數據。
// 從 tess 基礎數據中讀取坐標數組
A3DTessBaseData 待定;
A3D_INITIALIZE_DATA(A3DTessBaseData,待定);
if ( A3D_SUCCESS != A3DTessBaseGet( tess_base, &tbd ) ) {
  返回 空指針;
}
A3DDouble const *coords = tbd.m_pdCoords;
A3DUns32 const n_coords = tbd.m_uiCoordSize;
坐標數據以 C 樣式數組的形式提供 - 也就是說,它是一個指向指定長度的雙精度數組的指針。大小總是能被 3 整除。
下一個任務是獲取與具體細分類型相關的數據。我們將從獲取法線向量的 C 樣式數組開始。
3DTess3D數據 t3dd;
A3D_INITIALIZE_DATA(A3DTess3DData, t3dd);
if ( A3D_SUCCESS != A3DTess3DGet( tess_base, &t3dd ) ) {
A3DTessBaseGet( nullptr , &tbd);
  返回 空指針;
}
A3DDouble const *normals = t3dd.m_pdNormals;
A3DUns32 const n_normals = t3dd.m_uiNormalSize;
還存儲在對象A3DTess3DData數組中A3DTessFaceData,每個拓撲面在精確幾何表示中一個。現在我們有了坐標和法線向量的數組,我們可以遍歷面部數據并解釋其中引用的鑲嵌。當我們遍歷面時,我們將構建一個包含位置和法線向量的單個 Qt 緩沖區,以及一個簡單的“扁平化”索引數組。
每個實例都A3DTessFaceData包含一個位標志字段,用于描述三角形數據的存儲方式。通過使用 HOOPS Exchange 生成曲面細分,我們可以合理地確保只有基本三角形存在,因此我們不必擔心在從輸入文件本身。我們通過生成曲面細分對性能造成了影響,但好處是用于讀取生成的數據的簡化代碼塊。
這是從 HOOPS Exchange 讀取三角形數據的循環。它交錯三角形頂點位置及其法線向量,這通常在可視化工作流程中使用的頂點緩沖區對象中完成。
QVector<quint32> q_indices;
QByteArray 緩沖區字節;
quint32 const stride = sizeof (float) * 6; // 3 表示頂點 + 3 表示法線
對于(自動tess_face_idx = 0u; tess_face_idx < t3dd.m_uiFaceTessSize; ++tess_face_idx ) { A3DTessFaceData const &d = t3dd.m_psFaceTessData[tess_face_idx];
  自動sz_tri_idx = 0u;
  自動ti_index = d.m_uiStartTriangulated;
  if (kA3DTessFaceDataTriangle & d.m_usUsedEntitiesFlags) {
    auto  const num_tris = d.m_puiSizesTriangulated[sz_tri_idx++];
    自動 常量pt_count = num_tris * 3; // 每個三角形 3 分
    auto  const old_sz = bufferBytes.size();
    bufferBytes.resize(bufferBytes.size() + stride * pt_count);
    auto fptr = reinterpret_cast< float * > (bufferBytes.data() + old_sz);
    對于(自動三= 0u;三<num_tris;三++){
    對于(自動垂直= 0u;垂直<3u;垂直++){
      自動 常量&normal_index =
   ;   t3dd.m_puiTriangulatedIndexes[ti_index++];
      自動 常量&coord_index =
      t3dd.m_puiTriangulatedIndexes[ti_index++];
      *fptr++ = coords[coord_index];
      *fptr++ = coords[coord_index+1];
      *fptr++ = coords[coord_index+2];
      *fptr++ = normals[normal_index];
      *fptr++ = normals[normal_index+1];
      *fptr++ = normals[normal_index+2];
    &nbsp; q_indices.push_back(q_indices.size());
      }
    }
  }
}
當這個循環結束時,我們留下一個原始緩沖區,其中包含身體中每個三角形的浮點頂點位置和法線向量。它們按順序存儲,不考慮共享索引值的可能性。這導致緩沖區可能比需要的更大,但簡化了我們呈現的代碼。
我們從 Exchange 獲得了我們需要的所有數據,所以讓我們自己清理一下。
A3DTess3DGet( nullptr , &t3dd);
A3DTessBaseGet( nullptr , &tbd);
我們必須通過創建Qt3D渲染剛剛捕獲的數據所需的原語來完成該功能。正如本教程開頭所提到的,我們不會花太多時間來描述細節,Qt3D,而是根據需要呈現代碼:
auto buf = new Qt3DCore::QBuffer();
buf->setData(bufferBytes);
自動幾何=新的QGeometry;
auto position_attribute = new QAttribute(buf,
QAttribute::defaultPositionAttributeName(), QAttribute::Float, 3, q_indices.size(), 0, stride);
幾何->addAttribute(位置屬性);
auto normal_attribute = new QAttribute( buf,
QAttribute::defaultNormalAttributeName(), QAttribute::Float, 3, q_indices.size(), sizeof (float) * 3, stride );
幾何->addAttribute( normal_attribute );
QByteArray indexBytes;
QAttribute::VertexBaseType ty;
如果(q_indices.size() < 65536) {
  // 我們可以使用 USHORT
  ty = QAttribute::UnsignedShort;
  indexBytes.resize(q_indices.size() * sizeof (quint16));
  quint16 *usptr = reinterpret_cast< quint16* > (indexBytes.data());
&nbsp; for ( int i = 0; i < int(q_indices.size()); ++i)
    *usptr++ = static_cast<;quint16>(q_indices.at(i));
}其他{
  // 使用 UINT - 不需要轉換,但讓我們確保 int 是 32 位的!
  ty = QAttribute::UnsignedInt;
  Q_ASSERT( sizeof ( int ) == sizeof (quint32));
  indexBytes.resize(q_indices.size() * sizeof (quint32));
  memcpy(indexBytes.data(), reinterpret_cast< const char * > (q_indices.data()), indexBytes.size());
}
自動*indexBuffer = new Qt3DCore::QBuffer(); indexBuffer->setData(indexBytes);
QAttribute *indexAttribute = new QAttribute(indexBuffer, ty, 1, q_indices.size());
indexAttribute-&gt;setAttributeType(QAttribute::IndexAttribute);
幾何->addAttribute(indexAttribute);
自動渲染器 =新Qt3DRender::QGeometryRenderer();
渲染器->setGeometry(幾何);
返回渲染器
完成此步驟后,您已達到一個重要里程碑。現在,您可以加載單個零件并查看它。它將以默認顏色(紅色)顯示,但應該是可見的。程序集無法正確顯示,因為我們尚未處理轉換,但加載示例文件 samples/data/prc/Flange287. prc,您應該看到以下內容:
接下來,我們將專注于使轉換正確,以便我們可以正確地可視化程序集。
第 5 步:創建轉換
現在我們在屏幕上有了一些東西,讓我們添加在世界中正確定位對象所需的代碼。完成后,我們將能夠加載和查看程序集。
在程序集文件中,程序集樹的各個節點包含本地轉換。每個變換都相對于其父級應用。這意味著,要計算每個零件的世界變換,我們必須在通向零件實例的路徑中累積每個裝配節點的變換。
根據這個描述,我們可以開始編寫 createTransform(在 Transform.cpp 中找到)的實現,如下所示:
QMatrix4x4 網絡矩陣;
對于(自動 常量ntt:路徑){
  A3DMiscTransformation *xform = getTransform(ntt);
  net_matrix *= toMatrix( xform );
}
自動xform =新Qt3DCore::QTransform();
xform->setMatrix(net_matrix);
返回xform;
這個實現完全按照我們所描述的方便的事實來描述,路徑包括指向表示項的程序集層次結構中每個對象的順序句柄列表。它使用了兩個我們仍然必須定義的函數,getTransform我們toMatrix.將在上面的匿名命名空間中實現它們createTransform.
我們getTransform.將從它的用法開始,這個函數接受一個實體句柄并返回一個A3DMiscTransformation句柄。我們必須實現這個函數來確定傳入的實體的類型,并從它返回轉換(如果存在)。
在從模型文件到表示項的路徑中,唯一可能包含轉換的對象類型是A3DAsmProductOccurrence和A3DRiRepresentationItem.我們的代碼必須處理這兩種情況。實現getTransform功能如下:
命名空間{
    A3DMiscTransformation *getTransform( A3DEntity *ntt ) {

        A3DMiscTransformation *result = nullptr ;

  &nbsp;     A3DEEntityType ntt_type = kA3DTypeUnknown;
&nbsp;       A3DEntityGetType(ntt, &ntt_type );
        if ( kA3DTypeAsmProductOccurrence == ntt_type ) {
            A3DAsmProductOccurrenceData d;
            A3D_INITIALIZE_DATA(A3DAsmProductOccurrenceData, d);
&nbsp;        ;   A3DAsmProductOccurrenceGet(ntt, &d);
  &nbsp;         結果 = d.m_pLocation ?d.m_pLocation:getTransform(d.m_pPrototype);
    &nbsp;       A3DAsmProductOccurrenceGet( nullptr , &d);
        } else  if (ntt_type > kA3DTypeRi && ntt_type <= kA3DTypeRiCoordinateSystemItem) {
  &nbsp; &nbsp;       A3DRiRepresentationItemData d;
            A3D_INITIALIZE_DATA(A3DRiRepresentationItemData, d);
          &nbsp; A3DRiRepresentationItemGet(ntt, &d);
            如果(自動ti_cs = d.m_pCoordinateSystem){
                A3DRiCoordinateSystemData cs_d;
                A3D_INITIALIZE_DATA(A3DRiCoordinateSystemData, cs_d);
    &nbsp;     &nbsp;     A3DRiCoordinateSystemGet(d.m_pCoordinateSystem, &cs_d);
                結果 = cs_d.m_pTransformation;
                A3DRiCoordinateSystemGet( nullptr , &cs_d);
            }
&nbsp;           A3DRiRepresentationItemGet( nullptr , &d);
        }
        返回結果;
    }
}
在這個實現中有兩個值得注意的地方。也許你已經發現了它們。
首先,在 if 子句中,kA3DTypeAsmProductOccurrence,您可能已經注意到選項結果的三元運算符。如果為空,getTransform則使用原型指針遞歸調用。m_pLocation這是因為裝配節點在未被覆蓋時會從其原型“繼承”位置字段。
第二個注釋在 else if 條件本身中。因為A3DEntityGetType返回提供的實體的具體類型,所以我們必須使用這里介紹的邏輯來查看實體是否是所有可能的表示項類型中的任何一種。不幸的是,它依賴于枚舉值。我愿意接受有關處理此問題的更好方法的建議(ExchangeToolkit.h有一個名為 的函數isRepresentationItem)。
有了A3DMiscTransformation句柄,我們現在準備實現 toMatrix,它必須將句柄轉換為 aQMatrix4x4. A3DMiscTranformation是具有兩種可能的具體類型的基類句柄:A3DMiscCartesianTransformation我們A3DMiscGeneralTransformation.必須處理這兩種情況。為此,請使用以下代碼在匿名命名空間的頂部創建函數:
QMatrix4x4 toMatrix(A3DMiscTransformation *xfrm){
  如果(xfrm){
    A3DEEntityType xfrm_type = kA3DTypeUnknown;
    A3DEntityGetType(xfrm, &amp;xfrm_type);
    開關(xfrm_type){
      案例kA3DTypeMiscCartesianTransformation:
        返回getMatrixFromCartesian(xfrm);
        休息;
     ; 案例kA3DTypeMiscGeneralTransformation:
        返回getMatrixFromGeneralTransformation(xfrm);
        休息;
      默認:
        throw std::invalid_argument( "意外類型。" );
        休息;
    }
  }
  返回QMatrix4x4();
}
一般變換將其矩陣表示為代表 4x4 矩陣的 16 元素雙精度數組。QMatrix4x4將這些值復制到對象中很簡單。在匿名命名空間的頂部創建以下函數來處理這種情況。
QMatrix4x4 getMatrixFromGeneralTransformation(A3DMiscGeneralTransformation *xform){
  A3DMiscGeneralTransformationData d;
  A3D_INITIALIZE_DATA(A3DMiscGeneralTransformationData, d);
 ; A3DMiscGeneralTransformationGet(xform, &d);

  自動 常數系數 = d.m_adCoeff;
  QMatrix4x4 結果;
  for (自動行 = 0u; 行 < 4u; ++row ) {
    對于(自動col = 0u;col < 4u;++col){
  &nbsp;   結果(row,col) = static_cast< float > (coeff[row + col * 4]);
    }
  }
返回結果;
處理笛卡爾變換的情況要復雜一些。我們必須讀取基本數據并執行一些元素代數來計算矩陣的值。將此代碼添加到匿名命名空間以提取笛卡爾變換數據。
QMatrix4x4 getMatrixFromCartesian(A3DMiscCartesianTransformation *xform){
  A3DMiscCartesianTransformationData d;
  A3D_INITIALIZE_DATA(A3DMiscCartesianTransformationData, d);
  A3DMiscCartesianTransformationGet(xform, &d);
  auto&nbsp; const mirror = (d.m_ucBehaviour & kA3DTransformationMirror) ?-1。: 1.;
  auto  const s = toQVector3D(d.m_sScale);
  auto  const o = toQVector3D(d.m_sOrigin);
  auto ; const x = toQVector3D(d.m_sXVector);
  auto  const y = toQVector3D(d.m_sYVector);
  auto  const z = QVector3D::crossProduct( x, y ) * mirror;
  A3DMiscCartesianTransformationGet( nullptr , &d);
  返回QMatrix4x4(
  &nbsp; xx() * sx(), yx() * sy(), zx() * sz(), ox(),
    xy() * xx(), yy() * sy(), zy() * sz(), oy(),
&nbsp;   xz() * sx(), yz() * sy(), zz() * sz(), oz(),
    0.f, 0.f, 0.f, 1.f
  );
}
此代碼使用從對象toQVector3D創建 a的函數。它在Transform.h中實現。QVector3DA3DVector3DData
添加此功能后,您將擁有一個完整的實現以供測試。運行您的應用程序并加載一個程序集文件,例如data/prc/_micro engine.prc。
第 6 步:創建材料
本教程的最后一步是創建代表我們從 Exchange 讀取的樣式數據的 Qt3D 材質。要確定零件的外觀,我們必須依賴從第 3 步的級聯屬性助手中檢索到的數據。回想一下,可見性是由通過裝配的特定路徑決定的。應以相同的方式計算應繪制的部分樣式。在createScene,我們調用函數的主體中,createMaterial并從我們的級聯屬性助手中傳遞樣式數據。

打開文件材料。cpp 這樣我們就可以開始實現該功能了。您將看到創建了默認材質,這就是所有部件都顯示為紅色的原因。傳入此函數的樣式數據對象可以通過 3 種不同的方式指定材質信息。最簡單的情況是單色。讓我們從處理那個案例開始。
更新函數如下:
Qt3DCore::QComponent *createMaterial( A3DGraphStyleData const &style_data ) {
  自動材質 =新Qt3DExtras::QDiffuseSpecularMaterial();
  材料->setDiffuse(QColor(“紅色”));
  如果(!style_data.m_bMaterial){
    auto  const a = style_data.m_bIsTransparencyDefined ?style_data.m_ucTransparency:255u;
&nbsp;   材料->setDiffuse(getColor(style_data.m_uiRgbColorIndex, a));
  }
  退回材料;
}
在這里,我們使用了一個我們仍然必須實現的getColor.函數,這個函數接受一個 RGB 顏色索引(和 alpha)并在上面的匿名命名空間中返回一個QColor.實現getColorcreateMaterial.

命名空間{
  QColor getColor(A3DUns32 const &color_idx, int  const &a) {
    如果(A3D_DEFAULT_COLOR_INDEX == color_idx){
      返回QColor( 255, 0, 0 );
    }
  &nbsp; A3DGraphRgbColorData rgb_color_data;
    A3D_INITIALIZE_DATA(A3DGraphRgbColorData, rgb_color_data);
    A3DGlobalGetGraphRgbColorData(color_idx, &rgb_color_data);
    自動 常量&r = rgb_color_data.m_dRed;
    自動 常數&g = rgb_color_data.m_dGreen;
    自動 常量&b = rgb_color_data.m_dBlue;
    返回QColor( static_cast<int>(r * 255), static_cast<int>(g * 255), static_cast<int>(b * 255), a);
  }
}
顏色數據通過整數索引存儲在 中。這個實現首先檢查索引是否等于A3D_DEFAULT_COLOR_INDEX,表示沒有分配顏色。在這種情況下,我們返回紅色,你會認為這是我最喜歡的顏色,但你錯了。從 Exchange 的雙精度定義創建QColor對象是一件簡單的事情,自然而然。
通過此實現,您會發現許多部件現在將加載并以正確的顏色顯示。
讓我們添加一個額外的案例來處理樣式數據可以采用的兩種或三種形式。使用以下 else 塊更新 createMaterial 中的 if 子句。
否則{
  A3DBool is_textuture = false ;
  A3DGlobalIsMaterialTexture(style_data.m_uiRgbColorIndex, &is_texuture);
  如果(!is_textuture){
    A3DGraphMaterialData material_data;
    A3D_INITIALIZE_DATA(A3DGraphMaterialData, material_data);
    A3DGlobalGetGraphMaterialData(style_data.m_uiRgbColorIndex, &material_data);
    auto  constambient_color = getColor(material_data.m_uiAmbient, static_cast<int>(255 * material_data.m_dAmbientAlpha));
    auto  constdiffuse_color = getColor(material_data.m_uiDiffuse, static_cast<int>(255 * material_data.m_dDiffuseAlpha));
    if (ambient_color.alpha() == 255 &&diffuse_color.alpha() == 0) {
    材料->setDiffuse(ambient_color);
    }否則 if (ambient_color.alpha() == 0 &&diffuse_color.alpha() == 255) {
        材料->setDiffuse(diffuse_color);
    }
    材質->setSpecular(getColor(material_data.m_uiSpecular,material_data.m_dSpecularAlpha));
  }
}
這可以處理稍微復雜的材質定義。處理紋理超出了本基本查看教程的范圍。我們已經處理了兩種最常見的樣式定義情況,并且我們正在返回一個合理的 Qt3D 材料。
結論

恭喜!您已經完成了涵蓋基本查看工作流程的非常詳細的教程。在此過程中,您了解了 裝配結構及其對零件顯示方式的影響。您學習了如何閱讀曲面細分的基本形式并將其解釋為一種常見的基于緩沖區的查看技術。我們通過讀取變換數據將對象放置在正確的位置和方向上,最后為每個部分應用合理的材料,使它們看起來像預期的那樣。

了解HOOPS技術詳情歡迎進入


慧都科技是中國地區的指定經銷商,提供售賣、HOOPS 60天的免費試用、中文技術支持,同時提供工業3D解決方案如果您對此感興趣,歡迎電話咨詢:023-68661681

↓ ↓ 關注“HOOPS技術”微信公眾號,了解HOOPS技術的真實應用 ↓ ↓



標簽:

本站文章除注明轉載外,均為本站原創或翻譯。歡迎任何形式的轉載,但請務必注明出處、不得修改原文相關鏈接,如果存在內容上的異議請郵件反饋至chenjj@ke049m.cn


為你推薦

  • 推薦視頻
  • 推薦活動
  • 推薦產品
  • 推薦文章
  • 慧都慧問
相關產品
控件
  • 產品功能:三維矢量圖處理
  • 源 碼:非開源
  • 產品編號:14225
  • 當前版本:V2025.5.0 [銷售以商家最新版為準,如需其他版本,請來電咨詢]
  • 開 發 商: Tech Soft 3D 正式授權
  • ">HOOPS Exchange

    更快、更準確的CAD數據轉換工具包,3D數據格式轉換首選解決方案。

    控件
  • 產品功能:三維矢量圖處理
  • 源 碼:非開源
  • 產品編號:14227
  • 當前版本:V2025.5.0 [銷售以商家最新版為準,如需其他版本,請來電咨詢]
  • 開 發 商: Tech Soft 3D 正式授權
  • ">HOOPS Communicator

    工業級高性能3D渲染引擎,專注于Web端工程圖形渲染。

    控件
  • 產品功能:三維矢量圖處理
  • 源 碼:非開源
  • 產品編號:12161
  • 當前版本:V2025.5.0 [銷售以商家最新版為準,如需其他版本,請來電咨詢]
  • 開 發 商: Tech Soft 3D 正式授權
  • ">HOOPS Visualize

    強大的工業級3D渲染引擎,為您打造出眾的工程應用程序

    控件
  • 產品功能:三維矢量圖處理
  • 源 碼:非開源
  • 產品編號:14230
  • 當前版本:V2025.5.0 [銷售以商家最新版為準,如需其他版本,請來電咨詢]
  • 開 發 商: Tech Soft 3D 正式授權
  • ">HOOPS Publish

    讓您的應用可將3D數據以原生3D PDF、HTML和標準CAD格式進行發布。

    掃碼咨詢


    添加微信 立即咨詢

    電話咨詢

    客服熱線
    023-68661681

    TOP
    利記足球官網(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) 真人boyu·博魚滾球網(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) 最大網上PM娛樂城盤口(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) 正規雷火競技官方買球(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) 雷火競技權威十大網(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) boyu·博魚信譽足球官網(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) 權威188BET足球網(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) 正規188BET足球大全(官方)網站/網頁版登錄入口/手機版登錄入口-最新版(已更新) 国产成人av大片大片在线 | 国产成人高清亚洲综合 | 91在线综合| 国产毛片农村妇女 | 国产亚洲成av人片在线观看 | 国产极品粉嫩在线观看的软件 | 成人h动漫精品一区二区无码 | 国产毛片自拍 | 国产亚洲日韩a欧美在线 | 国产综合激情在线亚洲第一页 | 国产美女精品av免费专区 | 国精产品一区一区三区糖心 | 国产自愉自愉免费精品 | 白嫩美女国产在线观看 | a级国产乱理伦片野外 | 91日韩高清在线观看播放 | 国产精品一区二区无久久久 | 国产丝袜深夜福利院在线 | 91免费国产高清在线 | 国产一区二区成人电影 | 国产尤物在线无码福利网 | 国产在线观看www污污污 | 2025久久国产福利国产秒拍 | 国产二级毛片无码精品视频 | 国产成人免费高清直播网站 | 国产成版人视频app 国产成本人免费视频 | 国产毛片一级福利 | 97人妻在线免费视频 | 国产在线拍小情侣国产拍拍偷 | 国产一区二区三区在线视頻 | 国产精品大秀视频美女嫩模 | 国产成人午夜在线观看91 | 国产一区二区三区高潮老年人 | 18禁午夜福利a级污黄刺激 | 成人国产欧美日韩在 | 国产成人精品电影在线观看 | 精品熟女少妇av免费久久 | 成人精品综合免费视频 | 国产a久久 | 91精品亚洲影视在线观看 | 成人无码在线观看 | 国产成人深夜福利短视频99 | 丰满的岳乱妇久久久 | 成人精品国产亚洲欧洲 | 国产一区二区成人av在线播放 | 91精品国产亚洲爽啪在线观看 | 国产精品日本一区二区 | 操美女黄在线网站 | 激情无码人妻又粗又大 | 丰满爆乳肉感无码一区二区三区 | 91精品人妻一区二区三区蜜 | 国产a毛片高清日日夜 | 国产精品无码影院av | 91se在线看片国产免费观看 | 国产极品粉嫩在线观看的软件 | 国产高清在线观看免费不卡 | 国产精品久在线观不 | 国产av精品看片 | 国产一区二区在线视频 | 精品人妻系列无码专区 | 国产午夜福利精品无码 | 99久re热视频这只有精品6 | 国产自偷在线拍精品热乐播av | 国产午夜理论电影在线观看 | 精品国产免费人成在线观看 | 国产成人综合欧美精品久久 | 国产福利姬喷水福利在线观看 | 国产一本到最新视频直播在线 | 国产微拍精品一区在线观看 | 成人a级毛片免费观看av网站 | 国产丝袜无码一区二区视频 | 韩国公妇里乱片a片免费观看 | av高清在线 | 丰满少妇人妻无码专区 | 91九色国产视频 | 18处破外女出血视频在线观看 | 91极品尤物在线观看 | 国产精品国产三级国v麻豆 国产精品国产三级国产aⅴ | 国产精品无码无片在线观看 | 精品国产欧美一区二区三区成人 | 成人无码v在线播放 | 国在线产香蕉精 | 成人精品丝袜在线一区 | 国产黄频在线观看高清免费 | 国产福利萌白酱精品一区 | 国产在线观看免费无码 | 国产高清精品二区 | 91精品综合网站 | 国产在线视精品在二区 | 国产欧美另类久久久精品图片 | 国产福利电影一区二区三区 | 国产精品国无码麻豆视频 | 高潮一区二区三区在线 | 国产在线播放一区 | 精品五月天六月花一区二区 | 国产一区二区三区四区五区视频 | 国产白嫩美女 | 国产午夜福利影院在线观看 | 2025日本高清中文字幕在线 | 91日韩高清在线观看播放 | 国产午夜福利在线播放87 | 国产精品秘入口麻豆古装 | 国产精品偷伦视频观看 | 国产精品无码久久久久久电影 | a级毛片无码久久精品免费 a级毛片无码免费久久真人 | 国产欧美日韩一级视频在线观看 | 91夜色精品偷窥熟女精品网站 | 国产无套视频免费看 | 高清无码一区二区在线观看吞精 | 韩国三级大全久久网站中文字幕 | 国产粉嫩精品喷潮在线观看 | 91久久精品人妻 | 91九色老熟女免费 | a级毛片全部免费高清视频 a级毛片视频国产精品视频 | 国产精品另类激情久久久免费 | 国产成人精品麻豆免费网站 | 国产欧美精品一区二 | 国产成人综合精品日韩 | 国产成人无码精品一区二区三区 | 91桃色无码国产在线观看二区 | 国产欧美产日产综合在线 | 高清一区二区亚洲 | 国产在线拍揄自揄视精品不卡 | 69精品人伦一区二区三区 | 国产精品观看视频 | 18禁裸乳无遮挡自慰免费动漫 | 精品国产美女av久久久久 | 国产成人欧美在线视频 | 超碰精品无码一区二区 | 囯产精品无码一区二区三区 | 国产成人福利免费观看 | 国产在线拍揄自揄拍无码视 | 国产精品一区久久精品 | 国产裸体裸拍在线观看 | 国产原创一区在线 | 国产精品青草久久福利不卡 | 国精品人妻无码一区二区三区性色 | 国产偷伦视频片手机在线观看 | 国产精品毛片一区 | 国产美女黄色片网站 | 91尤物在线精品 | 2025国产精品香蕉在线观看 | 国产精品日韩欧美一区二区三区 | 精品久久久久久中文人妻 | 国产一区二区久久无码精品 | 精品午夜国产人人福利一区 | 精品亚洲av无码国产一二区在线 | 91精品亚洲国 | 国产成人午夜性a一级毛片 国产成人午夜一区二区 | 91精品综合在线偷观看视频 | 国产极品精品免费视频能看 | 国产午夜视频久久 | 高清一区高清二区视频 | 国产无码高清毛片 | 国产睡熟迷奷系列网站 | 国产人妖x0x0| 国产精品无码无卡 | 国产综合免费视频 | 国产毛a片久久久久久无码 国产毛多水多高潮高清 | 91免费在线观看精品视频 | 国产在线精品亚洲第一区香蕉 | 2025午夜国产精品福利 | 国产一级无码不卡视频 | 91精品久久久久久久久无码 | 国产午夜小视频在线 | 国产va免费精品观看精品 | 国产成人高清亚洲 | 精品国产福利在线观看 | 国产精品无码毛片一区 | av一区中文字幕在线观看 | 2025最新国产三级在线看 | 国产91无码免费一区二区三区 | 国产精品人妻一码二码 | 91麻豆精品在线观看 | va一区二区三区 | 精品国产一区二区三区国产区 | 成人免费观看的a级毛片 | 国产午夜精品毛片不卡 | 高潮湖久久久久久久久 | 国产高潮流白浆视频 | 国产精品好吊一区二区三 | 国产一区欧美日韩另类 | 国产在线观看片a免费观看 国产在线观看片免费人成视频 | 成人国产精品一区在线观看播放 | 爆乳邻居肉欲中文字幕樱花动 | 国产精品成人av片免费看 | 69麻豆天美精| 国产无码免费在线观看网站 | 99久久亚洲综合精 | 99久久久国产免费观看精品 | 国产欧美日韩综合视频专 | 国产av亚洲aⅴ | 国产成人毛片视频x8 | 2025热播电视剧电影动 | av每日更新在线观看 | 国产一区二区三区男同性恋 | 精品偷拍日韩第一页 | 高潮喷水在线观看 | 国产精品视频第一页 | 国产一级内谢a级高清毛片古装 | 国产精品无码免费视频二三区 | 国产精品自拍小视频 | 国产在线五月综合婷婷 | av无码理论片在线观看免费 | 国产成人精品一区二区三区视频 | 国产精品亚洲片在线va | 国产在线欧美日韩精品一区 | 国内午夜熟妇又乱又伦 | 国产精品美女久久久久网站9 | 国内精品久久无码人妻影院 | 国产自产一区二区三区视频在线 | 成人无码片黄网站a毛片免费 | 2025年最新无码福利视频 | 国产午夜一区二区三 | 国产成人综合激情 | 成人精品天堂一区二区三区 | 国产午夜福利不卡在线观看视频 | 91久久国产综合精品 | 国产喷水大秀在线观看2025 | 国产一区二区三区久久精品 | 国产午夜无码福利在线看网站 | 国产一区二区三精品久久久无 | 国产末成年av在线播放 | 国产mv天美传媒在线视频播放 | 1000部啪啪未满十八勿入免费 | 丰满人妻一区二区三区免费视频 | 国产宾馆精品在线播放 | 97人妻天天爽夜夜爽二区 | 国产午夜福利精品视频 | 国产成人99精品免费观看 | 成人国产片免费在线观看 | 国产3p单男绿帽在线 | 国产91成人在在线播放 | 国产日韩a片 | 国产高清一区二区不卡视频 | a级毛片在线播放 | 国产成人免费无码视频在线观 | 国产午夜大地久久 | 二区三区在 | 国产成人一区av | 国产高清女同学巨大乳在线 | 成年无码按摩av片在线观看 | 国产毛片久久久久久国产毛片无码 | 国产精品久久久久久 | 国产精品日韩欧美一区二区三区 | 精品无码国产一区二区 | 18禁美女黄网站色大片免费观 | 国产亚洲欧美日韩在线观看一区 | 2025精品久久久久久中文字 | 国产三级精品三级在专区 | 国产日韩亚洲欧美aap | 成人无码精品一区二区三区亚洲区 | 国产午夜福利不卡在线观看视频 | av免费无码不卡一区二区 | 国产午夜福利院757视频 | 国产在线视频主播区 | 成人精品无码av综合 | av无码久久不卡 | av在线资源网址 | 国产福利91精品一区二区三 | 国产激情久久久久影院蜜桃aⅴ | 18禁美女黄网站色大片免费看 | 国产福利日本一区二区三区 | 国产在线观看免费完整版 | 99久久精品这里只有精品 | 成人精品一区二区三区中文 | 国产毛片毛片精品天天看 | 高清中文字幕一区 | 国产av高清一 | 国产精品极品 | 国产成人精品亚洲男人的天堂 | 精品无码成人网站久久久久久无码 | 国产成人手机在线观看 | 国产一区二区三区观看 | 国产日韩一区二区不卡视频 | 91精品欧美产品免费观看 | 国产欧美日韩视频在线观看一区二区 | 成人区人妻| 国产视频在线观看福利 | 成年女人免费视频拍拍拍 | 国产av人人夜夜澡人人爽 | 国产一级a爱做片天天视频 国产一级a大黄片毛片视频 | a级在线中文字幕在线 | 果冻传媒av在线观看入口 | 国产精品欧美一区二区三区不卡 | 国产精品va在线观看不卡顿 | 国产a级毛片久久影院 | 91精品人妻一区二区三区蜜桃 | 91成人影院在线观看 | 国产区在线视频就爱 | 潮喷大喷水系列无码精品视频 | 国产福利视频在线 | 2025国内精品久久久久精品 | 国产做爰全免费的视频避孕 | 成人综合在线视频免费观看 | 国产av成人无码精品网站 | 岛国av无码免费无禁网站 | 国产欧美大片在线 | 国产不卡精品一区二区三区 | 国产真实露脸在线观看 | 国产成人啪精品视频免费软件 | 成人亚洲国产精品无码久久一线 | 国产91在线午夜小电 | 国产精品一区二区宅男宅女 | 国产成人三级在线 | 国内福利在线视频 | 爆乳少妇无码中出在线播放 | 91看片淫黄大片一级在线观看 | av无码不卡 | 国产按摩推油一区二区三区在线 | 97色伦午夜国产亚洲精品 | 国产高清在线精品一区免费 | 极品销魂美女一区 | 国产黄网站手机在线观看 | 国产免费无码成人a片在线观看 | 国产无套高潮在线观看 | 国产末成年女av片 | 国产粉嫩嫩00在线正在播放 | 国产精品无码专区免费不卡 | 高清中文无码久久 | 国产91在线播放稀缺 | 国产精品一区二区在线观看欲臀 | 国产精品亚洲玖玖玖在线观看 | 国产福利一区二区精品免费 | 国产亚洲日韩网曝欧美精品 | av免费| 18禁免费无码无遮挡 | 国产无码电影一区二区三区 | 国产成人无码a在线观 | 2025国产精品久久精品 | 成人免费xxxxx疯狂做受 | 成人年无码av片 | 91午夜福利国产 | 国产剧情自在拍精品在线播放 | 激情欧美日韩一区二区 | 国产91丝袜在线观看 | 国产在线高清伦免费理视频 | 国产成人亚洲精品狼色在线 | 国产自美女在线精品尤物 | 精品国产福利在线观看一区 | 国产精品蜜臀在线观看 | 国产一区日韩二区欧美三区 | 国产成人免费a在线播放 | 高清一区二区三区不卡免费 | 国产a级三级三级三级 | 国产一区二区欧美日韩 | 国产一区二区三区精品美女 | 成人在线一区二区三区四区 | 精品无人乱码一区二 | 高清日韩精品一区二区三区 | 国产+日韩+另类+视频一区爱 | av在线播放不卡播放 | 国产精品毛片一区二区三区四区 | 果冻传媒麻豆系列视频 | 国产女人高潮免费视频在线观看 | 国产亚洲精品97aa片在线播放 | 高清无码专区av | 国产a级理论片免费播放 | 国产在线小视频 | 91福利国产在线观看香蕉 | 国产精品亚洲av电 | 国内盗摄视频一区二区 | 东京热久久精品视频 | 极品粉嫩虎白女毛片 | 成人无码免费午夜福利在线看片 | 成人亚洲精品久久久久软件 | 精品国产三级a∨在线观看 精品国产三级a∨在线欧美 | 99精品久久久久久人妻精品 | 高潮喷水bd在线观看 | 91亚洲自偷手机 | 动漫av纯肉无码国产av | 精品国偷自产在线视 | 国产精品丰满大屁股流白浆 | 国产精品成人免费视频网站 | 成人黄色网站·大全 | 国产福利片在线 | 国产成人涩涩涩视频在线观看 | 大尺码无码小黄片在线免费观看 | 国产欧美高清在线观看 | 国产sm调教在线观看 | 国产成人精品精品日本亚洲 | 国产麻豆视频 | 国产免费一区二区三区香蕉精 | 99久久这里只有免费精品 | 精品日韩欧美一区传媒精品网站 | 91精品手机国产在线破解版 | 国产高清一区二区在线免费观看 | 成人国产精品一级毛片视频毛片 | 国产精品一区二区免费影院 | 国产日韩一区二区三区视频免费 | 国产成人无码专区 | 精品国产免费一区二区三区五区 | 国产一区亚洲二区三区 | 丰满美女冒白浆久久久久久久 | 国产午夜精品久久久久精 | 成年人午夜视频网站 | 91麻豆精品国产91久久久久 | 国产亚洲精品久久久美女 | 国产麻豆剧果冻传媒视频免费 | 成人综合国语对白 | 国产成人欧美综合在线 | 成人国产一区二区精品 | 精品国产99久久久久久宅男i | 精品无码日韩国产不卡aⅴ 精品无码日韩国产不卡av | 国产日韩一区二区三免费高清 | 国产一级片内射视频播放 | 国产成人精品自在线拍 | 国产亚洲欧美日韩在线观看 | 国产美女爽到喷出水 | 国产精品原创中文巨作av | 国产猛男gay1069 | 国产成人午夜视频影院免费观看 | 国产美女极度色诱视频www | 成在人线v无码免费高潮喷水 | 成人欧美一区二区三区1314 | 国产一区二区精品九九 | 国产国语对白在线高清视频 | 国产无套护士丝袜在线观看 | 成人三级视频在线观看一区二区 | 国产91av视频在线播放 | 91麻豆国产激情在线 | 国外免费人妖网视频在线观看 | 国产女同调教 | 国产成人综合在线观看 | 成人国产亚洲日韩欧美亚州 | 国产在线精品一区 | 精品人妻一区二区三区日产乱码 | 国产精品原创巨作在线影院 | 国产成人无码v在线播放不卡 | 高清无码在线观看亚洲 | 国产亚洲视频在线观看网 | 国产精品白丝喷浆 | 成人精品视频一区二区三区不卡 | 国产三级国产精品国产国在线观看 | 精品久久久久久久无码人妻热 | 国产福利精品一 | 国产精品偷伦免费视频 | av无码中文一区二区三区四区 | 成人中文乱幕日产无线码 | 国产成人无码一区二区在线播放 | 2025国产精品自产拍在线观看 | 91成人免费观看网站 | 国产精品系列一区二区三区 | 国产精品中文字幕亚洲欧美 | 国产综合精品一区二区青青 | 国产免费午夜a无码v视频 | 国产麻豆成人av色影视 | 精品无码国产一区二区三区在线看 | 国产一区二区三区丝袜精品 | 国产成人女人毛片视频在线 | 国产精品毛片更新无码 | 国产三级小视频在线观看 | 国产精品乱人伦一区二区三区 | 韩国午夜aaa三级三级 | 丰满人妻被中出中文字幕 | 观看影视少妇三级 | 国产成人精品男人的天堂下载 | 国产日韩欧美一区二区三区在线 | 精品久久久久久无码 | 国产熟女高潮精选 | 动漫精品一区二区三区视频 | 丰满人妻一区二区三区四季av | 精品国产一区二区三区四区特色 | 不卡国产在线看h | 国产一级毛片aa无码 | 国产精品日日摸夜夜添夜夜添无 | 国产成人综合亚洲不在线 | 91久久| 国产精品毛片久久久久 | 国产午夜无码精品免费看 | 国产你懂的在线看网址 | 国产午夜三级一区二区三区 | 潮喷失禁大喷 | 国产成人深夜福利短视频99 | 囯产精品一品二区三区 | 国产三级不卡在 | 91丝袜国产欧美 | 高清成人爽a毛片免费 | 国产午夜av秒播在线观看 | 国产精品日本在线观看 | 91麻豆精品国产一区色欲噜噜噜 | 东京热之中文字 | 国产综合精品一区二区三区 | 国精产品砖一区二区三区糖心 | 国产日韩欧美动漫自拍区制服 | 精品一极毛片一区二区 | 国产亚洲三级在线视频 | 精品国产午夜福利在线观看蜜月 | 国产精品大秀视频美女嫩模 | 国产精品国内自产拍在线播放 | 国产成人精品免费视频大全动 | 国产在线观看免费a∨ | 成人无码区免费a片在线软件 | 国产无套内射毛片 | 国产精品午夜理论片在线播放 | 成人黄色在线观看一区 | 成人区人妻精品一区二视频 | 成人无码激情视频在线观看 | av在线不卡无码 | 国产欧美亚洲一区二区 | 丰满人妻久久中文字幕免费 | 91av视频在线播放 | 国产一区二区三区免费视频 | 丰满人妻一区二区三区46 | 国产偷窥熟女高潮精 | 成人毛片十八女人毛片视频 | 3d动漫精品啪啪一区二区中 | 精品熟女少妇av免费久久 | 国产一区二区在免费观看 | 18禁喷水流白浆自慰视噜噜噜 | 国产精品一区日韩欧美 | 91视频免费看平台安全吗 | 国产一区在线观看视频 | 高潮国产白浆抽搐福利日本 | 精品丝袜国产自在线拍 | 国精产品一区一区三区糖心 | 国产av无码片毛片一级流奶水 | 99精品国产色综合久久不卡 | 国产一区二区黄片 | 国产一区二区在线观看涩爱 | 成人国产一区二区日韩 | jk制服美女高潮中出视频 | 91精品在线国产 | 国产中文字幕在线播放 | 国产午夜福利不卡免费播放 | 99国产欧美久久久精品蜜桃 | 国产91精品免费在线观看 | 丰满人妻一区二区 | 国产一级久久毛片 | 大尺度做爰无遮挡动漫 | 国产高清在线精品一区在线 | 精品国产一区二区三区2025 | 国产一区免费看久久无码精品 | 18处破外女出血在线 | 97无码久久久久中文字幕精品 | 国产高清无码在线一区二区 | 成人午夜免费福利视频 | av五月天不卡网 | 国产69精品久久久久久99尤物 | 91精品国产乱码久 | 成人无码一区二区三区网站 | 国产成人久久精品激情 | 国产精品成人va在线观看 | 国产精品女同一区二区 | 91成人在线观看无码 | 国产亚洲人成在线v网站 | 18禁超污无遮挡无码网址 | 国产成人另类视频在线播放 | 高清少妇综合亚洲 | 国产午夜福利亚洲第一 | 国产精品成人无码视频 | heyzo专区无码综合久久 | 国产极品美女到高潮无套久久 | 国产一区二三区视频在线播放 | 91精品视频网 | 高清无码v视频日本www | 国产av巨作情欲放纵无码 | 国产av一区二区三区久久 | 国产成人亚洲精品无码h在线 | 国产亚洲欧美观看在线一区 | 国产在线观看麻豆91精 | 国产欧美日韩主播在线观看 | 国产高清无套 | av在线无修一区 | 国产午夜成人久久无码一区二区 | 成人欧美一区在线视频在线观看 | 精品无码视频一区二区三区 | 国产精品制服一区二区 | 精品无码免费在线播放 | 97无码免费人妻超级碰碰夜夜 | 高清偷自拍亚洲精品三区 | 岛国无码av潮喷 | 国产av无码专区亚洲aⅴ | 国产精品探花一区在线观看 | 福利在线一区二 | 10000部拍拍拍免费视频 | 国产成人无码a区在线观看软件 | 激情啪啪精品一区二区 | 国产成人女人视频在线观看 | 国产一区二区二区无码网站 | 国产午夜亚洲精 | 成人欧美一区二区三区a片 成人欧美一区二区三区白 成人欧美一区二区三区白人 | 国产成a人亚洲精ⅴ品无码樱花 | 69久久夜色精品国产69乱 | 91久久香蕉囯产熟女线看 | 2025国产麻豆| 国产高清av一级av毛片 | 国产成人综合亚洲av成人专区 | 成片在线看一区二区草莓 | 国产一级无码午夜大片在线观看 | 精品私库av在线 | 国产精品先锋中文在线第一页 | av蜜桃| av在线有码人| 国产真实喝醉系列39女同事 | 国产麻豆成av人片在线观看 | 国产三区不卡在线观看 | 18禁无遮挡羞羞漫画在线播放 | 国产精品视频一区二区三区经 | 国产精品盗摄!偷窥盗摄 | 国产精品无套 | 国产在线精品国自产拍影院同 | 国产在线高清伦免费理视频 | 99久久久精品免费观看国产 | 国产成人午夜视频影院免费观看 | 国产一区麻豆剧传媒果 | 国产粉嫩精品喷潮在线观看 | 国产精品日韩电影制服丝袜 | 国产一区二区精品久久不卡 | 国产狂喷潮视频免费观看 | 国产成人综合洲欧美在线 | 国产成人午夜福利在线播放 | 国产a级毛片久久久久久精品 | 国产一区专区 | 国产91精品久久久久久 | 成人国产精品一区在线观看播放 | av午夜福利一片免费看久久 | 91精品日韩 | 国产尤物精品不卡 | 国产成人无码av在线影院 | 国产在线精品一区二区高清 | av免费网站在线观看 | 精品免费av | 爆乳老师护士中 | 高潮又爽又无遮挡又免费 | 成人欧美日韩一区二区三区 | 国产丝袜在 | 99久久久国产精品免费不卡 | 18国产精品白浆在线观看免费 | 精品视频日本 | 成人午夜精品无码区久久漫画 | 激情综合欧美一区二区三区 | 国产在线观看在线播出 | 国产aⅴ精品一区二区 | 国产午夜精品久久久久九九电影 | 精品国产v无码大片在线观看 | 高潮毛片无遮挡免费高清69 | 国产精品国偷自产在线 | 精品一区二区三区波多野 | 国产电影一区二区三区 | 国产私拍一区二区三区 | 国产午夜理论片不卡在线观看 | 精品无码一区二区三区电影婷婷 | 国产中文欧美在线视频 | 2025最新国产在线人成 | 国产成人三级视频在 | 国产9191精品| 成人免费a级毛片无码片2 | 国产毛片基地 | 国产一级无码视频在线观看 | 国产精品日本一区二区在线播放 | 国产成人a码男人的天堂 | 国产亚洲精久久久久久无 | 国产精品毛片久久 | 国产成人午夜福利在线观看 | 岛国av无码免费无禁网站麦芽 | 国产精品美女久久久网站动漫 | 国产妇少水多毛多高潮a片小说 | 国产一级a爱片在线播放 | 69视频在线 | 2025国内精品久久久久精免费 | 91麻豆精品国产一区色欲噜噜噜 | 国产精品白浆无码流出 | 91麻豆精品国产91久久 | 国产精品亚韩精品无码a在线 | 精品国产一区二区三区av | 国产极品精品无码在线播出 | 国产亚洲情侣一区二区无 | 精品人妻少妇嫩草av无 | 91精品专区国产在线观看高清 | 精品一区二区三区蜜桃臀小说 | 国产人妻人伦精品1国产 | 东京热av加勒比一区二区 | 国产精品亚洲三区在线 | 99久久亚洲视频 | 国产精品视频第二区 | 国产成人精品免费视频版大全软件 | 91国自产精品中文字幕亚洲 | 国产人妻人伦精 | 99ri精品 | 国产一区二区在线午夜福利 | 精品国产成人一区二区99 | 国产精品日日爱 | 国产精品无码专区在线观看不卡 | 国产精品一区二区午夜嘿嘿嘿小说 | 97人妻天天爽夜夜爽二区 | 国产精品成久久久久三级 | 国产丝袜视频一区二区三区 | 97精品国产自产在线观看 | 成人午夜福利免费 | 国产成人av片免费 | 国产在线视精品在二区 | 91在线免费| 国产精品成人无码一区二区 | 91制服丝袜在线 | 2025经典日韩动漫在线观看 | 国产精品视频系列专区 | 国产精品一二三四区视频 | 国产片婬乱一级毛片调 | 2025国产精品自拍视频 | 国产午夜福利 | 国产在线一区二区三区旡码 | 按摩调教在线观看 | 国产午夜激无码av毛片亚洲 | 国产丝袜二区在线播放 | 国产精品任我爽爆在线播放 | 国产成a人亚洲精ⅴ品无码性色 | 91成人18禁 | 国产精品午夜高清在线观看 | 精品久久久久久久中文字幕 | 精品国产自在精品国产精 | 91久久香蕉囯产熟女线看 | 国产日韩欧美拔插一区 | 国产在线精品国自 | 精品国产a∨无码一区二区三 | 国产成人精品三级麻豆 | 国模无水印无码在线 | 国产午夜成人免费看片 | 精品无码久久91毛片视频 | 91精品丝袜国产高跟在线一区 | 国产精品无码永久免费男叫 | 国产va免费精品观看 | 东京热一精品无码一二三 | av在线网站观看网址入口 | 国模无码视频一区 | 国产精品一区二区久久精品不卡 | 国产午夜视频网站国 | 国产无套内射一级毛片农民工 | 国产内射合集颜射 | 丰满人妻熟妇av无码区a | 国产精品毛片一区二区 | 精品国产一区二区av片 | 国产欧美久久久精品影院 | 国产极品美女高潮无套在线 | 国产激情视频在线观看的 | 国产99爱在线视频免费观看 | 精品无码专区久久久水蜜桃 | 精品欧洲av无码一区二区三区 | 爆乳熟妇一区二区三区 | 国产欧美成人不卡视频 | 国产韩国视频一区二区三区 | 国产超a级动作大片中文字幕 | 国产激情久久久久久老熟女影视 | 国产精品欧美中文字幕 | 国产50岁露脸老熟女正在播放 | 国产麻烦进 | 99久久国语露脸 | 99久久精品国产成人综合 | 国产无码动漫一区二区三区 | 国产精品欧美在线另类小说 | 国产高潮流白浆视频在线观看 | heyzo在线视频| 国产寡妇乱子伦一区二区三区。 | 国产精品理论 | 国产在线91手机观 | 国产无码视频在线播放 | 国产成人麻豆精品午夜福利在 | 国产美女高潮久久久久白浆 | 国产日韩美国成人 | 18禁黄无码免费网站高潮 | 国产对白精品刺激一区二区 | 成人av在线播放免费 | 国产亚洲大尺度无码无码专线 | 国产综合在线小说 | 69国产成人综合久久精 | 91麻豆精品国产91久 | 成人午夜电影大全在线观看 | 国产高清不卡在线 | 精品蜜桃秘一区二区三区 | 国产日韩另类视频一区 | 国产精品三p一区二区视频 国产精品三级 | 国产毛片网站视频在观 | 国产一区日韩精品 | 爆乳系列无码 | 国产精品免费露脸视频 | 国产午夜精品片一区二区三区 | 国产aⅴ无码专区久久精品国产 | 激情久久一区二区三区 | 国产欧美亚洲一区二区 | 91久久久久精品无码专区 | 精品国偷自产在线视 | 国产精品成久久久久三级6二k | 国产三级国产精品国产普男人 | 国产无码高清在线观看 | 国产成人精品一区二区三区视 | 2025国产大陆天免费看黄色视频 | 国产精品成人v | 国产成人高清综合在线 | 国产成人高清精品亚洲网站 | 按摩已婚人妻中文字幕[猫腻] | 精品国产午夜理论片在线 | 91视频下载免费高清在线观看 | 韩国黄色片免费在线观看 | 国产av无码专区亚洲av毛网站 | 黑人无码在线精品视频 | 国产无套乱子伦精彩是白视频 | 91久久香蕉 | av无码专区亚洲avl在线观看 | 成人午夜在线观看国产 |