diff --git a/dox/FILES_HTML.txt b/dox/FILES_HTML.txt index 6cc2a4ae41..3e7f0e7559 100644 --- a/dox/FILES_HTML.txt +++ b/dox/FILES_HTML.txt @@ -11,6 +11,9 @@ overview/overview.md ../samples/CSharp/ReadMe.md ../samples/CSharp/ReadMe_D3D.md +../samples/qt/AndroidQt/ReadMe.md +../samples/java/jniviewer/ReadMe.md + tutorial/tutorial.md technical_overview/technical_overview.md diff --git a/dox/overview/images/samples_java_android_occt.jpg b/dox/overview/images/samples_java_android_occt.jpg new file mode 100644 index 0000000000..ecba47332b Binary files /dev/null and b/dox/overview/images/samples_java_android_occt.jpg differ diff --git a/dox/overview/images/samples_qml_android_occt.jpg b/dox/overview/images/samples_qml_android_occt.jpg new file mode 100644 index 0000000000..3fbdbd34a3 Binary files /dev/null and b/dox/overview/images/samples_qml_android_occt.jpg differ diff --git a/dox/overview/overview.md b/dox/overview/overview.md index a2f39328d4..2608d9d8fe 100644 --- a/dox/overview/overview.md +++ b/dox/overview/overview.md @@ -620,3 +620,16 @@ There is also another C# example with the same functionality, which demonstrates See \subpage samples_csharp_direct3d "Readme" for details. +@subsubsection OCCT_OVW_SECTION_7_3_4 Android + +There are two samples are representing usage OCCT framework on Android mobile platform. They represent an OCCT-based 3D-viewer with CAD import support in formats BREP, STEP and IGES: jniviewer (java) and AndroidQt (qt+qml) + +jniviewer +@image html /overview/images/samples_java_android_occt.jpg +@image latex /overview/images/samples_java_android_occt.jpg +Java - See \subpage samples_java_android_occt "Readme" for details. + +AndroidQt +@image html /overview/images/samples_qml_android_occt.jpg +@image latex /overview/images/samples_qml_android_occt.jpg +Qt - See \subpage samples_qml_android_occt "Readme" for details. diff --git a/samples/qt/AndroidQt/.gitignore b/samples/qt/AndroidQt/.gitignore new file mode 100644 index 0000000000..2c0f0cc01a --- /dev/null +++ b/samples/qt/AndroidQt/.gitignore @@ -0,0 +1,2 @@ +3rdparty +occt \ No newline at end of file diff --git a/samples/qt/AndroidQt/AndroidQt.cxx b/samples/qt/AndroidQt/AndroidQt.cxx new file mode 100644 index 0000000000..927c6f01b1 --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt.cxx @@ -0,0 +1,292 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +// ======================================================================= +// function : AndroidQt +// purpose : +// ======================================================================= +AndroidQt::AndroidQt() +: myFitAllAction (false) +{ + connect (this, SIGNAL(windowChanged(QQuickWindow*)), this, SLOT(handleWindowChanged(QQuickWindow*))); + + // set shaders location variable + QByteArray aDataRoot = "/data/data/org.qtproject.example.AndroidQt/files/opencascade/shared"; + qputenv ("CSF_ShadersDirectory", aDataRoot + "/Shaders"); +} + +// ======================================================================= +// function : ReadShapeFromFile +// purpose : +// ======================================================================= +bool AndroidQt::ReadShapeFromFile (QString theFilePath) +{ + QUrl aFileUrl (theFilePath); + QString aFilePath = theFilePath; + if (aFileUrl.isLocalFile()) + { + aFilePath = QUrl (theFilePath).toLocalFile(); + } + + if (!QFile (aFilePath).exists()) + { + return false; + } + + TopoDS_Shape aShape; + BRep_Builder aBuildTool; + try + { + OCC_CATCH_SIGNALS + + if (!BRepTools::Read (aShape, aFilePath.toStdString().c_str(), aBuildTool)) + { + return false; + } + + if (!myContext.IsNull()) + { + myContext->EraseAll (Standard_False); + + Handle(AIS_Shape) aShapePrs = new AIS_Shape (aShape); + aShapePrs->SetColor (Quantity_Color(1.0, 0.73, 0.2, Quantity_TOC_RGB)); + + myContext->Display (aShapePrs, Standard_False); + myContext->SetDisplayMode (aShapePrs, AIS_Shaded, Standard_False); + } + + myMutex.lock(); + myFitAllAction = true; + myMutex.unlock(); + + if (window()) + { + window()->update(); + } + } + catch (Standard_Failure) + { + return false; + } + + return true; +} + +// ======================================================================= +// function : InitTouch +// purpose : +// ======================================================================= +void AndroidQt::InitTouch (const double theX, + const double theY) +{ + myMutex.lock(); + myTouchPoint.SetStarts (theX, theY); + myMutex.unlock(); +} + +// ======================================================================= +// function : UpdateTouch +// purpose : +// ======================================================================= +void AndroidQt::UpdateTouch (const double theX, + const double theY) +{ + myMutex.lock(); + myTouchPoint.SetEnds (theX, theY); + myMutex.unlock(); + + if (window()) + window()->update(); +} + +// ======================================================================= +// function : handleWindowChanged +// purpose : +// ======================================================================= +void AndroidQt::handleWindowChanged (QQuickWindow* theWin) +{ + if (theWin == NULL) + { + return; + } + + connect(theWin, SIGNAL(beforeSynchronizing()), this, SLOT(sync()), Qt::DirectConnection); + + theWin->setClearBeforeRendering (false); +} + +// ======================================================================= +// function : sync +// purpose : +// ======================================================================= +void AndroidQt::sync() +{ + myViewportSize = window()->size() * window()->devicePixelRatio(); + + if (myViewer.IsNull()) + { + initViewer(); + connect (window(), SIGNAL(beforeRendering()), this, SLOT(paint()), Qt::DirectConnection); + } + else + { + Handle(OpenGl_GraphicDriver) aDriver = Handle(OpenGl_GraphicDriver)::DownCast (myViewer->Driver()); + if (aDriver->getRawGlContext() != eglGetCurrentContext()) + { + initViewer(); + } + else + { + Handle(AndroidQt_Window) aWindow = Handle(AndroidQt_Window)::DownCast (myView->Window()); + aWindow->SetSize (myViewportSize.width(), myViewportSize.height()); + //myView->MustBeResized(); // can be used instead of SetWindow() when EGLsurface has not been changed + + EGLContext anEglContext = eglGetCurrentContext(); + myView->SetWindow (aWindow, (Aspect_RenderingContext )anEglContext, NULL, NULL); + } + } +} + +// ======================================================================= +// function : paint +// purpose : +// ======================================================================= +void AndroidQt::paint() +{ + myMutex.lock(); + + if (Abs(myTouchPoint.DevX()) + Abs(myTouchPoint.DevY()) > 1) + { + myView->StartRotation (myTouchPoint.X().first, myTouchPoint.Y().first); + myView->Rotation (myTouchPoint.X().second, myTouchPoint.Y().second); + + myTouchPoint.ClearDev(); + } + + if (myFitAllAction) + { + myView->FitAll(); + myFitAllAction = false; + } + + myMutex.unlock(); + + myView->Redraw(); +} + +// ======================================================================= +// function : initViewer +// purpose : +// ======================================================================= +bool AndroidQt::initViewer() +{ + EGLint aCfgId = 0; + int aWidth = 0, aHeight = 0; + EGLDisplay anEglDisplay = eglGetCurrentDisplay(); + EGLContext anEglContext = eglGetCurrentContext(); + EGLSurface anEglSurf = eglGetCurrentSurface (EGL_DRAW); + + if (anEglDisplay == EGL_NO_DISPLAY + || anEglContext == EGL_NO_CONTEXT + || anEglSurf == EGL_NO_SURFACE) + { + release(); + return false; + } + + eglQuerySurface (anEglDisplay, anEglSurf, EGL_WIDTH, &aWidth); + eglQuerySurface (anEglDisplay, anEglSurf, EGL_HEIGHT, &aHeight); + eglQuerySurface (anEglDisplay, anEglSurf, EGL_CONFIG_ID, &aCfgId); + + const EGLint aConfigAttribs[] = { EGL_CONFIG_ID, aCfgId, EGL_NONE }; + EGLint aNbConfigs = 0; + void* anEglConfig = NULL; + + if (eglChooseConfig (anEglDisplay, aConfigAttribs, &anEglConfig, 1, &aNbConfigs) != EGL_TRUE) + { + release(); + return false; + } + + if (!myViewer.IsNull()) + { + Handle(OpenGl_GraphicDriver) aDriver = Handle(OpenGl_GraphicDriver)::DownCast (myViewer->Driver()); + Handle(AndroidQt_Window) aWindow = Handle(AndroidQt_Window)::DownCast (myView->Window()); + if (!aDriver->InitEglContext (anEglDisplay, anEglContext, anEglConfig)) + { + release(); + return false; + } + + aWindow->SetSize (aWidth, aHeight); + myView->SetWindow (aWindow, (Aspect_RenderingContext )anEglContext, NULL, NULL); + } + + Handle(OpenGl_GraphicDriver) aDriver = new OpenGl_GraphicDriver (NULL, Standard_False); + aDriver->ChangeOptions().buffersNoSwap = Standard_True; + //aDriver->ChangeOptions().glslWarnings = Standard_True; // for GLSL shaders debug + + if (!aDriver->InitEglContext (anEglDisplay, anEglContext, anEglConfig)) + { + release(); + return false; + } + + // create viewer + myViewer = new V3d_Viewer (aDriver, TCollection_ExtendedString("Viewer").ToExtString(), "", 1000.0, + V3d_XposYnegZpos, AndroidQt_UserInteractionParameters::BgColor.Name(), V3d_ZBUFFER, V3d_GOURAUD, V3d_WAIT, + Standard_True, Standard_False); + myViewer->SetDefaultLights(); + myViewer->SetLightOn(); + + // create AIS context + myContext = new AIS_InteractiveContext (myViewer); + myContext->SetDisplayMode (AIS_Shaded); + + Handle(AndroidQt_Window) aWindow = new AndroidQt_Window (aWidth, aHeight); + myView = myViewer->CreateView(); + + myView->SetWindow (aWindow, (Aspect_RenderingContext )anEglContext, NULL, NULL); + myView->TriedronDisplay (Aspect_TOTP_RIGHT_LOWER, Quantity_NOC_WHITE, 0.08, V3d_ZBUFFER); + + return true; +} + +// ======================================================================= +// function : release +// purpose : +// ======================================================================= +void AndroidQt::release() +{ + myContext.Nullify(); + myView.Nullify(); + myViewer.Nullify(); +} diff --git a/samples/qt/AndroidQt/AndroidQt.h b/samples/qt/AndroidQt/AndroidQt.h new file mode 100644 index 0000000000..9bc0e57b78 --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt.h @@ -0,0 +1,88 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef ANDROIDQT_H +#define ANDROIDQT_H + +#include + +// workaround broken definitions in Qt +#define GLdouble GLdouble + +#include +#include + +#undef GLdouble + +#include +#include + +#include + +#include + +//! QML item with embedded OCCT viewer. +class AndroidQt : public QQuickItem +{ + Q_OBJECT + +public: + //! Default constructor. + AndroidQt(); + + //! Display shape from file. + Q_INVOKABLE bool ReadShapeFromFile (QString theFilePath); + + //! Handle touch event. + Q_INVOKABLE void InitTouch (const double theX, + const double theY); + + //! Handle touch event. + Q_INVOKABLE void UpdateTouch (const double theX, + const double theY); + +public slots: + + //! Handle OpenGL context creation and window resize events. + void sync(); + + //! Redraw OCCT viewer and handle pending viewer events in rendering thread. + void paint(); + +private slots: + + //! Handle window change event. + void handleWindowChanged (QQuickWindow* theWin); + +private: + + //! (Re-)initialize viewer on OpenGL context change. + bool initViewer(); + + //! Close viewer + void release(); + +private: + + Handle(V3d_Viewer) myViewer; //!< 3D viewer + Handle(V3d_View) myView; //!< 3D view + Handle(AIS_InteractiveContext) myContext; //!< interactive context + + QMutex myMutex; //!< mutex for interconnection with rendering thread + QSize myViewportSize; //!< QML item size + AndroidQt_TouchParameters myTouchPoint; //!< cached state of touch event + bool myFitAllAction; //!< queued viewer FitALL event + +}; + +#endif // ANDROIDQT_H diff --git a/samples/qt/AndroidQt/AndroidQt.pro b/samples/qt/AndroidQt/AndroidQt.pro new file mode 100644 index 0000000000..7ddc34b91d --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt.pro @@ -0,0 +1,31 @@ +TEMPLATE = app + +QT += qml quick widgets + +SOURCES += Main.cxx \ + AndroidQt.cxx \ + AndroidQt_Window.cxx \ + AndroidQt_TouchParameters.cxx + +RESOURCES += AndroidQt.qrc + +# Additional import path used to resolve QML modules in Qt Creator's code model +QML_IMPORT_PATH = + +# OCCT +include(OCCT.pri) + +# Default rules for deployment. +include(Deployment.pri) + +HEADERS += \ + AndroidQt.h \ + AndroidQt_Window.h \ + AndroidQt_TouchParameters.h \ + AndroidQt_UserInteractionParameters.h + +OTHER_FILES += \ + android/src/org/qtproject/example/AndroidQt/AndroidQt.java \ + android/AndroidManifest.xml + +ANDROID_PACKAGE_SOURCE_DIR = $$PWD/android diff --git a/samples/qt/AndroidQt/AndroidQt.qrc b/samples/qt/AndroidQt/AndroidQt.qrc new file mode 100644 index 0000000000..bcead84dcb --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt.qrc @@ -0,0 +1,8 @@ + + + res/qml/main.qml + + + res/icons/ic_action_collection.png + + diff --git a/samples/qt/AndroidQt/AndroidQt_TouchParameters.cxx b/samples/qt/AndroidQt/AndroidQt_TouchParameters.cxx new file mode 100644 index 0000000000..c844e26175 --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt_TouchParameters.cxx @@ -0,0 +1,107 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include + +// ======================================================================= +// function : AndroidQt_TouchParameters +// purpose : +// ======================================================================= +AndroidQt_TouchParameters::AndroidQt_TouchParameters() +: myXStart (0.0), + myXEnd (0.0), + myYStart (0.0), + myYEnd (0.0) +{ +} + +// ======================================================================= +// function : AndroidQt_TouchParameters +// purpose : +// ======================================================================= +AndroidQt_TouchParameters::AndroidQt_TouchParameters (const double theX, + const double theY) +: myXStart (theX), + myXEnd (theX), + myYStart (theY), + myYEnd (theY) +{ +} + +// ======================================================================= +// function : X +// purpose : +// ======================================================================= +QPair AndroidQt_TouchParameters::X() const +{ + return qMakePair(myXStart, myXEnd); +} + +// ======================================================================= +// function : DevX +// purpose : +// ======================================================================= +double AndroidQt_TouchParameters::DevX() const +{ + return myXEnd - myXStart; +} + +// ======================================================================= +// function : Y +// purpose : +// ======================================================================= +QPair AndroidQt_TouchParameters::Y() const +{ + return qMakePair(myYStart, myYEnd); +} + +// ======================================================================= +// function : DevY +// purpose : +// ======================================================================= +double AndroidQt_TouchParameters::DevY() const +{ + return myYEnd - myYStart; +} + +// ======================================================================= +// function : SetStarts +// purpose : +// ======================================================================= +void AndroidQt_TouchParameters::SetStarts (const double theXStart, + const double theYStart) +{ + myXStart = theXStart; + myYStart = theYStart; +} + +// ======================================================================= +// function : SetEnds +// purpose : +// ======================================================================= +void AndroidQt_TouchParameters::SetEnds (const double theXEnd, + const double theYEnd) +{ + myXEnd = theXEnd; + myYEnd = theYEnd; +} + +// ======================================================================= +// function : ClearDev +// purpose : +// ======================================================================= +void AndroidQt_TouchParameters::ClearDev() +{ + myXStart = myXEnd; + myYStart = myYEnd; +} diff --git a/samples/qt/AndroidQt/AndroidQt_TouchParameters.h b/samples/qt/AndroidQt/AndroidQt_TouchParameters.h new file mode 100644 index 0000000000..1debd06167 --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt_TouchParameters.h @@ -0,0 +1,60 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef ANDROIDQT_TOUCHPARAMETERS_H +#define ANDROIDQT_TOUCHPARAMETERS_H + +#include + +//! Class holding touch event state. +class AndroidQt_TouchParameters +{ + +public: + + //! Empty constructor. + AndroidQt_TouchParameters(); + + //! Default constructor. + AndroidQt_TouchParameters (const double theX, + const double theY); + + //! x coord + QPair X() const; + double DevX() const; + + //! y coord + QPair Y() const; + double DevY() const; + + //! Start coords + void SetStarts (const double theXStart, + const double theYStart); + + //! End coords + void SetEnds (const double theXEnd, + const double theYEnd); + + void ClearDev(); + +private: + + double myXStart; + double myXEnd; + + double myYStart; + double myYEnd; + +}; + +#endif // ANDROIDQT_TOUCHPARAMETERS_H diff --git a/samples/qt/AndroidQt/AndroidQt_UserInteractionParameters.h b/samples/qt/AndroidQt/AndroidQt_UserInteractionParameters.h new file mode 100644 index 0000000000..2010046397 --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt_UserInteractionParameters.h @@ -0,0 +1,28 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef ANDROIDQT_USERINTERACTIONPARAMETERS_H +#define ANDROIDQT_USERINTERACTIONPARAMETERS_H + +#include + +namespace AndroidQt_UserInteractionParameters +{ + const double RotationThreshold = 2; // [pixel] + const double PanThreshold = 4; // [pixel] + const double ZoomThreshold = 6; // [pixel] + const double ZoomRatio = 0.13; // distance ratio + const Quantity_Color BgColor = Quantity_Color(0.145, 0.145, 0.145, Quantity_TOC_RGB); // color of viewer's background +} + +#endif // USERINTERACTIONPARAMETERS_H diff --git a/samples/qt/AndroidQt/AndroidQt_Window.cxx b/samples/qt/AndroidQt/AndroidQt_Window.cxx new file mode 100644 index 0000000000..5fc200c952 --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt_Window.cxx @@ -0,0 +1,87 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include + +IMPLEMENT_STANDARD_HANDLE (AndroidQt_Window, Aspect_Window) +IMPLEMENT_STANDARD_RTTIEXT(AndroidQt_Window, Aspect_Window) + +// ======================================================================= +// function : AndroidQt_Window +// purpose : +// ======================================================================= +AndroidQt_Window::AndroidQt_Window (const int theWidth, const int theHeight, + const int theX1, const int theX2, + const int theY1, const int theY2) +: myWidth (theWidth), myHeight(theHeight), + myX1 (theX1), myX2 (theX2), + myY1 (theY1), myY2 (theY2) +{ + if (myX1 == -1) myX1 = 0; + if (myX2 == -1) myX2 = myWidth; + + if (myY1 == -1) myY1 = 0; + if (myY2 == -1) myY2 = myHeight; +} + +// ======================================================================= +// function : Position +// purpose : +// ======================================================================= +void AndroidQt_Window::Position (Standard_Integer& theX1, + Standard_Integer& theY1, + Standard_Integer& theX2, + Standard_Integer& theY2) const +{ + theX1 = myX1; + theX2 = myX2; + theY1 = myY1; + theY2 = myY2; +} + +// ======================================================================= +// function : SetPosition +// purpose : +// ======================================================================= +void AndroidQt_Window::SetPosition (const Standard_Integer theX1, + const Standard_Integer theY1, + const Standard_Integer theX2, + const Standard_Integer theY2) +{ + myX1 = theX1; + myX2 = theX2; + myY1 = theY1; + myY2 = theY2; +} + +// ======================================================================= +// function : Size +// purpose : +// ======================================================================= +void AndroidQt_Window::Size (Standard_Integer& theWidth, + Standard_Integer& theHeight) const +{ + theWidth = myWidth; + theHeight = myHeight; +} + +// ======================================================================= +// function : SetSize +// purpose : +// ======================================================================= +void AndroidQt_Window::SetSize (const Standard_Integer theWidth, + const Standard_Integer theHeight) +{ + myWidth = theWidth; + myHeight = theHeight; +} diff --git a/samples/qt/AndroidQt/AndroidQt_Window.h b/samples/qt/AndroidQt/AndroidQt_Window.h new file mode 100644 index 0000000000..575b1940ea --- /dev/null +++ b/samples/qt/AndroidQt/AndroidQt_Window.h @@ -0,0 +1,95 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef ANDROIDQT_WINDOW_H +#define ANDROIDQT_WINDOW_H + +#include + +//! This class defines dummy window. +//! The main functionality is viewport dimensions. +class AndroidQt_Window : public Aspect_Window +{ + +public: + + //! Creates a wrapper over existing Window handle + AndroidQt_Window(const int theWidth, const int theHeight, + const int theX1 = -1, const int theX2 = -1, + const int theY1 = -1, const int theY2 = -1); + + //! Returns native Window handle + virtual Aspect_Drawable NativeHandle() const { return 0; } + + //! Returns parent of native Window handle. + virtual Aspect_Drawable NativeParentHandle() const { return 0; } + + virtual void Destroy() {} + + //! Opens the window + virtual void Map() const {} + + //! Closes the window + virtual void Unmap() const {} + + //! Applies the resizing to the window + virtual Aspect_TypeOfResize DoResize() const { return Aspect_TOR_UNKNOWN; } + + //! Apply the mapping change to the window + virtual Standard_Boolean DoMapping() const { return Standard_True; } + + //! Returns True if the window is opened + virtual Standard_Boolean IsMapped() const { return Standard_True; } + + //! Returns The Window RATIO equal to the physical WIDTH/HEIGHT dimensions + virtual Quantity_Ratio Ratio() const { return 1.0; } + + //! Returns The Window POSITION in PIXEL + virtual void Position (Standard_Integer& theX1, + Standard_Integer& theY1, + Standard_Integer& theX2, + Standard_Integer& theY2) const; + + //! Set The Window POSITION in PIXEL + virtual void SetPosition (const Standard_Integer theX1, + const Standard_Integer theY1, + const Standard_Integer theX2, + const Standard_Integer theY2); + + //! Returns The Window SIZE in PIXEL + virtual void Size (Standard_Integer& theWidth, + Standard_Integer& theHeight) const; + + //! Set The Window SIZE in PIXEL + virtual void SetSize (const Standard_Integer theWidth, + const Standard_Integer theHeight); + +private: + + int myWidth; + int myHeight; + + int myX1; + int myX2; + int myY1; + int myY2; + +public: + + DEFINE_STANDARD_RTTI(AndroidQt_Window) + +}; + +DEFINE_STANDARD_HANDLE(AndroidQt_Window, Aspect_Window) + +#endif // ANDROIDQT_WINDOW_H diff --git a/samples/qt/AndroidQt/Deployment.pri b/samples/qt/AndroidQt/Deployment.pri new file mode 100644 index 0000000000..5441b63dc8 --- /dev/null +++ b/samples/qt/AndroidQt/Deployment.pri @@ -0,0 +1,27 @@ +android-no-sdk { + target.path = /data/user/qt + export(target.path) + INSTALLS += target +} else:android { + x86 { + target.path = /libs/x86 + } else: armeabi-v7a { + target.path = /libs/armeabi-v7a + } else { + target.path = /libs/armeabi + } + export(target.path) + INSTALLS += target +} else:unix { + isEmpty(target.path) { + qnx { + target.path = /tmp/$${TARGET}/bin + } else { + target.path = /opt/$${TARGET}/bin + } + export(target.path) + } + INSTALLS += target +} + +export(INSTALLS) diff --git a/samples/qt/AndroidQt/Main.cxx b/samples/qt/AndroidQt/Main.cxx new file mode 100644 index 0000000000..48102a5902 --- /dev/null +++ b/samples/qt/AndroidQt/Main.cxx @@ -0,0 +1,29 @@ +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include +#include + +#include + +int main(int argc, char *argv[]) +{ + QApplication app(argc, argv); + + qmlRegisterType("AndroidQt", 1, 0, "AndroidQt"); + + QQmlApplicationEngine engine; + engine.load (QUrl (QStringLiteral ("qrc:///res/qml/main.qml"))); + + return app.exec(); +} diff --git a/samples/qt/AndroidQt/OCCT.pri b/samples/qt/AndroidQt/OCCT.pri new file mode 100644 index 0000000000..200a4e6f3e --- /dev/null +++ b/samples/qt/AndroidQt/OCCT.pri @@ -0,0 +1,53 @@ +# +INCLUDEPATH += $$_PRO_FILE_PWD_/occt/inc $$_PRO_FILE_PWD_/3rdparty/include +DEPENDPATH += $$_PRO_FILE_PWD_/occt/inc $$_PRO_FILE_PWD_/3rdparty/include + +DEFINES += OCC_CONVERT_SIGNALS CSFDB + +QMAKE_CFLAGS += -fexceptions -Wno-ignored-qualifiers +QMAKE_CXXFLAGS += -fexceptions -Wno-ignored-qualifiers + +CONFIG(debug,debug|release) { + DEFINES += DEB +} + +occt_lib_subpath = libs/armeabi-v7a + +occt_lib_path = $$_PRO_FILE_PWD_/occt/$$occt_lib_subpath +3rdparty_lib_path = $$_PRO_FILE_PWD_/3rdparty/$$occt_lib_subpath + + +LIBS += -L$$occt_lib_path \ + -lTKernel \ + -lTKMath \ + -lTKG2d \ + -lTKG3d \ + -lTKGeomBase \ + -lTKBRep \ + -lTKGeomAlgo \ + -lTKTopAlgo \ + -lTKShHealing \ + -lTKService \ + -lTKMesh \ + -lTKHLR \ + -lTKV3d \ + -lTKOpenGl \ + -lEGL + +# IMPORTANT. load libraries in a proper order +ANDROID_EXTRA_LIBS = $$3rdparty_lib_path/libfreeimage.so \ + $$3rdparty_lib_path/libfreetype.so \ + $$occt_lib_path/libTKernel.so \ + $$occt_lib_path/libTKMath.so \ + $$occt_lib_path/libTKG2d.so \ + $$occt_lib_path/libTKG3d.so \ + $$occt_lib_path/libTKGeomBase.so \ + $$occt_lib_path/libTKBRep.so \ + $$occt_lib_path/libTKGeomAlgo.so \ + $$occt_lib_path/libTKTopAlgo.so \ + $$occt_lib_path/libTKShHealing.so \ + $$occt_lib_path/libTKService.so \ + $$occt_lib_path/libTKMesh.so \ + $$occt_lib_path/libTKHLR.so \ + $$occt_lib_path/libTKV3d.so \ + $$occt_lib_path/libTKOpenGl.so diff --git a/samples/qt/AndroidQt/ReadMe.md b/samples/qt/AndroidQt/ReadMe.md new file mode 100644 index 0000000000..080f5cbfb3 --- /dev/null +++ b/samples/qt/AndroidQt/ReadMe.md @@ -0,0 +1,54 @@ +OCCT AndroidQt sample for Android {#samples_qml_android_occt} +================== + +This sample demonstrates a simple way of using OCCT libraries in Android application written using Qt/Qml. + +The connection between Qt/Qml and OCCT (C++) level is provided by proxy library, libAndroidQt.so, written in C++. +The proxy library contains single C++ class AndroidQt encapsulating OCCT viewer and providing functionality to manipulate this viewer +and to import OCCT shapes from supported format of CAD file (BREP). + +Requirements for building sample: +* Java Development Kit 1.7 or higher +* Qt 5.3 or higher +* Android SDK from 2014.07.02 or newer +* Android NDK r9d or newer +* Apache Ant 1.9.4 or higher + +Configure project for building sample: + +In QtCreator, open AndroidQt.pro project-file: +~~~~ + File -> Open file or Project... +~~~~ + +Specify Android configurations: +~~~~ +Tools->Options->Android +~~~~ +* In JDK location specify path to Java Development Kit +* In Android SDK location specify path to Android SDK +* In Android NDK location specify path to Android NDK +* In Ant executable specify path to ant.bat file located in Apache Ant bin directory + +Make sure that "Android for armeabi-v7a" kit has been detected +~~~~ +Tools->Options->Build & Run +~~~~ + +The paths to OCCT and 3rdparty libraries are specified in "OCCT.pri" file: + +the paths to the headers: +~~~~ +INCLUDEPATH += /occt/inc /3rdparty/include +DEPENDPATH += /occt/inc /3rdparty/include +~~~~ + +the libraries location: +~~~~ +LIBS += -L/occt/libs/armeabi-v7a +~~~~ + +OCCT resources (Shaders, SHMessage, StdResource, TObj, UnitsAPI and XSMessage folder) should be copied to androidqt_dir/android/assets/opencascade/shared/ directory. Current sample requires at least Shaders folder. + +Select build configuration: Debug or Release and click Build->Build Project "AndroidQt" or (Ctrl + B). +After successful build the application can be deployed to device or emulator. diff --git a/samples/qt/AndroidQt/android/AndroidManifest.xml b/samples/qt/AndroidQt/android/AndroidManifest.xml new file mode 100644 index 0000000000..4d5be71a28 --- /dev/null +++ b/samples/qt/AndroidQt/android/AndroidManifest.xml @@ -0,0 +1,82 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/samples/qt/AndroidQt/android/assets/opencascade/shared/.gitignore b/samples/qt/AndroidQt/android/assets/opencascade/shared/.gitignore new file mode 100644 index 0000000000..51e1556e0c --- /dev/null +++ b/samples/qt/AndroidQt/android/assets/opencascade/shared/.gitignore @@ -0,0 +1 @@ +Shaders diff --git a/samples/qt/AndroidQt/android/res/drawable-hdpi/ic_launcher.png b/samples/qt/AndroidQt/android/res/drawable-hdpi/ic_launcher.png new file mode 100644 index 0000000000..d27ba82c09 Binary files /dev/null and b/samples/qt/AndroidQt/android/res/drawable-hdpi/ic_launcher.png differ diff --git a/samples/qt/AndroidQt/android/res/drawable-mdpi/ic_launcher.png b/samples/qt/AndroidQt/android/res/drawable-mdpi/ic_launcher.png new file mode 100644 index 0000000000..4b86dbf5e9 Binary files /dev/null and b/samples/qt/AndroidQt/android/res/drawable-mdpi/ic_launcher.png differ diff --git a/samples/qt/AndroidQt/android/res/drawable-xhdpi/ic_launcher.png b/samples/qt/AndroidQt/android/res/drawable-xhdpi/ic_launcher.png new file mode 100644 index 0000000000..cd79bea162 Binary files /dev/null and b/samples/qt/AndroidQt/android/res/drawable-xhdpi/ic_launcher.png differ diff --git a/samples/qt/AndroidQt/android/res/drawable-xxhdpi/ic_launcher.png b/samples/qt/AndroidQt/android/res/drawable-xxhdpi/ic_launcher.png new file mode 100644 index 0000000000..a34301f386 Binary files /dev/null and b/samples/qt/AndroidQt/android/res/drawable-xxhdpi/ic_launcher.png differ diff --git a/samples/qt/AndroidQt/android/res/values/strings.xml b/samples/qt/AndroidQt/android/res/values/strings.xml new file mode 100644 index 0000000000..4f59f34eb2 --- /dev/null +++ b/samples/qt/AndroidQt/android/res/values/strings.xml @@ -0,0 +1,4 @@ + + + AndroidQt + diff --git a/samples/qt/AndroidQt/android/src/org/qtproject/example/AndroidQt/AndroidQt.java b/samples/qt/AndroidQt/android/src/org/qtproject/example/AndroidQt/AndroidQt.java new file mode 100644 index 0000000000..d67d8fd091 --- /dev/null +++ b/samples/qt/AndroidQt/android/src/org/qtproject/example/AndroidQt/AndroidQt.java @@ -0,0 +1,99 @@ +package org.qtproject.example.AndroidQt; + +import android.content.Intent; +import android.content.res.AssetManager; +import android.os.Bundle; + +import java.util.List; +import java.io.File; +import java.io.FileNotFoundException; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; + +public class AndroidQt extends org.qtproject.qt5.android.bindings.QtActivity +{ + @Override public void onCreate(Bundle theBundle) + { + super.onCreate(theBundle); + + // copy OCCT resources + String aResFolder = getFilesDir().getAbsolutePath(); + copyAssetFolder (getAssets(), "opencascade", aResFolder + "/opencascade"); + + } + + //! Copy folder from assets + private boolean copyAssetFolder (AssetManager theAssetMgr, + String theAssetFolder, + String theFolderPathTo) + { + try + { + String[] aFiles = theAssetMgr.list (theAssetFolder); + File aFolder = new File (theFolderPathTo); + aFolder.mkdirs(); + boolean isOk = true; + for (String aFileIter : aFiles) + { + if (aFileIter.contains (".")) + { + isOk &= copyAsset (theAssetMgr, + theAssetFolder + "/" + aFileIter, + theFolderPathTo + "/" + aFileIter); + } + else + { + isOk &= copyAssetFolder (theAssetMgr, + theAssetFolder + "/" + aFileIter, + theFolderPathTo + "/" + aFileIter); + } + } + return isOk; + } + catch (Exception theError) + { + theError.printStackTrace(); + return false; + } + } + + //! Copy single file from assets + private boolean copyAsset (AssetManager theAssetMgr, + String thePathFrom, + String thePathTo) + { + try + { + InputStream aStreamIn = theAssetMgr.open (thePathFrom); + File aFileTo = new File (thePathTo); + aFileTo.createNewFile(); + OutputStream aStreamOut = new FileOutputStream (thePathTo); + copyStreamContent (aStreamIn, aStreamOut); + aStreamIn.close(); + aStreamIn = null; + aStreamOut.flush(); + aStreamOut.close(); + aStreamOut = null; + return true; + } + catch (Exception theError) + { + theError.printStackTrace(); + return false; + } + } + + //! Copy single file + private static void copyStreamContent (InputStream theIn, + OutputStream theOut) throws IOException + { + byte[] aBuffer = new byte[1024]; + int aNbReadBytes = 0; + while ((aNbReadBytes = theIn.read (aBuffer)) != -1) + { + theOut.write (aBuffer, 0, aNbReadBytes); + } + } +} diff --git a/samples/qt/AndroidQt/res/icons/ic_action_collection.png b/samples/qt/AndroidQt/res/icons/ic_action_collection.png new file mode 100644 index 0000000000..c41ca8c8b2 Binary files /dev/null and b/samples/qt/AndroidQt/res/icons/ic_action_collection.png differ diff --git a/samples/qt/AndroidQt/res/qml/main.qml b/samples/qt/AndroidQt/res/qml/main.qml new file mode 100644 index 0000000000..c80be25fcc --- /dev/null +++ b/samples/qt/AndroidQt/res/qml/main.qml @@ -0,0 +1,82 @@ +// Created: 2014-08-28 +// +// Copyright (c) 2014 OPEN CASCADE SAS +// +// This file is part of commercial software by OPEN CASCADE SAS. +// +// This software is furnished in accordance with the terms and conditions +// of the contract and with the inclusion of this copyright notice. +// This software or any other copy thereof may not be provided or otherwise +// be made available to any third party. +// No ownership title to the software is transferred hereby. +// +// OPEN CASCADE SAS makes no representation or warranties with respect to the +// performance of this software, and specifically disclaims any responsibility +// for any damages, special or consequential, connected with its use. + +import QtQuick 2.2 +import QtQuick.Window 2.1 + +import QtQuick.Dialogs 1.2 + +import QtQuick.Controls 1.2 +import QtQuick.Controls.Styles 1.2 + +import AndroidQt 1.0 + +Window { + id: root_window + visible: true + + Item { + id: root_item + anchors.fill: parent + + AndroidQt { + id: viewer + } + + MouseArea { + anchors.fill: parent + + onPressed: viewer.InitTouch(mouseX, mouseY) + onPositionChanged: viewer.UpdateTouch (mouseX, mouseY) + } + + // open button + Rectangle { + id: open_button + + // align + anchors.top: parent.top + anchors.left: parent.left + + // size + width: 200 + height: 200 + + color: "white" + + // image + Image { + anchors.horizontalCenter: parent.horizontalCenter + anchors.verticalCenter: parent.verticalCenter + + source: "qrc:///icons/res/icons/ic_action_collection.png" + } + + MouseArea { + anchors.fill: parent + onClicked: file_dialog.open() + } + } + } + + FileDialog { + id: file_dialog + title: "Please choose a file" + selectMultiple: false + nameFilters: [ "BRep files (*.brep)", "All files (*)" ] + onAccepted: viewer.ReadShapeFromFile(file_dialog.fileUrl) + } +}