Skip to content
  • Categories
  • Recent
  • Tags
  • Popular
  • Users
  • Groups
  • Search
  • Get Qt Extensions
  • Unsolved
Collapse
Brand Logo
  1. Home
  2. Qt Development
  3. Mobile and Embedded
  4. Save Image to Android device
QtWS25 Last Chance

Save Image to Android device

Scheduled Pinned Locked Moved Solved Mobile and Embedded
androidqurlimagesaveqt 5.7
24 Posts 5 Posters 14.4k Views
  • Oldest to Newest
  • Newest to Oldest
  • Most Votes
Reply
  • Reply as topic
Log in to reply
This topic has been deleted. Only users with topic management privileges can see it.
  • eirihamE Offline
    eirihamE Offline
    eiriham
    wrote on last edited by
    #9

    Thanks!

    Well, I have not fixed step 1 yet, but I did exactly as in step 2. Again, the picture was showing only after a restart. What is the difference between the code in step 2 that you presented and the code I wrote in the original post?

    They both seem to do the same thing? Or would your code work with the mediascanner, and mine wont?

    raven-worxR 1 Reply Last reply
    0
    • eirihamE eiriham

      Thanks!

      Well, I have not fixed step 1 yet, but I did exactly as in step 2. Again, the picture was showing only after a restart. What is the difference between the code in step 2 that you presented and the code I wrote in the original post?

      They both seem to do the same thing? Or would your code work with the mediascanner, and mine wont?

      raven-worxR Offline
      raven-worxR Offline
      raven-worx
      Moderators
      wrote on last edited by
      #10

      @eiriham
      i just added my code, so we can check the error message

      Are you triggering the media scanner already?!

      --- SUPPORT REQUESTS VIA CHAT WILL BE IGNORED ---
      If you have a question please use the forum so others can benefit from the solution in the future

      1 Reply Last reply
      0
      • eirihamE Offline
        eirihamE Offline
        eiriham
        wrote on last edited by
        #11

        No, I am not triggering it! Still trying to understand how one does that.

        raven-worxR 1 Reply Last reply
        0
        • eirihamE eiriham

          No, I am not triggering it! Still trying to understand how one does that.

          raven-worxR Offline
          raven-worxR Offline
          raven-worx
          Moderators
          wrote on last edited by
          #12

          @eiriham
          because you were wondering why it still shows up after a device restart.
          There is no way around using the media scanner if you want to have it immediately listed in the media db.

          --- SUPPORT REQUESTS VIA CHAT WILL BE IGNORED ---
          If you have a question please use the forum so others can benefit from the solution in the future

          1 Reply Last reply
          0
          • eirihamE Offline
            eirihamE Offline
            eiriham
            wrote on last edited by
            #13

            Any help would be appreciated. I am looking at this post now, but it is not solved: http://lists.qt-project.org/pipermail/interest/2015-April/016263.html.

            Also, I saw your post about the Android Extra JNI, but how do I call the scanFile from there?

            This is the start of the code I am trying to write:
            @
            void SnapshotController::trigMediaScanner(QString path)
            {
            QAndroidJniObject string = QAndroidJniObject::fromString(path);

            QAndroidJniEnvironment test2;
            

            }
            @

            1 Reply Last reply
            0
            • eirihamE Offline
              eirihamE Offline
              eiriham
              wrote on last edited by
              #14

              Any help? Tried a lot of stuff, but could not make it work

              1 Reply Last reply
              0
              • M Offline
                M Offline
                mvuori
                wrote on last edited by
                #15

                This is a funny issue. People claim that calling scanFile should work, but I have seen no sign that anyone has actually managed to make it work from Qt...

                Yes, I am trying the same thing, but currently, on a personal app, just use Android app media.re.Scan: as needed.

                1 Reply Last reply
                0
                • eirihamE Offline
                  eirihamE Offline
                  eiriham
                  wrote on last edited by
                  #16

                  Using an app wont solve my problem, but thanks for the advice.

                  1 Reply Last reply
                  0
                  • QojoteQ Offline
                    QojoteQ Offline
                    Qojote
                    wrote on last edited by
                    #17

                    I had the same issue to show a picture taken by my app immediatly after saving it. And scanFile works for me.

                    I wrote a java class extending org.qtproject.qt5.android.bindings.QtActivity and declared it in the manifest file.

                    public class AndroidAccess extends org.qtproject.qt5.android.bindings.QtActivity
                    {
                      public static MediaScannerConnection s_mMs;
                    
                      @Override
                      public void onCreate(Bundle savedInstanceState)
                      {
                         ...
                         s_mMs = new MediaScannerConnection(getApplicationContext(), null);
                         s_mMs.connect();
                      }
                    
                      public static void scanForPicture(String name)
                      {
                         s_mMs.scanFile(name, null);
                      }
                    }
                    

                    and then in my Qt class i simply call:

                    QAndroidJniObject::callStaticMethod<void>("org/tud/qpcam/AndroidAccess",
                                                               "scanForPicture",
                                                               "(Ljava/lang/String;)V",
                                                               QAndroidJniObject::fromString(this->getSavePicturesDir().absolutePath()).object<jstring>());
                    

                    whereas getSavePicturesDir() returns the directory where the pictures were saved to.
                    Works like charm with Qt 5.6 and android 4.4 - 6.0.

                    1 Reply Last reply
                    1
                    • eirihamE Offline
                      eirihamE Offline
                      eiriham
                      wrote on last edited by eiriham
                      #18

                      Thanks! I am trying to get this to work now.
                      How did you register it in the manifest file?
                      And how do I know the path? I mean, how did you know you had to put in "org/tud/qpcam/AndroidAccess"?

                      I try this but the isAvailable returns false:

                      
                          qDebug()<< "Is available: " << QAndroidJniObject::isClassAvailable("MediaScanner");
                      
                          QAndroidJniObject::callStaticMethod<void>("MediaScanner",
                                                                     "scanForPicture",
                                                                     "(Ljava/lang/String;)V",
                                                                     QAndroidJniObject::fromString(this->getSavePicturesDir().absolutePath()).object<jstring>());
                      
                      D 1 Reply Last reply
                      0
                      • eirihamE eiriham

                        Thanks! I am trying to get this to work now.
                        How did you register it in the manifest file?
                        And how do I know the path? I mean, how did you know you had to put in "org/tud/qpcam/AndroidAccess"?

                        I try this but the isAvailable returns false:

                        
                            qDebug()<< "Is available: " << QAndroidJniObject::isClassAvailable("MediaScanner");
                        
                            QAndroidJniObject::callStaticMethod<void>("MediaScanner",
                                                                       "scanForPicture",
                                                                       "(Ljava/lang/String;)V",
                                                                       QAndroidJniObject::fromString(this->getSavePicturesDir().absolutePath()).object<jstring>());
                        
                        D Offline
                        D Offline
                        Devopia53
                        wrote on last edited by
                        #19

                        @eiriham

                        You can use the sendBroadcast(). MediaScanner is simple to implement only JNI without JAVA code.

                        1 Reply Last reply
                        0
                        • QojoteQ Offline
                          QojoteQ Offline
                          Qojote
                          wrote on last edited by Qojote
                          #20

                          Well, i tried a lot of ways and sendBroadcast() did not work for me. So maybe my way isnt the shortest one, but it works and thats what matters, doesnt it?

                          • Create the android files "create Android APK"

                          • set the package name, e.g. "aha.androscanner" in manifest

                          • open manifest in text mode and replace android:name="org.qtproject.qt5.android.bindings.QtActivity" with android:name="aha.androscanner.AndroScannerInJava"

                          • Create a subdirectory below the android folder, e.g. ...android/src/aha/androscanner

                          • Put in a java file, named exactly like in point 3, e.g. AndroScannerInJava.java and add it to project

                          • add "android: QT += androidextras" to your .pro file

                          • Fill in the code

                          AndroScannerInJava.java

                          package aha.androscanner;
                          
                          import android.media.MediaScannerConnection;
                          import android.os.Bundle;
                          
                          public class AndroScannerInJava extends org.qtproject.qt5.android.bindings.QtActivity
                          {
                              public static MediaScannerConnection s_mMs;
                          
                              @Override
                              public void onCreate(Bundle savedInstanceState)
                              {
                                  System.out.println("in the activity");
                          
                                  super.onCreate(savedInstanceState);
                          
                                  s_mMs = new MediaScannerConnection(getApplicationContext(), null);
                                  s_mMs.connect();
                          
                              }
                          
                              public static void scanForPicture(String name)
                              {
                                  System.out.print("scan for picture -> ");
                                  System.out.println(name);
                          
                                  s_mMs.scanFile(name, null);
                              }
                          }
                          

                          Within Qt code you call:

                          void MainWindow::saveThatPicture()
                          {
                              qDebug() << "saveThatPicture!";
                          
                              QPixmap pix(700,700);
                              pix.fill(Qt::red);
                          
                              QDir picDir = QDir(QStandardPaths::writableLocation(QStandardPaths::PicturesLocation));
                              QDir saveDir = QDir(picDir.filePath("aha"));
                              if(!saveDir.exists())
                                  picDir.mkdir("aha");
                          
                              QString filePath = saveDir.filePath("test.jpg");
                              pix.save(filePath);
                          
                          
                              QAndroidJniObject::callStaticMethod<void>("aha/androscanner/AndroScannerInJava",
                                                                         "scanForPicture",
                                                                         "(Ljava/lang/String;)V",
                                                                         QAndroidJniObject::fromString(filePath).object<jstring>());
                          }
                          

                          Manifest.xml

                          <?xml version='1.0' encoding='utf-8'?>
                          <manifest package="aha.androscanner" xmlns:android="http://schemas.android.com/apk/res/android" android:versionName="1.0" android:versionCode="1" android:installLocation="auto">
                              <application android:hardwareAccelerated="true" android:name="org.qtproject.qt5.android.bindings.QtApplication" android:label="-- %%INSERT_APP_NAME%% --">
                                  <activity android:configChanges="orientation|uiMode|screenLayout|screenSize|smallestScreenSize|layoutDirection|locale|fontScale|keyboard|keyboardHidden|navigation"
                                            android:name="aha.androscanner.AndroScannerInJava"
                                            android:label="-- %%INSERT_APP_NAME%% --"
                                            android:screenOrientation="unspecified"
                                            android:launchMode="singleTop">
                                      <intent-filter>
                                          <action android:name="android.intent.action.MAIN"/>
                                          <category android:name="android.intent.category.LAUNCHER"/>
                                      </intent-filter>
                                      <meta-data android:name="android.app.lib_name" android:value="-- %%INSERT_APP_LIB_NAME%% --"/>
                                      <meta-data android:name="android.app.qt_sources_resource_id" android:resource="@array/qt_sources"/>
                                      <meta-data android:name="android.app.repository" android:value="default"/>
                                      <meta-data android:name="android.app.qt_libs_resource_id" android:resource="@array/qt_libs"/>
                                      <meta-data android:name="android.app.bundled_libs_resource_id" android:resource="@array/bundled_libs"/>
                                      <!-- Deploy Qt libs as part of package -->
                                      <meta-data android:name="android.app.bundle_local_qt_libs" android:value="-- %%BUNDLE_LOCAL_QT_LIBS%% --"/>
                                      <meta-data android:name="android.app.bundled_in_lib_resource_id" android:resource="@array/bundled_in_lib"/>
                                      <meta-data android:name="android.app.bundled_in_assets_resource_id" android:resource="@array/bundled_in_assets"/>
                                      <!-- Run with local libs -->
                                      <meta-data android:name="android.app.use_local_qt_libs" android:value="-- %%USE_LOCAL_QT_LIBS%% --"/>
                                      <meta-data android:name="android.app.libs_prefix" android:value="/data/local/tmp/qt/"/>
                                      <meta-data android:name="android.app.load_local_libs" android:value="-- %%INSERT_LOCAL_LIBS%% --"/>
                                      <meta-data android:name="android.app.load_local_jars" android:value="-- %%INSERT_LOCAL_JARS%% --"/>
                                      <meta-data android:name="android.app.static_init_classes" android:value="-- %%INSERT_INIT_CLASSES%% --"/>
                                      <!--  Messages maps -->
                                      <meta-data android:value="@string/ministro_not_found_msg" android:name="android.app.ministro_not_found_msg"/>
                                      <meta-data android:value="@string/ministro_needed_msg" android:name="android.app.ministro_needed_msg"/>
                                      <meta-data android:value="@string/fatal_error_msg" android:name="android.app.fatal_error_msg"/>
                                      <!--  Messages maps -->
                          
                                      <!-- Splash screen -->
                                      <!--
                                      <meta-data android:name="android.app.splash_screen_drawable" android:resource="@drawable/logo"/>
                                      -->
                                      <!-- Splash screen -->
                          
                                      <!-- Background running -->
                                      <!-- Warning: changing this value to true may cause unexpected crashes if the
                                                    application still try to draw after
                                                    "applicationStateChanged(Qt::ApplicationSuspended)"
                                                    signal is sent! -->
                                      <meta-data android:name="android.app.background_running" android:value="false"/>
                                      <!-- Background running -->
                          
                                      <!-- auto screen scale factor -->
                                      <meta-data android:name="android.app.auto_screen_scale_factor" android:value="false"/>
                                      <!-- auto screen scale factor -->
                                  </activity>
                              </application>
                              <uses-sdk android:minSdkVersion="9" android:targetSdkVersion="14"/>
                              <supports-screens android:largeScreens="true" android:normalScreens="true" android:anyDensity="true" android:smallScreens="true"/>
                          
                              <!-- The following comment will be replaced upon deployment with default permissions based on the dependencies of the application.
                                   Remove the comment if you do not require these default permissions. -->
                              <!-- %%INSERT_PERMISSIONS -->
                          
                              <!-- The following comment will be replaced upon deployment with default features based on the dependencies of the application.
                                   Remove the comment if you do not require these default features. -->
                              <!-- %%INSERT_FEATURES -->
                          
                          <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
                          <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
                          
                          </manifest>
                          

                          Works fine.

                          M 1 Reply Last reply
                          3
                          • eirihamE Offline
                            eirihamE Offline
                            eiriham
                            wrote on last edited by eiriham
                            #21

                            Thanks! While following your steps I get an error. In the application output it says; "Unable to start "pdapp.mediascanner"."

                            Using logcat I get:

                            I/ActivityManager(  712): START u0 {flg=0x10000000 cmp=pdapp.mediascanner/.MediaScanner} from uid 2000 on display 0
                            W/ActivityManager(  712): Permission Denial: starting Intent { flg=0x10000000 cmp=pdapp.mediascanner/.MediaScanner } from null (pid=21117, uid=2000) not exported from uid 10104
                            I/art     (21117): System.exit called, status: 1
                            I/AndroidRuntime(21117): VM exiting with result code 1.
                            
                            

                            And also this:

                            W/PackageManager( 1369): Failure retrieving resources for pdapp.mediascanner: Resource ID #0x0
                            
                            

                            I followed your steps to the point.

                            1 Reply Last reply
                            1
                            • QojoteQ Offline
                              QojoteQ Offline
                              Qojote
                              wrote on last edited by
                              #22

                              Please make sure that you set the permissions to read and write external storage. Furthermore check consitent naming of your package and class names.
                              Which android version ar you targeting? Do you have installed the android sdk?

                              1 Reply Last reply
                              0
                              • eirihamE Offline
                                eirihamE Offline
                                eiriham
                                wrote on last edited by
                                #23

                                Thanks!

                                I had forgotten permissions for reading externally.
                                Your solution works perfectly.

                                1 Reply Last reply
                                0
                                • QojoteQ Qojote

                                  Well, i tried a lot of ways and sendBroadcast() did not work for me. So maybe my way isnt the shortest one, but it works and thats what matters, doesnt it?

                                  • Create the android files "create Android APK"

                                  • set the package name, e.g. "aha.androscanner" in manifest

                                  • open manifest in text mode and replace android:name="org.qtproject.qt5.android.bindings.QtActivity" with android:name="aha.androscanner.AndroScannerInJava"

                                  • Create a subdirectory below the android folder, e.g. ...android/src/aha/androscanner

                                  • Put in a java file, named exactly like in point 3, e.g. AndroScannerInJava.java and add it to project

                                  • add "android: QT += androidextras" to your .pro file

                                  • Fill in the code

                                  AndroScannerInJava.java

                                  package aha.androscanner;
                                  
                                  import android.media.MediaScannerConnection;
                                  import android.os.Bundle;
                                  
                                  public class AndroScannerInJava extends org.qtproject.qt5.android.bindings.QtActivity
                                  {
                                      public static MediaScannerConnection s_mMs;
                                  
                                      @Override
                                      public void onCreate(Bundle savedInstanceState)
                                      {
                                          System.out.println("in the activity");
                                  
                                          super.onCreate(savedInstanceState);
                                  
                                          s_mMs = new MediaScannerConnection(getApplicationContext(), null);
                                          s_mMs.connect();
                                  
                                      }
                                  
                                      public static void scanForPicture(String name)
                                      {
                                          System.out.print("scan for picture -> ");
                                          System.out.println(name);
                                  
                                          s_mMs.scanFile(name, null);
                                      }
                                  }
                                  

                                  Within Qt code you call:

                                  void MainWindow::saveThatPicture()
                                  {
                                      qDebug() << "saveThatPicture!";
                                  
                                      QPixmap pix(700,700);
                                      pix.fill(Qt::red);
                                  
                                      QDir picDir = QDir(QStandardPaths::writableLocation(QStandardPaths::PicturesLocation));
                                      QDir saveDir = QDir(picDir.filePath("aha"));
                                      if(!saveDir.exists())
                                          picDir.mkdir("aha");
                                  
                                      QString filePath = saveDir.filePath("test.jpg");
                                      pix.save(filePath);
                                  
                                  
                                      QAndroidJniObject::callStaticMethod<void>("aha/androscanner/AndroScannerInJava",
                                                                                 "scanForPicture",
                                                                                 "(Ljava/lang/String;)V",
                                                                                 QAndroidJniObject::fromString(filePath).object<jstring>());
                                  }
                                  

                                  Manifest.xml

                                  <?xml version='1.0' encoding='utf-8'?>
                                  <manifest package="aha.androscanner" xmlns:android="http://schemas.android.com/apk/res/android" android:versionName="1.0" android:versionCode="1" android:installLocation="auto">
                                      <application android:hardwareAccelerated="true" android:name="org.qtproject.qt5.android.bindings.QtApplication" android:label="-- %%INSERT_APP_NAME%% --">
                                          <activity android:configChanges="orientation|uiMode|screenLayout|screenSize|smallestScreenSize|layoutDirection|locale|fontScale|keyboard|keyboardHidden|navigation"
                                                    android:name="aha.androscanner.AndroScannerInJava"
                                                    android:label="-- %%INSERT_APP_NAME%% --"
                                                    android:screenOrientation="unspecified"
                                                    android:launchMode="singleTop">
                                              <intent-filter>
                                                  <action android:name="android.intent.action.MAIN"/>
                                                  <category android:name="android.intent.category.LAUNCHER"/>
                                              </intent-filter>
                                              <meta-data android:name="android.app.lib_name" android:value="-- %%INSERT_APP_LIB_NAME%% --"/>
                                              <meta-data android:name="android.app.qt_sources_resource_id" android:resource="@array/qt_sources"/>
                                              <meta-data android:name="android.app.repository" android:value="default"/>
                                              <meta-data android:name="android.app.qt_libs_resource_id" android:resource="@array/qt_libs"/>
                                              <meta-data android:name="android.app.bundled_libs_resource_id" android:resource="@array/bundled_libs"/>
                                              <!-- Deploy Qt libs as part of package -->
                                              <meta-data android:name="android.app.bundle_local_qt_libs" android:value="-- %%BUNDLE_LOCAL_QT_LIBS%% --"/>
                                              <meta-data android:name="android.app.bundled_in_lib_resource_id" android:resource="@array/bundled_in_lib"/>
                                              <meta-data android:name="android.app.bundled_in_assets_resource_id" android:resource="@array/bundled_in_assets"/>
                                              <!-- Run with local libs -->
                                              <meta-data android:name="android.app.use_local_qt_libs" android:value="-- %%USE_LOCAL_QT_LIBS%% --"/>
                                              <meta-data android:name="android.app.libs_prefix" android:value="/data/local/tmp/qt/"/>
                                              <meta-data android:name="android.app.load_local_libs" android:value="-- %%INSERT_LOCAL_LIBS%% --"/>
                                              <meta-data android:name="android.app.load_local_jars" android:value="-- %%INSERT_LOCAL_JARS%% --"/>
                                              <meta-data android:name="android.app.static_init_classes" android:value="-- %%INSERT_INIT_CLASSES%% --"/>
                                              <!--  Messages maps -->
                                              <meta-data android:value="@string/ministro_not_found_msg" android:name="android.app.ministro_not_found_msg"/>
                                              <meta-data android:value="@string/ministro_needed_msg" android:name="android.app.ministro_needed_msg"/>
                                              <meta-data android:value="@string/fatal_error_msg" android:name="android.app.fatal_error_msg"/>
                                              <!--  Messages maps -->
                                  
                                              <!-- Splash screen -->
                                              <!--
                                              <meta-data android:name="android.app.splash_screen_drawable" android:resource="@drawable/logo"/>
                                              -->
                                              <!-- Splash screen -->
                                  
                                              <!-- Background running -->
                                              <!-- Warning: changing this value to true may cause unexpected crashes if the
                                                            application still try to draw after
                                                            "applicationStateChanged(Qt::ApplicationSuspended)"
                                                            signal is sent! -->
                                              <meta-data android:name="android.app.background_running" android:value="false"/>
                                              <!-- Background running -->
                                  
                                              <!-- auto screen scale factor -->
                                              <meta-data android:name="android.app.auto_screen_scale_factor" android:value="false"/>
                                              <!-- auto screen scale factor -->
                                          </activity>
                                      </application>
                                      <uses-sdk android:minSdkVersion="9" android:targetSdkVersion="14"/>
                                      <supports-screens android:largeScreens="true" android:normalScreens="true" android:anyDensity="true" android:smallScreens="true"/>
                                  
                                      <!-- The following comment will be replaced upon deployment with default permissions based on the dependencies of the application.
                                           Remove the comment if you do not require these default permissions. -->
                                      <!-- %%INSERT_PERMISSIONS -->
                                  
                                      <!-- The following comment will be replaced upon deployment with default features based on the dependencies of the application.
                                           Remove the comment if you do not require these default features. -->
                                      <!-- %%INSERT_FEATURES -->
                                  
                                  <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
                                  <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
                                  
                                  </manifest>
                                  

                                  Works fine.

                                  M Offline
                                  M Offline
                                  mvuori
                                  wrote on last edited by
                                  #24

                                  @Qojote Thank you and congratulations for giving all the details for solving this old and apparently common problem! Without the smallest details, this kind of things are never solved.

                                  1 Reply Last reply
                                  0

                                  • Login

                                  • Login or register to search.
                                  • First post
                                    Last post
                                  0
                                  • Categories
                                  • Recent
                                  • Tags
                                  • Popular
                                  • Users
                                  • Groups
                                  • Search
                                  • Get Qt Extensions
                                  • Unsolved